Makefile 9.4 KB

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