mix.exs 38 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415
  1. defmodule EMQXUmbrella.MixProject do
  2. use Mix.Project
  3. @moduledoc """
  4. The purpose of this file is to configure the release of EMQX under
  5. Mix. Since EMQX uses its own configuration conventions and startup
  6. procedures, one cannot simply use `iex -S mix`. Instead, it's
  7. recommended to build and use the release.
  8. ## Profiles
  9. To control the profile and edition to build, we case split on the
  10. MIX_ENV value.
  11. The following profiles are valid:
  12. * `emqx`
  13. * `emqx-enterprise`
  14. * `emqx-pkg`
  15. * `emqx-enterprise-pkg`
  16. * `dev` -> same as `emqx`, for convenience
  17. ## Release Environment Variables
  18. The release build is controlled by a few environment variables.
  19. * `ELIXIR_MAKE_TAR` - If set to `yes`, will produce a `.tar.gz`
  20. tarball along with the release.
  21. """
  22. # TODO: remove once we switch to the new mix build
  23. def new_mix_build?() do
  24. System.get_env("NEW_MIX_BUILD") == "1"
  25. end
  26. def project() do
  27. profile_info = check_profile!()
  28. version = pkg_vsn()
  29. if new_mix_build?() do
  30. [
  31. apps_path: "apps",
  32. erlc_options: erlc_options(profile_info, version),
  33. version: version,
  34. deps: deps(profile_info, version),
  35. releases: releases(),
  36. aliases: aliases()
  37. ]
  38. else
  39. # TODO: this check and clause will be removed when we switch to using mix as the
  40. # manager for all umbrella apps.
  41. [
  42. app: :emqx_mix,
  43. erlc_options: erlc_options(profile_info, version),
  44. version: version,
  45. deps: deps(profile_info, version),
  46. releases: releases(),
  47. aliases: aliases()
  48. ]
  49. end
  50. end
  51. @doc """
  52. Please try to add dependencies that used by a single umbrella application in the
  53. application's own `mix.exs` file, if possible. If it's shared by more than one
  54. application, or if the dependency requires an `override: true` option, add a new clause
  55. to `common_dep/1` so that we centralize versions in this root `mix.exs` file as much as
  56. possible.
  57. Here, transitive dependencies from our app dependencies should be placed when there's a
  58. need to override them. For example, since `jsone` is a dependency to `rocketmq` and to
  59. `erlavro`, which are both dependencies and not umbrella apps, we need to add the
  60. override here. Also, there are cases where adding `override: true` to the umbrella
  61. application dependency simply won't satisfy mix. In such cases, it's fine to add it
  62. here.
  63. """
  64. def deps(profile_info, version) do
  65. # we need several overrides here because dependencies specify
  66. # other exact versions, and not ranges.
  67. if new_mix_build?() do
  68. new_deps()
  69. else
  70. old_deps(profile_info, version)
  71. end
  72. end
  73. def new_deps() do
  74. common_deps() ++
  75. quicer_dep() ++
  76. jq_dep() ++
  77. extra_release_apps() ++
  78. overridden_deps()
  79. end
  80. ## TODO: this should be removed once we migrate the release build to mix
  81. defp old_deps(profile_info, version) do
  82. rebar3_umbrella_apps = emqx_apps(profile_info, version) ++ enterprise_deps(profile_info)
  83. common_deps() ++
  84. extra_release_apps() ++
  85. overridden_deps() ++
  86. jq_dep() ++
  87. quicer_dep() ++ rebar3_umbrella_apps
  88. end
  89. def overridden_deps() do
  90. [
  91. common_dep(:lc),
  92. common_dep(:typerefl),
  93. common_dep(:ehttpc),
  94. common_dep(:gproc),
  95. common_dep(:jiffy),
  96. common_dep(:cowboy),
  97. common_dep(:esockd),
  98. common_dep(:rocksdb),
  99. common_dep(:ekka),
  100. common_dep(:gen_rpc),
  101. common_dep(:grpc),
  102. common_dep(:minirest),
  103. common_dep(:ecpool),
  104. common_dep(:replayq),
  105. # maybe forbid to fetch quicer
  106. common_dep(:emqtt),
  107. common_dep(:rulesql),
  108. common_dep(:telemetry),
  109. # in conflict by emqtt and hocon
  110. common_dep(:getopt),
  111. common_dep(:snabbkaffe),
  112. common_dep(:hocon),
  113. common_dep(:emqx_http_lib),
  114. common_dep(:jose),
  115. # in conflict by ehttpc and emqtt
  116. common_dep(:gun),
  117. # in conflict by emqx_connector and system_monitor
  118. common_dep(:epgsql),
  119. # in conflict by emqx and observer_cli
  120. {:recon, github: "ferd/recon", tag: "2.5.1", override: true},
  121. common_dep(:jsx),
  122. # in conflict by erlavro and rocketmq
  123. common_dep(:jsone),
  124. # dependencies of dependencies; we choose specific refs to match
  125. # what rebar3 chooses.
  126. # in conflict by gun and emqtt
  127. common_dep(:cowlib),
  128. # in conflict by cowboy_swagger and cowboy
  129. common_dep(:ranch),
  130. # in conflict by grpc and eetcd
  131. common_dep(:gpb),
  132. {:hackney, github: "emqx/hackney", tag: "1.18.1-1", override: true},
  133. # set by hackney (dependency)
  134. {:ssl_verify_fun, "1.1.7", override: true},
  135. common_dep(:bcrypt),
  136. common_dep(:uuid),
  137. {:quickrand, github: "okeuday/quickrand", tag: "v2.0.6", override: true},
  138. common_dep(:ra),
  139. {:mimerl, "1.2.0", override: true},
  140. common_dep(:sasl_auth),
  141. # avlizer currently uses older :erlavro version
  142. common_dep(:erlavro),
  143. # in conflict by erlavro
  144. common_dep(:snappyer),
  145. common_dep(:crc32cer)
  146. ]
  147. end
  148. def extra_release_apps() do
  149. [
  150. common_dep(:redbug),
  151. common_dep(:observer_cli),
  152. common_dep(:system_monitor)
  153. ]
  154. end
  155. def common_dep(dep_name, overrides) do
  156. case common_dep(dep_name) do
  157. {^dep_name, opts} ->
  158. {dep_name, Keyword.merge(opts, overrides)}
  159. {^dep_name, tag, opts} when is_binary(tag) ->
  160. {dep_name, tag, Keyword.merge(opts, overrides)}
  161. end
  162. end
  163. def common_dep(:ekka), do: {:ekka, github: "emqx/ekka", tag: "0.19.7", override: true}
  164. def common_dep(:esockd), do: {:esockd, github: "emqx/esockd", tag: "5.13.0", override: true}
  165. def common_dep(:gproc), do: {:gproc, github: "emqx/gproc", tag: "0.9.0.1", override: true}
  166. def common_dep(:hocon), do: {:hocon, github: "emqx/hocon", tag: "0.43.4", override: true}
  167. def common_dep(:lc), do: {:lc, github: "emqx/lc", tag: "0.3.3", override: true}
  168. # in conflict by ehttpc and emqtt
  169. def common_dep(:gun), do: {:gun, github: "emqx/gun", tag: "1.3.11", override: true}
  170. # in conflict by cowboy_swagger and cowboy
  171. def common_dep(:ranch), do: {:ranch, github: "emqx/ranch", tag: "1.8.1-emqx", override: true}
  172. def common_dep(:ehttpc),
  173. do: {:ehttpc, github: "emqx/ehttpc", tag: "0.6.0", override: true}
  174. def common_dep(:jiffy), do: {:jiffy, github: "emqx/jiffy", tag: "1.0.6", override: true}
  175. def common_dep(:grpc),
  176. do:
  177. {:grpc,
  178. github: "emqx/grpc-erl", tag: "0.6.12", override: true, system_env: emqx_app_system_env()}
  179. def common_dep(:cowboy), do: {:cowboy, github: "emqx/cowboy", tag: "2.9.2", override: true}
  180. def common_dep(:jsone), do: {:jsone, github: "emqx/jsone", tag: "1.7.1", override: true}
  181. def common_dep(:ecpool), do: {:ecpool, github: "emqx/ecpool", tag: "0.6.1", override: true}
  182. def common_dep(:replayq), do: {:replayq, github: "emqx/replayq", tag: "0.3.10", override: true}
  183. def common_dep(:jsx), do: {:jsx, github: "talentdeficit/jsx", tag: "v3.1.0", override: true}
  184. # in conflict by emqtt and hocon
  185. def common_dep(:getopt), do: {:getopt, "1.0.2", override: true}
  186. def common_dep(:telemetry), do: {:telemetry, "1.1.0", override: true}
  187. # in conflict by grpc and eetcd
  188. def common_dep(:gpb), do: {:gpb, "4.19.9", override: true, runtime: false}
  189. def common_dep(:ra), do: {:ra, "2.15.0", override: true}
  190. # in conflict by emqx_connector and system_monitor
  191. def common_dep(:epgsql), do: {:epgsql, github: "emqx/epgsql", tag: "4.7.1.3", override: true}
  192. def common_dep(:sasl_auth), do: {:sasl_auth, "2.3.3", override: true}
  193. def common_dep(:gen_rpc), do: {:gen_rpc, github: "emqx/gen_rpc", tag: "3.4.1", override: true}
  194. def common_dep(:system_monitor),
  195. do: {:system_monitor, github: "ieQu1/system_monitor", tag: "3.0.5"}
  196. def common_dep(:uuid), do: {:uuid, github: "okeuday/uuid", tag: "v2.0.6", override: true}
  197. def common_dep(:redbug), do: {:redbug, github: "emqx/redbug", tag: "2.0.10"}
  198. def common_dep(:observer_cli), do: {:observer_cli, "1.7.5"}
  199. def common_dep(:jose),
  200. do: {:jose, github: "potatosalad/erlang-jose", tag: "1.11.2", override: true}
  201. def common_dep(:rulesql), do: {:rulesql, github: "emqx/rulesql", tag: "0.2.1"}
  202. def common_dep(:bcrypt),
  203. do: {:bcrypt, github: "emqx/erlang-bcrypt", tag: "0.6.2", override: true}
  204. def common_dep(:minirest),
  205. do: {:minirest, github: "emqx/minirest", tag: "1.4.4", override: true}
  206. # maybe forbid to fetch quicer
  207. def common_dep(:emqtt),
  208. do:
  209. {:emqtt,
  210. github: "emqx/emqtt", tag: "1.13.5", override: true, system_env: maybe_no_quic_env()}
  211. def common_dep(:typerefl),
  212. do: {:typerefl, github: "ieQu1/typerefl", tag: "0.9.1", override: true}
  213. def common_dep(:rocksdb),
  214. do: {:rocksdb, github: "emqx/erlang-rocksdb", tag: "1.8.0-emqx-6", override: true}
  215. def common_dep(:emqx_http_lib),
  216. do: {:emqx_http_lib, github: "emqx/emqx_http_lib", tag: "0.5.3", override: true}
  217. def common_dep(:cowlib),
  218. do:
  219. {:cowlib,
  220. github: "ninenines/cowlib", ref: "c6553f8308a2ca5dcd69d845f0a7d098c40c3363", override: true}
  221. def common_dep(:snabbkaffe),
  222. do: {
  223. :snabbkaffe,
  224. ## without this, snabbkaffe is compiled with `-define(snk_kind, '$kind')`, which
  225. ## will basically make events in tests never match any predicates.
  226. github: "kafka4beam/snabbkaffe",
  227. tag: "1.0.10",
  228. override: true,
  229. system_env: emqx_app_system_env()
  230. }
  231. def common_dep(:influxdb),
  232. do: {:influxdb, github: "emqx/influxdb-client-erl", tag: "1.1.13", override: true}
  233. def common_dep(:wolff), do: {:wolff, "4.0.4"}
  234. def common_dep(:brod_gssapi), do: {:brod_gssapi, "0.1.3"}
  235. def common_dep(:kafka_protocol),
  236. do: {:kafka_protocol, "4.1.10", override: true}
  237. def common_dep(:brod), do: {:brod, "4.3.1"}
  238. ## TODO: remove `mix.exs` from `wolff` and remove this override
  239. ## TODO: remove `mix.exs` from `pulsar` and remove this override
  240. def common_dep(:snappyer), do: {:snappyer, "1.2.10", override: true}
  241. def common_dep(:crc32cer), do: {:crc32cer, "0.1.11", override: true}
  242. def common_dep(:jesse), do: {:jesse, github: "emqx/jesse", tag: "1.8.1.1"}
  243. def common_dep(:erlavro), do: {:erlavro, github: "emqx/erlavro", tag: "2.10.0", override: true}
  244. ###############################################################################################
  245. # BEGIN DEPRECATED FOR MIX BLOCK
  246. # These should be removed once we fully migrate to mix
  247. ###############################################################################################
  248. defp emqx_apps(profile_info, version) do
  249. apps = umbrella_apps(profile_info) ++ enterprise_apps(profile_info)
  250. set_emqx_app_system_env(apps, profile_info, version)
  251. end
  252. defp umbrella_apps(profile_info = %{release_type: release_type}) do
  253. enterprise_apps = enterprise_umbrella_apps(release_type)
  254. excluded_apps = excluded_apps(release_type)
  255. "apps/*"
  256. |> Path.wildcard()
  257. |> Enum.map(fn path ->
  258. app =
  259. path
  260. |> Path.basename()
  261. |> String.to_atom()
  262. {app, path: path, manager: :rebar3, override: true}
  263. end)
  264. |> Enum.reject(fn dep_spec ->
  265. dep_spec
  266. |> elem(0)
  267. |> then(&MapSet.member?(enterprise_apps, &1))
  268. end)
  269. |> Enum.reject(fn {app, _} ->
  270. case profile_info do
  271. %{edition_type: :enterprise} ->
  272. app == :emqx_telemetry
  273. _ ->
  274. false
  275. end
  276. end)
  277. |> Enum.reject(fn {app, _} -> app == :emqx_mix_utils end)
  278. |> Enum.reject(fn {app, _} -> app in excluded_apps end)
  279. end
  280. defp enterprise_apps(_profile_info = %{release_type: release_type, edition_type: :enterprise}) do
  281. Enum.map(enterprise_umbrella_apps(release_type), fn app_name ->
  282. path = "apps/#{app_name}"
  283. {app_name, path: path, manager: :rebar3, override: true}
  284. end)
  285. end
  286. defp enterprise_apps(_profile_info) do
  287. []
  288. end
  289. # need to remove those when listing `/apps/`...
  290. defp enterprise_umbrella_apps(:standard) do
  291. MapSet.new([
  292. :emqx_connector_aggregator,
  293. :emqx_bridge_kafka,
  294. :emqx_bridge_confluent,
  295. :emqx_bridge_gcp_pubsub,
  296. :emqx_bridge_cassandra,
  297. :emqx_bridge_opents,
  298. :emqx_bridge_dynamo,
  299. :emqx_bridge_es,
  300. :emqx_bridge_greptimedb,
  301. :emqx_bridge_hstreamdb,
  302. :emqx_bridge_influxdb,
  303. :emqx_bridge_iotdb,
  304. :emqx_bridge_matrix,
  305. :emqx_bridge_mongodb,
  306. :emqx_bridge_mysql,
  307. :emqx_bridge_pgsql,
  308. :emqx_bridge_redis,
  309. :emqx_bridge_rocketmq,
  310. :emqx_bridge_tdengine,
  311. :emqx_bridge_timescale,
  312. :emqx_bridge_sqlserver,
  313. :emqx_bridge_pulsar,
  314. :emqx_oracle,
  315. :emqx_bridge_oracle,
  316. :emqx_bridge_rabbitmq,
  317. :emqx_bridge_clickhouse,
  318. :emqx_ft,
  319. :emqx_license,
  320. :emqx_opentelemetry,
  321. :emqx_s3,
  322. :emqx_bridge_s3,
  323. :emqx_bridge_azure_blob_storage,
  324. :emqx_bridge_couchbase,
  325. :emqx_bridge_snowflake,
  326. :emqx_schema_registry,
  327. :emqx_schema_validation,
  328. :emqx_message_transformation,
  329. :emqx_enterprise,
  330. :emqx_bridge_kinesis,
  331. :emqx_bridge_azure_event_hub,
  332. :emqx_gcp_device,
  333. :emqx_dashboard_rbac,
  334. :emqx_dashboard_sso,
  335. :emqx_audit,
  336. :emqx_gateway_gbt32960,
  337. :emqx_gateway_ocpp,
  338. :emqx_gateway_jt808,
  339. :emqx_bridge_syskeeper,
  340. :emqx_ds_shared_sub,
  341. :emqx_auth_ext,
  342. :emqx_cluster_link,
  343. :emqx_ds_builtin_raft,
  344. :emqx_auth_kerberos,
  345. :emqx_bridge_datalayers,
  346. :emqx_auth_cinfo
  347. ])
  348. end
  349. defp enterprise_umbrella_apps(:platform) do
  350. MapSet.union(
  351. enterprise_umbrella_apps(:standard),
  352. MapSet.new([
  353. :emqx_fdb_ds,
  354. :emqx_fdb_cli,
  355. :emqx_fdb_management,
  356. :emqx_event_history,
  357. :emqx_ds_fdb_backend
  358. ])
  359. )
  360. end
  361. defp enterprise_deps(_profile_info = %{edition_type: :enterprise}) do
  362. [
  363. {:hstreamdb_erl,
  364. github: "hstreamdb/hstreamdb_erl", tag: "0.5.18+v0.18.1+ezstd-v1.0.5-emqx1"},
  365. common_dep(:influxdb),
  366. common_dep(:wolff),
  367. common_dep(:kafka_protocol),
  368. common_dep(:brod_gssapi),
  369. common_dep(:brod),
  370. common_dep(:snappyer),
  371. common_dep(:crc32cer),
  372. {:opentsdb, github: "emqx/opentsdb-client-erl", tag: "v0.5.1", override: true},
  373. {:greptimedb,
  374. github: "GreptimeTeam/greptimedb-ingester-erl", tag: "v0.1.8", override: true},
  375. # The following two are dependencies of rabbit_common. They are needed here to
  376. # make mix not complain about conflicting versions
  377. {:thoas, github: "emqx/thoas", tag: "v1.0.0", override: true},
  378. {:credentials_obfuscation,
  379. github: "emqx/credentials-obfuscation", tag: "v3.2.0", override: true},
  380. {:rabbit_common,
  381. github: "emqx/rabbitmq-server",
  382. tag: "v3.11.13.2",
  383. sparse: "deps/rabbit_common",
  384. override: true},
  385. {:amqp_client,
  386. github: "emqx/rabbitmq-server",
  387. tag: "v3.11.13.2",
  388. sparse: "deps/amqp_client",
  389. override: true}
  390. ]
  391. end
  392. defp enterprise_deps(_profile_info) do
  393. []
  394. end
  395. defp set_emqx_app_system_env(apps, profile_info, version) do
  396. system_env = emqx_app_system_env(profile_info, version) ++ maybe_no_quic_env()
  397. Enum.map(
  398. apps,
  399. fn {app, opts} ->
  400. {app,
  401. Keyword.update(
  402. opts,
  403. :system_env,
  404. system_env,
  405. &Keyword.merge(&1, system_env)
  406. )}
  407. end
  408. )
  409. end
  410. def emqx_app_system_env(profile_info, version) do
  411. erlc_options(profile_info, version)
  412. |> dump_as_erl()
  413. |> then(&[{"ERL_COMPILER_OPTIONS", &1}])
  414. end
  415. def emqx_app_system_env() do
  416. k = {__MODULE__, :emqx_app_system_env}
  417. get_memoized(k, fn ->
  418. emqx_app_system_env(profile_info(), pkg_vsn())
  419. end)
  420. end
  421. ###############################################################################################
  422. # END DEPRECATED FOR MIX BLOCK
  423. ###############################################################################################
  424. defp erlc_options(%{edition_type: edition_type}, version) do
  425. [
  426. :debug_info,
  427. {:compile_info, [{:emqx_vsn, String.to_charlist(version)}]},
  428. {:d, :EMQX_RELEASE_EDITION, erlang_edition(edition_type)},
  429. {:d, :EMQX_ELIXIR},
  430. {:d, :EMQX_FLAVOR, get_emqx_flavor()},
  431. {:d, :snk_kind, :msg}
  432. ] ++
  433. singleton(test_env?(), {:d, :TEST}) ++
  434. singleton(not enable_quicer?(), {:d, :BUILD_WITHOUT_QUIC}) ++
  435. singleton(store_state_in_ds?(), {:d, :STORE_STATE_IN_DS, true})
  436. end
  437. defp store_state_in_ds?() do
  438. "1" == System.get_env("STORE_STATE_IN_DS")
  439. end
  440. defp singleton(false, _value), do: []
  441. defp singleton(true, value), do: [value]
  442. def profile_info() do
  443. k = {__MODULE__, :profile_info}
  444. get_memoized(k, &check_profile!/0)
  445. end
  446. def pkg_vsn() do
  447. k = {__MODULE__, :pkg_vsn}
  448. get_memoized(k, &do_pkg_vsn/0)
  449. end
  450. def common_deps() do
  451. if test_env?() do
  452. [
  453. {:bbmustache, "1.10.0"},
  454. {:cth_readable, "1.5.1"},
  455. {:proper, "1.4.0"},
  456. {:meck, "0.9.2"}
  457. ]
  458. else
  459. []
  460. end
  461. end
  462. def extra_applications() do
  463. k = {__MODULE__, :extra_applications}
  464. get_memoized(k, fn ->
  465. if test_env?() do
  466. [:eunit, :common_test, :dialyzer, :mnesia]
  467. else
  468. []
  469. end
  470. end)
  471. end
  472. def erlc_paths() do
  473. k = {__MODULE__, :erlc_paths}
  474. get_memoized(k, fn ->
  475. if test_env?() do
  476. ["src", "test"]
  477. else
  478. ["src"]
  479. end
  480. end)
  481. end
  482. def erlc_options() do
  483. k = {__MODULE__, :erlc_options}
  484. get_memoized(k, fn ->
  485. profile_info = profile_info()
  486. version = pkg_vsn()
  487. erlc_options(profile_info, version)
  488. end)
  489. end
  490. def test_env?() do
  491. k = {__MODULE__, :test_env?}
  492. get_memoized(k, fn ->
  493. env = to_string(Mix.env())
  494. System.get_env("TEST") == "1" || env =~ ~r/-test$/
  495. end)
  496. end
  497. defp set_test_env!(test_env?) do
  498. k = {__MODULE__, :test_env?}
  499. :persistent_term.put(k, test_env?)
  500. end
  501. defp get_memoized(k, compute_fn) do
  502. case :persistent_term.get(k, :undefined) do
  503. :undefined ->
  504. res = compute_fn.()
  505. :persistent_term.put(k, res)
  506. res
  507. res ->
  508. res
  509. end
  510. end
  511. def maybe_no_quic_env() do
  512. if not enable_quicer?() do
  513. [{"BUILD_WITHOUT_QUIC", "true"}]
  514. else
  515. []
  516. end
  517. end
  518. defp releases() do
  519. [
  520. emqx: fn ->
  521. %{
  522. release_type: release_type,
  523. package_type: package_type,
  524. edition_type: edition_type
  525. } = check_profile!()
  526. base_steps = [
  527. &merge_config/1,
  528. &make_docs/1,
  529. :assemble,
  530. &create_RELEASES/1,
  531. &copy_files(&1, release_type, package_type, edition_type),
  532. &copy_escript(&1, "nodetool"),
  533. &copy_escript(&1, "install_upgrade.escript")
  534. ]
  535. steps =
  536. if System.get_env("ELIXIR_MAKE_TAR") == "yes" do
  537. base_steps ++ [&prepare_tar_overlays/1, :tar]
  538. else
  539. base_steps
  540. end
  541. [
  542. applications: applications(release_type, edition_type),
  543. skip_mode_validation_for: [
  544. :lc,
  545. :emqx_mix,
  546. :emqx_machine,
  547. :emqx_gateway,
  548. :emqx_gateway_stomp,
  549. :emqx_gateway_mqttsn,
  550. :emqx_gateway_coap,
  551. :emqx_gateway_lwm2m,
  552. :emqx_gateway_exproto,
  553. :emqx_dashboard,
  554. :emqx_dashboard_sso,
  555. :emqx_audit,
  556. :emqx_resource,
  557. :emqx_connector,
  558. :emqx_exhook,
  559. :emqx_bridge,
  560. :emqx_bridge_mqtt,
  561. :emqx_modules,
  562. :emqx_management,
  563. :emqx_retainer,
  564. :emqx_prometheus,
  565. :emqx_rule_engine,
  566. :emqx_auto_subscribe,
  567. :emqx_slow_subs,
  568. :emqx_plugins,
  569. :emqx_ft,
  570. :emqx_s3,
  571. :emqx_opentelemetry,
  572. :emqx_durable_storage,
  573. :emqx_ds_builtin_local,
  574. :emqx_ds_builtin_raft,
  575. :rabbit_common,
  576. :emqx_eviction_agent,
  577. :emqx_node_rebalance
  578. ],
  579. steps: steps,
  580. strip_beams: false
  581. ]
  582. end
  583. ]
  584. end
  585. def applications(release_type, edition_type) do
  586. {:ok,
  587. [
  588. %{
  589. db_apps: db_apps,
  590. system_apps: system_apps,
  591. common_business_apps: common_business_apps,
  592. ee_business_apps: ee_business_apps,
  593. ce_business_apps: ce_business_apps
  594. }
  595. ]} = :file.consult("apps/emqx_machine/priv/reboot_lists.eterm")
  596. edition_specific_apps =
  597. if edition_type == :enterprise do
  598. ee_business_apps
  599. else
  600. ce_business_apps
  601. end
  602. business_apps = common_business_apps ++ edition_specific_apps
  603. excluded_apps = excluded_apps(release_type)
  604. system_apps =
  605. Enum.map(system_apps, fn app ->
  606. if is_atom(app), do: {app, :permanent}, else: app
  607. end)
  608. db_apps = Enum.map(db_apps, &{&1, :load})
  609. business_apps = Enum.map(business_apps, &{&1, :load})
  610. [system_apps, db_apps, [emqx_ctl: :permanent, emqx_machine: :permanent], business_apps]
  611. |> List.flatten()
  612. |> Keyword.reject(fn {app, _type} ->
  613. app in excluded_apps ||
  614. (edition_type == :enterprise && app == :emqx_telemetry)
  615. end)
  616. end
  617. defp excluded_apps(:standard) do
  618. %{
  619. mnesia_rocksdb: enable_rocksdb?(),
  620. quicer: enable_quicer?(),
  621. jq: enable_jq?(),
  622. observer: is_app?(:observer),
  623. emqx_fdb_ds: false,
  624. emqx_fdb_cli: false,
  625. emqx_fdb_management: false,
  626. emqx_event_history: false,
  627. emqx_ds_fdb_backend: false
  628. }
  629. |> Enum.reject(&elem(&1, 1))
  630. |> Enum.map(&elem(&1, 0))
  631. end
  632. defp excluded_apps(:platform) do
  633. %{
  634. mnesia_rocksdb: enable_rocksdb?(),
  635. quicer: enable_quicer?(),
  636. jq: enable_jq?(),
  637. observer: is_app?(:observer)
  638. }
  639. |> Enum.reject(&elem(&1, 1))
  640. |> Enum.map(&elem(&1, 0))
  641. end
  642. defp is_app?(name) do
  643. case Application.load(name) do
  644. :ok ->
  645. true
  646. {:error, {:already_loaded, _}} ->
  647. true
  648. _ ->
  649. false
  650. end
  651. end
  652. def check_profile!() do
  653. valid_envs = [
  654. :emqx,
  655. :"emqx-test",
  656. :"emqx-pkg",
  657. :"emqx-enterprise",
  658. :"emqx-enterprise-test",
  659. :"emqx-enterprise-pkg"
  660. ]
  661. if Mix.env() == :dev do
  662. env_profile = System.get_env("PROFILE")
  663. if env_profile do
  664. # copy from PROFILE env var
  665. System.get_env("PROFILE")
  666. |> String.to_atom()
  667. |> Mix.env()
  668. else
  669. Mix.shell().info([
  670. :yellow,
  671. "Warning: env var PROFILE is unset; defaulting to emqx"
  672. ])
  673. Mix.env(:emqx)
  674. end
  675. end
  676. if Mix.env() not in valid_envs do
  677. formatted_envs =
  678. valid_envs
  679. |> Enum.map(&" * #{&1}")
  680. |> Enum.join("\n")
  681. Mix.raise("""
  682. Invalid env #{Mix.env()}. Valid options are:
  683. #{formatted_envs}
  684. """)
  685. end
  686. mix_env = Mix.env()
  687. {
  688. release_type,
  689. package_type,
  690. edition_type
  691. } =
  692. case mix_env do
  693. :dev ->
  694. {:standard, :bin, :community}
  695. :emqx ->
  696. {:standard, :bin, :community}
  697. :"emqx-test" ->
  698. {:standard, :bin, :community}
  699. :"emqx-enterprise" ->
  700. {:standard, :bin, :enterprise}
  701. :"emqx-enterprise-test" ->
  702. {:standard, :bin, :enterprise}
  703. :"emqx-pkg" ->
  704. {:standard, :pkg, :community}
  705. :"emqx-enterprise-pkg" ->
  706. {:standard, :pkg, :enterprise}
  707. end
  708. test? = to_string(mix_env) =~ ~r/-test$/ || test_env?()
  709. normalize_env!(test?)
  710. # Mix.debug(true)
  711. if Mix.debug?() do
  712. Mix.shell().info([
  713. :blue,
  714. "mix_env: #{Mix.env()}",
  715. "; release type: #{release_type}",
  716. "; package type: #{package_type}",
  717. "; edition type: #{edition_type}",
  718. "; test env?: #{test?}"
  719. ])
  720. end
  721. test? = to_string(mix_env) =~ ~r/-test$/ || test_env?()
  722. normalize_env!(test?)
  723. # Mix.debug(true)
  724. if Mix.debug?() do
  725. Mix.shell().info([
  726. :blue,
  727. "mix_env: #{Mix.env()}",
  728. "; release type: #{release_type}",
  729. "; package type: #{package_type}",
  730. "; edition type: #{edition_type}",
  731. "; test env?: #{test?}"
  732. ])
  733. end
  734. %{
  735. release_type: release_type,
  736. package_type: package_type,
  737. edition_type: edition_type,
  738. test?: test?
  739. }
  740. end
  741. #############################################################################
  742. # Custom Steps
  743. #############################################################################
  744. # Gathers i18n files and merge them before producing docs and schemas.
  745. defp merge_config(release) do
  746. {_, 0} = System.cmd("bash", ["-c", "./scripts/merge-config.escript"])
  747. release
  748. end
  749. defp make_docs(release) do
  750. profile = System.get_env("MIX_ENV")
  751. os_cmd("build", [profile, "docs"])
  752. release
  753. end
  754. defp copy_files(release, release_type, package_type, edition_type) do
  755. overwrite? = Keyword.get(release.options, :overwrite, false)
  756. bin = Path.join(release.path, "bin")
  757. etc = Path.join(release.path, "etc")
  758. log = Path.join(release.path, "log")
  759. plugins = Path.join(release.path, "plugins")
  760. Mix.Generator.create_directory(bin)
  761. Mix.Generator.create_directory(etc)
  762. Mix.Generator.create_directory(log)
  763. Mix.Generator.create_directory(plugins)
  764. Mix.Generator.create_directory(Path.join(etc, "certs"))
  765. Enum.each(
  766. ["mnesia", "configs", "patches", "scripts"],
  767. fn dir ->
  768. path = Path.join([release.path, "data", dir])
  769. Mix.Generator.create_directory(path)
  770. end
  771. )
  772. Mix.Generator.copy_file(
  773. "apps/emqx_auth/etc/acl.conf",
  774. Path.join(etc, "acl.conf"),
  775. force: overwrite?
  776. )
  777. # required by emqx_auth
  778. File.cp_r!(
  779. "apps/emqx/etc/certs",
  780. Path.join(etc, "certs")
  781. )
  782. profile = System.get_env("MIX_ENV")
  783. File.cp_r!(
  784. "rel/config/examples",
  785. Path.join(etc, "examples"),
  786. force: overwrite?
  787. )
  788. # copy /rel/config/ee-examples if profile is enterprise
  789. case profile do
  790. "emqx-enterprise" ->
  791. File.cp_r!(
  792. "rel/config/ee-examples",
  793. Path.join(etc, "examples"),
  794. force: overwrite?
  795. )
  796. _ ->
  797. :ok
  798. end
  799. # this is required by the produced escript / nodetool
  800. Mix.Generator.copy_file(
  801. Path.join(release.version_path, "start_clean.boot"),
  802. Path.join(bin, "no_dot_erlang.boot"),
  803. force: overwrite?
  804. )
  805. assigns = template_vars(release, release_type, package_type, edition_type)
  806. # This is generated by `scripts/merge-config.escript` or `make merge-config`
  807. # So, this should be run before the release.
  808. # TODO: run as a "compiler" step???
  809. render_template(
  810. "apps/emqx_conf/etc/emqx.conf.all",
  811. assigns,
  812. Path.join(etc, "emqx.conf")
  813. )
  814. render_template(
  815. "apps/emqx_conf/etc/base.hocon",
  816. assigns,
  817. Path.join(etc, "base.hocon")
  818. )
  819. render_template(
  820. "rel/emqx_vars",
  821. assigns,
  822. Path.join([release.path, "releases", "emqx_vars"])
  823. )
  824. vm_args_template_path =
  825. case release_type do
  826. _ ->
  827. "apps/emqx/etc/vm.args.cloud"
  828. end
  829. render_template(
  830. vm_args_template_path,
  831. assigns,
  832. [
  833. Path.join(etc, "vm.args"),
  834. Path.join(release.version_path, "vm.args")
  835. ]
  836. )
  837. for name <- [
  838. "emqx",
  839. "emqx_ctl"
  840. ] do
  841. Mix.Generator.copy_file(
  842. "bin/#{name}",
  843. Path.join(bin, name),
  844. force: overwrite?
  845. )
  846. # Files with the version appended are expected by the release
  847. # upgrade script `install_upgrade.escript`
  848. Mix.Generator.copy_file(
  849. Path.join(bin, name),
  850. Path.join(bin, name <> "-#{release.version}"),
  851. force: overwrite?
  852. )
  853. end
  854. for base_name <- ["emqx", "emqx_ctl"],
  855. suffix <- ["", "-#{release.version}"] do
  856. name = base_name <> suffix
  857. File.chmod!(Path.join(bin, name), 0o755)
  858. end
  859. Mix.Generator.copy_file(
  860. "bin/node_dump",
  861. Path.join(bin, "node_dump"),
  862. force: overwrite?
  863. )
  864. File.chmod!(Path.join(bin, "node_dump"), 0o755)
  865. Mix.Generator.copy_file(
  866. "bin/emqx_cluster_rescue",
  867. Path.join(bin, "emqx_cluster_rescue"),
  868. force: overwrite?
  869. )
  870. File.chmod!(Path.join(bin, "emqx_cluster_rescue"), 0o755)
  871. render_template(
  872. "rel/BUILD_INFO",
  873. assigns,
  874. Path.join(release.version_path, "BUILD_INFO")
  875. )
  876. release
  877. end
  878. defp render_template(template, assigns, target) when is_binary(target) do
  879. render_template(template, assigns, [target])
  880. end
  881. defp render_template(template, assigns, tartgets) when is_list(tartgets) do
  882. rendered =
  883. File.read!(template)
  884. |> from_rebar_to_eex_template()
  885. |> EEx.eval_string(assigns)
  886. for target <- tartgets do
  887. File.write!(target, rendered)
  888. end
  889. end
  890. # needed by nodetool and by release_handler
  891. defp create_RELEASES(release) do
  892. apps =
  893. Enum.map(release.applications, fn {app_name, app_props} ->
  894. app_vsn = Keyword.fetch!(app_props, :vsn)
  895. app_path =
  896. "./lib"
  897. |> Path.join("#{app_name}-#{app_vsn}")
  898. |> to_charlist()
  899. {app_name, app_vsn, app_path}
  900. end)
  901. release_entry = [
  902. {
  903. :release,
  904. to_charlist(release.name),
  905. to_charlist(release.version),
  906. release.erts_version,
  907. apps,
  908. :permanent
  909. }
  910. ]
  911. release.path
  912. |> Path.join("releases")
  913. |> Path.join("RELEASES")
  914. |> File.open!([:write, :utf8], fn handle ->
  915. IO.puts(handle, "%% coding: utf-8")
  916. :io.format(handle, ~c"~tp.~n", [release_entry])
  917. end)
  918. release
  919. end
  920. defp copy_escript(release, escript_name) do
  921. [shebang, rest] =
  922. "bin/#{escript_name}"
  923. |> File.read!()
  924. |> String.split("\n", parts: 2)
  925. # the elixir version of escript + start.boot required the boot_var
  926. # RELEASE_LIB to be defined.
  927. # enable-feature is not required when 1.6.x
  928. boot_var = "%%!-boot_var RELEASE_LIB $RUNNER_ROOT_DIR/lib -enable-feature maybe_expr"
  929. # Files with the version appended are expected by the release
  930. # upgrade script `install_upgrade.escript`
  931. Enum.each(
  932. [escript_name, escript_name <> "-" <> release.version],
  933. fn name ->
  934. path = Path.join([release.path, "bin", name])
  935. File.write!(path, [shebang, "\n", boot_var, "\n", rest])
  936. end
  937. )
  938. release
  939. end
  940. # The `:tar` built-in step in Mix Release does not currently add the
  941. # `etc` directory into the resulting tarball. The workaround is to
  942. # add those to the `:overlays` key before running `:tar`.
  943. # See: https://hexdocs.pm/mix/1.13.4/Mix.Release.html#__struct__/0
  944. defp prepare_tar_overlays(release) do
  945. Map.update!(
  946. release,
  947. :overlays,
  948. &[
  949. "etc",
  950. "data",
  951. "plugins",
  952. "bin/node_dump"
  953. | &1
  954. ]
  955. )
  956. end
  957. #############################################################################
  958. # Helper functions
  959. #############################################################################
  960. defp template_vars(release, release_type, :bin = _package_type, edition_type) do
  961. [
  962. emqx_default_erlang_cookie: default_cookie(),
  963. emqx_configuration_doc: emqx_configuration_doc(edition_type, :root),
  964. emqx_configuration_doc_log: emqx_configuration_doc(edition_type, :log),
  965. platform_data_dir: "data",
  966. platform_etc_dir: "etc",
  967. platform_plugins_dir: "plugins",
  968. runner_bin_dir: "$RUNNER_ROOT_DIR/bin",
  969. emqx_etc_dir: "$RUNNER_ROOT_DIR/etc",
  970. runner_lib_dir: "$RUNNER_ROOT_DIR/lib",
  971. runner_log_dir: "$RUNNER_ROOT_DIR/log",
  972. runner_user: "",
  973. release_version: release.version,
  974. erts_vsn: release.erts_version,
  975. # FIXME: this is empty in `make emqx` ???
  976. erl_opts: "",
  977. emqx_description: emqx_description(release_type, edition_type),
  978. emqx_schema_mod: emqx_schema_mod(edition_type),
  979. is_elixir: "yes",
  980. is_enterprise: if(edition_type == :enterprise, do: "yes", else: "no")
  981. ] ++ build_info()
  982. end
  983. defp template_vars(release, release_type, :pkg = _package_type, edition_type) do
  984. [
  985. emqx_default_erlang_cookie: default_cookie(),
  986. emqx_configuration_doc: emqx_configuration_doc(edition_type, :root),
  987. emqx_configuration_doc_log: emqx_configuration_doc(edition_type, :log),
  988. platform_data_dir: "/var/lib/emqx",
  989. platform_etc_dir: "/etc/emqx",
  990. platform_plugins_dir: "/var/lib/emqx/plugins",
  991. runner_bin_dir: "/usr/bin",
  992. emqx_etc_dir: "/etc/emqx",
  993. runner_lib_dir: "$RUNNER_ROOT_DIR/lib",
  994. runner_log_dir: "/var/log/emqx",
  995. runner_user: "emqx",
  996. release_version: release.version,
  997. erts_vsn: release.erts_version,
  998. # FIXME: this is empty in `make emqx` ???
  999. erl_opts: "",
  1000. emqx_description: emqx_description(release_type, edition_type),
  1001. emqx_schema_mod: emqx_schema_mod(edition_type),
  1002. is_elixir: "yes",
  1003. is_enterprise: if(edition_type == :enterprise, do: "yes", else: "no")
  1004. ] ++ build_info()
  1005. end
  1006. defp default_cookie() do
  1007. "emqx50elixir"
  1008. end
  1009. defp emqx_description(release_type, edition_type) do
  1010. case {release_type, edition_type} do
  1011. {_, :enterprise} ->
  1012. case get_emqx_flavor() do
  1013. :official ->
  1014. "EMQX Enterprise"
  1015. flavor ->
  1016. "EMQX Enterprise(#{flavor})"
  1017. end
  1018. {_, :community} ->
  1019. "EMQX"
  1020. end
  1021. end
  1022. defp emqx_configuration_doc(:enterprise, :root),
  1023. do: "https://docs.emqx.com/en/enterprise/latest/configuration/configuration.html"
  1024. defp emqx_configuration_doc(:enterprise, :log),
  1025. do: "https://docs.emqx.com/en/enterprise/latest/configuration/logs.html"
  1026. defp emqx_configuration_doc(:community, :root),
  1027. do: "https://www.emqx.io/docs/en/latest/configuration/configuration.html"
  1028. defp emqx_configuration_doc(:community, :log),
  1029. do: "https://www.emqx.io/docs/en/latest/configuration/logs.html"
  1030. defp emqx_schema_mod(:enterprise), do: :emqx_enterprise_schema
  1031. defp emqx_schema_mod(:community), do: :emqx_conf_schema
  1032. def jq_dep() do
  1033. if enable_jq?(),
  1034. do: [{:jq, github: "emqx/jq", tag: "v0.3.12", override: true}],
  1035. else: []
  1036. end
  1037. def quicer_dep() do
  1038. if enable_quicer?(),
  1039. # in conflict with emqx and emqtt
  1040. do: [
  1041. {:quicer, github: "emqx/quic", tag: "0.1.10", override: true}
  1042. ],
  1043. else: []
  1044. end
  1045. defp enable_jq?() do
  1046. not Enum.any?([
  1047. build_without_jq?()
  1048. ])
  1049. end
  1050. def enable_quicer?() do
  1051. "1" == System.get_env("BUILD_WITH_QUIC") or
  1052. not build_without_quic?()
  1053. end
  1054. def get_emqx_flavor() do
  1055. case System.get_env("EMQX_FLAVOR") do
  1056. nil -> :official
  1057. "" -> :official
  1058. flavor -> String.to_atom(flavor)
  1059. end
  1060. end
  1061. defp enable_rocksdb?() do
  1062. not Enum.any?([
  1063. raspbian?(),
  1064. build_without_rocksdb?()
  1065. ])
  1066. end
  1067. defp do_pkg_vsn() do
  1068. %{edition_type: edition_type} = check_profile!()
  1069. basedir = Path.dirname(__ENV__.file)
  1070. script = Path.join(basedir, "pkg-vsn.sh")
  1071. os_cmd(script, [Atom.to_string(edition_type)])
  1072. end
  1073. defp os_cmd(script, args) do
  1074. {str, 0} = System.cmd("bash", [script | args])
  1075. String.trim(str)
  1076. end
  1077. defp raspbian?() do
  1078. os_cmd("./scripts/get-distro.sh", []) =~ "raspbian"
  1079. end
  1080. defp build_without_jq?() do
  1081. opt = System.get_env("BUILD_WITHOUT_JQ", "false")
  1082. String.downcase(opt) != "false"
  1083. end
  1084. def build_without_quic?() do
  1085. opt = System.get_env("BUILD_WITHOUT_QUIC", "false")
  1086. String.downcase(opt) != "false"
  1087. end
  1088. defp build_without_rocksdb?() do
  1089. opt = System.get_env("BUILD_WITHOUT_ROCKSDB", "false")
  1090. String.downcase(opt) != "false"
  1091. end
  1092. defp from_rebar_to_eex_template(str) do
  1093. # we must not consider surrounding space in the template var name
  1094. # because some help strings contain informative variables that
  1095. # should not be interpolated, and those have no spaces.
  1096. Regex.replace(
  1097. ~r/\{\{ ([a-zA-Z0-9_]+) \}\}/,
  1098. str,
  1099. "<%= \\g{1} %>"
  1100. )
  1101. end
  1102. defp build_info() do
  1103. [
  1104. build_info_arch: to_string(:erlang.system_info(:system_architecture)),
  1105. build_info_wordsize: wordsize(),
  1106. build_info_os: os_cmd("./scripts/get-distro.sh", []),
  1107. build_info_erlang: otp_release(),
  1108. build_info_elixir: System.version(),
  1109. build_info_relform: System.get_env("EMQX_REL_FORM", "tgz")
  1110. ]
  1111. end
  1112. # https://github.com/erlang/rebar3/blob/e3108ac187b88fff01eca6001a856283a3e0ec87/src/rebar_utils.erl#L142
  1113. defp wordsize() do
  1114. size =
  1115. try do
  1116. :erlang.system_info({:wordsize, :external})
  1117. rescue
  1118. ErlangError ->
  1119. :erlang.system_info(:wordsize)
  1120. end
  1121. to_string(8 * size)
  1122. end
  1123. defp normalize_env!(test_env?) do
  1124. env =
  1125. case Mix.env() do
  1126. :dev ->
  1127. :emqx
  1128. env ->
  1129. env
  1130. end
  1131. if test_env? do
  1132. ensure_test_mix_env!()
  1133. end
  1134. Mix.env(env)
  1135. end
  1136. # As from Erlang/OTP 17, the OTP release number corresponds to the
  1137. # major OTP version number. No erlang:system_info() argument gives
  1138. # the exact OTP version.
  1139. # https://www.erlang.org/doc/man/erlang.html#system_info_otp_release
  1140. # https://github.com/erlang/rebar3/blob/e3108ac187b88fff01eca6001a856283a3e0ec87/src/rebar_utils.erl#L572-L577
  1141. defp otp_release() do
  1142. major_version = System.otp_release()
  1143. root_dir = to_string(:code.root_dir())
  1144. [root_dir, "releases", major_version, "OTP_VERSION"]
  1145. |> Path.join()
  1146. |> File.read()
  1147. |> case do
  1148. {:error, _} ->
  1149. major_version
  1150. {:ok, version} ->
  1151. version
  1152. |> String.trim()
  1153. |> String.split("**")
  1154. |> List.first()
  1155. end
  1156. end
  1157. defp dump_as_erl(term) do
  1158. term
  1159. |> then(&:io_lib.format("~0p", [&1]))
  1160. |> :erlang.iolist_to_binary()
  1161. end
  1162. defp erlang_edition(:community), do: :ce
  1163. defp erlang_edition(:enterprise), do: :ee
  1164. defp aliases() do
  1165. [
  1166. ct: &do_ct/1,
  1167. eunit: &do_eunit/1,
  1168. proper: &do_proper/1,
  1169. dialyzer: &do_dialyzer/1
  1170. ]
  1171. end
  1172. defp do_ct(args) do
  1173. IO.inspect(args)
  1174. Mix.shell().info("testing")
  1175. ensure_test_mix_env!()
  1176. set_test_env!(true)
  1177. Mix.Task.run("emqx.ct", args)
  1178. end
  1179. defp do_eunit(args) do
  1180. ensure_test_mix_env!()
  1181. set_test_env!(true)
  1182. Mix.Task.run("emqx.eunit", args)
  1183. end
  1184. defp do_proper(args) do
  1185. ensure_test_mix_env!()
  1186. set_test_env!(true)
  1187. Mix.Task.run("emqx.proper", args)
  1188. end
  1189. defp do_dialyzer(args) do
  1190. Mix.Task.run("emqx.dialyzer", args)
  1191. end
  1192. defp ensure_test_mix_env!() do
  1193. Mix.env()
  1194. |> to_string()
  1195. |> then(fn env ->
  1196. if String.ends_with?(env, "-test") do
  1197. env
  1198. else
  1199. env <> "-test"
  1200. end
  1201. end)
  1202. |> String.to_atom()
  1203. |> Mix.env()
  1204. end
  1205. end