Makefile 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. ifeq ($(DEBUG),1)
  2. DEBUG_INFO = $(info $1)
  3. else
  4. DEBUG_INFO = @:
  5. endif
  6. REBAR = $(CURDIR)/rebar3
  7. BUILD = $(CURDIR)/build
  8. SCRIPTS = $(CURDIR)/scripts
  9. export EMQX_RELUP ?= true
  10. export EMQX_DEFAULT_BUILDER = ghcr.io/emqx/emqx-builder/5.1-4:1.14.5-25.3.2-2-debian11
  11. export EMQX_DEFAULT_RUNNER = public.ecr.aws/debian/debian:11-slim
  12. export EMQX_REL_FORM ?= tgz
  13. export QUICER_DOWNLOAD_FROM_RELEASE = 1
  14. ifeq ($(OS),Windows_NT)
  15. export REBAR_COLOR=none
  16. FIND=/usr/bin/find
  17. else
  18. FIND=find
  19. endif
  20. # Dashboard version
  21. # from https://github.com/emqx/emqx-dashboard5
  22. export EMQX_DASHBOARD_VERSION ?= v1.5.0
  23. export EMQX_EE_DASHBOARD_VERSION ?= e1.3.0
  24. PROFILE ?= emqx
  25. REL_PROFILES := emqx emqx-enterprise
  26. PKG_PROFILES := emqx-pkg emqx-enterprise-pkg
  27. PROFILES := $(REL_PROFILES) $(PKG_PROFILES) default
  28. CT_NODE_NAME ?= 'test@127.0.0.1'
  29. CT_READABLE ?= true
  30. CT_COVER_EXPORT_PREFIX ?= $(PROFILE)
  31. export REBAR_GIT_CLONE_OPTIONS += --depth=1
  32. .PHONY: nothing
  33. nothing:
  34. @:
  35. .PHONY: default
  36. default: $(REBAR) $(PROFILE)
  37. .prepare:
  38. @$(SCRIPTS)/git-hooks-init.sh # this is no longer needed since 5.0 but we keep it anyway
  39. @$(SCRIPTS)/prepare-build-deps.sh
  40. @touch .prepare
  41. .PHONY: all
  42. all: $(REBAR) $(PROFILES)
  43. .PHONY: ensure-rebar3
  44. ensure-rebar3:
  45. @$(SCRIPTS)/ensure-rebar3.sh
  46. $(REBAR): .prepare ensure-rebar3
  47. .PHONY: ensure-hex
  48. ensure-hex:
  49. @mix local.hex --if-missing --force
  50. .PHONY: ensure-mix-rebar3
  51. ensure-mix-rebar3: $(REBAR)
  52. @mix local.rebar rebar3 $(CURDIR)/rebar3 --if-missing --force
  53. .PHONY: ensure-mix-rebar
  54. ensure-mix-rebar: $(REBAR)
  55. @mix local.rebar --if-missing --force
  56. .PHONY: mix-deps-get
  57. mix-deps-get: $(ELIXIR_COMMON_DEPS)
  58. @mix deps.get
  59. .PHONY: eunit
  60. eunit: $(REBAR) merge-config
  61. @$(REBAR) eunit --name eunit@127.0.0.1 -v -c --cover_export_name $(CT_COVER_EXPORT_PREFIX)-eunit
  62. .PHONY: proper
  63. proper: $(REBAR)
  64. @$(REBAR) proper -d test/props -c
  65. .PHONY: test-compile
  66. test-compile: $(REBAR) merge-config
  67. $(REBAR) as test compile
  68. .PHONY: $(REL_PROFILES:%=%-compile)
  69. $(REL_PROFILES:%=%-compile): $(REBAR) merge-config
  70. $(REBAR) as $(@:%-compile=%) compile
  71. .PHONY: ct
  72. ct: $(REBAR) merge-config
  73. @$(REBAR) ct --name $(CT_NODE_NAME) -c -v --cover_export_name $(CT_COVER_EXPORT_PREFIX)-ct
  74. ## only check bpapi for enterprise profile because it's a super-set.
  75. .PHONY: static_checks
  76. static_checks:
  77. @$(REBAR) as check do xref, dialyzer
  78. @if [ "$${PROFILE}" = 'emqx-enterprise' ]; then $(REBAR) ct --suite apps/emqx/test/emqx_static_checks --readable $(CT_READABLE); fi
  79. ./scripts/check-i18n-style.sh
  80. ./scripts/check_missing_reboot_apps.exs
  81. # Allow user-set CASES environment variable
  82. ifneq ($(CASES),)
  83. CASES_ARG := --case $(CASES)
  84. endif
  85. # Allow user-set GROUPS environment variable
  86. ifneq ($(GROUPS),)
  87. GROUPS_ARG := --groups $(GROUPS)
  88. endif
  89. ## example:
  90. ## env SUITES=apps/appname/test/test_SUITE.erl CASES=t_foo make apps/appname-ct
  91. define gen-app-ct-target
  92. $1-ct: $(REBAR) merge-config clean-test-cluster-config
  93. $(eval SUITES := $(shell $(SCRIPTS)/find-suites.sh $1))
  94. ifneq ($(SUITES),)
  95. @$(REBAR) ct -c -v \
  96. --readable=$(CT_READABLE) \
  97. --name $(CT_NODE_NAME) \
  98. --cover_export_name $(CT_COVER_EXPORT_PREFIX)-$(subst /,-,$1) \
  99. --suite $(SUITES) \
  100. $(GROUPS_ARG) \
  101. $(CASES_ARG)
  102. else
  103. @echo 'No suites found for $1'
  104. endif
  105. endef
  106. ifneq ($(filter %-ct,$(MAKECMDGOALS)),)
  107. app_to_test := $(patsubst %-ct,%,$(filter %-ct,$(MAKECMDGOALS)))
  108. $(call DEBUG_INFO,app_to_test $(app_to_test))
  109. $(eval $(call gen-app-ct-target,$(app_to_test)))
  110. endif
  111. ## apps/name-prop targets
  112. define gen-app-prop-target
  113. $1-prop:
  114. $(REBAR) proper -d test/props -v -m $(shell $(SCRIPTS)/find-props.sh $1)
  115. endef
  116. ifneq ($(filter %-prop,$(MAKECMDGOALS)),)
  117. app_to_test := $(patsubst %-prop,%,$(filter %-prop,$(MAKECMDGOALS)))
  118. $(call DEBUG_INFO,app_to_test $(app_to_test))
  119. $(eval $(call gen-app-prop-target,$(app_to_test)))
  120. endif
  121. .PHONY: ct-suite
  122. ct-suite: $(REBAR) merge-config clean-test-cluster-config
  123. ifneq ($(TESTCASE),)
  124. ifneq ($(GROUP),)
  125. $(REBAR) ct -v --readable=$(CT_READABLE) --name $(CT_NODE_NAME) --suite $(SUITE) --case $(TESTCASE) --group $(GROUP)
  126. else
  127. $(REBAR) ct -v --readable=$(CT_READABLE) --name $(CT_NODE_NAME) --suite $(SUITE) --case $(TESTCASE)
  128. endif
  129. else ifneq ($(GROUP),)
  130. $(REBAR) ct -v --readable=$(CT_READABLE) --name $(CT_NODE_NAME) --suite $(SUITE) --group $(GROUP)
  131. else
  132. $(REBAR) ct -v --readable=$(CT_READABLE) --name $(CT_NODE_NAME) --suite $(SUITE)
  133. endif
  134. .PHONY: cover
  135. cover: $(REBAR)
  136. @ENABLE_COVER_COMPILE=1 $(REBAR) as test cover
  137. .PHONY: coveralls
  138. coveralls: $(REBAR)
  139. @ENABLE_COVER_COMPILE=1 $(REBAR) as test coveralls send
  140. COMMON_DEPS := $(REBAR)
  141. .PHONY: $(REL_PROFILES)
  142. $(REL_PROFILES:%=%): $(COMMON_DEPS)
  143. @$(BUILD) $(@) rel
  144. .PHONY: compile $(PROFILES:%=compile-%)
  145. compile: $(PROFILES:%=compile-%)
  146. $(PROFILES:%=compile-%):
  147. @$(BUILD) $(@:compile-%=%) apps
  148. .PHONY: $(PROFILES:%=compile-%-elixir)
  149. $(PROFILES:%=compile-%-elixir):
  150. @env IS_ELIXIR=yes $(BUILD) $(@:compile-%-elixir=%) apps
  151. ## Not calling rebar3 clean because
  152. ## 1. rebar3 clean relies on rebar3, meaning it reads config, fetches dependencies etc.
  153. ## 2. it's slow
  154. ## NOTE: this does not force rebar3 to fetch new version dependencies
  155. ## make clean-all to delete all fetched dependencies for a fresh start-over
  156. .PHONY: clean $(PROFILES:%=clean-%)
  157. clean: $(PROFILES:%=clean-%)
  158. $(PROFILES:%=clean-%):
  159. @if [ -d _build/$(@:clean-%=%) ]; then \
  160. rm -f rebar.lock; \
  161. rm -rf _build/$(@:clean-%=%)/rel; \
  162. $(FIND) _build/$(@:clean-%=%) -name '*.beam' -o -name '*.so' -o -name '*.app' -o -name '*.appup' -o -name '*.o' -o -name '*.d' -type f | xargs rm -f; \
  163. $(FIND) _build/$(@:clean-%=%) -type l -delete; \
  164. fi
  165. .PHONY: clean-all
  166. clean-all:
  167. @rm -f rebar.lock
  168. @rm -rf deps
  169. @rm -rf _build
  170. @rm -f emqx_dialyzer_*_plt
  171. .PHONY: deps-all
  172. deps-all: $(REBAR) $(PROFILES:%=deps-%)
  173. @make clean # ensure clean at the end
  174. ## deps-<profile> is used in CI scripts to download deps and the
  175. ## share downloads between CI steps and/or copied into containers
  176. ## which may not have the right credentials
  177. .PHONY: $(PROFILES:%=deps-%)
  178. $(PROFILES:%=deps-%): $(COMMON_DEPS)
  179. @$(SCRIPTS)/pre-compile.sh $(@:deps-%=%)
  180. @$(REBAR) as $(@:deps-%=%) get-deps
  181. @rm -f rebar.lock
  182. .PHONY: xref
  183. xref: $(REBAR)
  184. @$(REBAR) as check xref
  185. .PHONY: dialyzer
  186. dialyzer: $(REBAR)
  187. @$(REBAR) as check dialyzer
  188. ## rel target is to create release package without relup
  189. .PHONY: $(REL_PROFILES:%=%-rel) $(PKG_PROFILES:%=%-rel)
  190. $(REL_PROFILES:%=%-rel) $(PKG_PROFILES:%=%-rel): $(COMMON_DEPS)
  191. @$(BUILD) $(subst -rel,,$(@)) rel
  192. ## download relup base packages
  193. .PHONY: $(REL_PROFILES:%=%-relup-downloads)
  194. define download-relup-packages
  195. $1-relup-downloads:
  196. @if [ "$${EMQX_RELUP}" = "true" ]; then $(SCRIPTS)/relup-build/download-base-packages.sh $1; fi
  197. endef
  198. ALL_ZIPS = $(REL_PROFILES)
  199. $(foreach zt,$(ALL_ZIPS),$(eval $(call download-relup-packages,$(zt))))
  200. ## relup target is to create relup instructions
  201. .PHONY: $(REL_PROFILES:%=%-relup)
  202. define gen-relup-target
  203. $1-relup: $1-relup-downloads $(COMMON_DEPS)
  204. @$(BUILD) $1 relup
  205. endef
  206. ALL_TGZS = $(REL_PROFILES)
  207. $(foreach zt,$(ALL_TGZS),$(eval $(call gen-relup-target,$(zt))))
  208. ## tgz target is to create a release package .tar.gz with relup
  209. .PHONY: $(REL_PROFILES:%=%-tgz)
  210. define gen-tgz-target
  211. $1-tgz: $1-relup
  212. @$(BUILD) $1 tgz
  213. endef
  214. ALL_TGZS = $(REL_PROFILES)
  215. $(foreach zt,$(ALL_TGZS),$(eval $(call gen-tgz-target,$(zt))))
  216. ## A pkg target depend on a regular release
  217. .PHONY: $(PKG_PROFILES)
  218. define gen-pkg-target
  219. $1: $(COMMON_DEPS)
  220. @$(BUILD) $1 pkg
  221. endef
  222. $(foreach pt,$(PKG_PROFILES),$(eval $(call gen-pkg-target,$(pt))))
  223. .PHONY: run
  224. run: compile-$(PROFILE) quickrun
  225. .PHONY: quickrun
  226. quickrun:
  227. ./dev -p $(PROFILE)
  228. ## Take the currently set PROFILE
  229. docker:
  230. @$(BUILD) $(PROFILE) docker
  231. ## docker target is to create docker instructions
  232. .PHONY: $(REL_PROFILES:%=%-docker) $(REL_PROFILES:%=%-elixir-docker)
  233. define gen-docker-target
  234. $1-docker: $(COMMON_DEPS)
  235. @$(BUILD) $1 docker
  236. endef
  237. ALL_DOCKERS = $(REL_PROFILES) $(REL_PROFILES:%=%-elixir)
  238. $(foreach zt,$(ALL_DOCKERS),$(eval $(call gen-docker-target,$(zt))))
  239. .PHONY:
  240. merge-config:
  241. @$(SCRIPTS)/merge-config.escript
  242. ## elixir target is to create release packages using Elixir's Mix
  243. .PHONY: $(REL_PROFILES:%=%-elixir) $(PKG_PROFILES:%=%-elixir)
  244. $(REL_PROFILES:%=%-elixir) $(PKG_PROFILES:%=%-elixir): $(COMMON_DEPS)
  245. @env IS_ELIXIR=yes $(BUILD) $(subst -elixir,,$(@)) elixir
  246. .PHONY: $(REL_PROFILES:%=%-elixir-pkg)
  247. define gen-elixir-pkg-target
  248. # the Elixir places the tar in a different path than Rebar3
  249. $1-elixir-pkg: $(COMMON_DEPS)
  250. @env TAR_PKG_DIR=_build/$1-pkg \
  251. IS_ELIXIR=yes \
  252. $(BUILD) $1-pkg pkg
  253. endef
  254. $(foreach pt,$(REL_PROFILES),$(eval $(call gen-elixir-pkg-target,$(pt))))
  255. .PHONY: $(REL_PROFILES:%=%-elixir-tgz)
  256. define gen-elixir-tgz-target
  257. $1-elixir-tgz: $(COMMON_DEPS)
  258. @env IS_ELIXIR=yes $(BUILD) $1 tgz
  259. endef
  260. ALL_ELIXIR_TGZS = $(REL_PROFILES)
  261. $(foreach tt,$(ALL_ELIXIR_TGZS),$(eval $(call gen-elixir-tgz-target,$(tt))))
  262. .PHONY: fmt
  263. fmt: $(REBAR)
  264. @$(SCRIPTS)/erlfmt -w '{apps,lib-ee}/*/{src,include,priv,test,integration_test}/**/*.{erl,hrl,app.src,eterm}'
  265. @$(SCRIPTS)/erlfmt -w 'rebar.config.erl'
  266. @mix format
  267. .PHONY: clean-test-cluster-config
  268. clean-test-cluster-config:
  269. @rm -f apps/emqx_conf/data/configs/cluster.hocon || true
  270. .PHONY: spellcheck
  271. spellcheck:
  272. ./scripts/spellcheck/spellcheck.sh _build/docgen/$(PROFILE)/schema-en.json