mix.exs 37 KB

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