mix.exs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727
  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. recommendd 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-edge`
  14. * `emqx-enterprise`
  15. * `emqx-pkg`
  16. * `emqx-edge-pkg`
  17. * `emqx-enterprise-pkg`
  18. * `dev` -> same as `emqx`, for convenience
  19. ## Release Environment Variables
  20. The release build is controlled by a few environment variables.
  21. * `ELIXIR_MAKE_TAR` - If set to `yes`, will produce a `.tar.gz`
  22. tarball along with the release.
  23. """
  24. def project() do
  25. check_profile!()
  26. [
  27. app: :emqx_mix,
  28. version: pkg_vsn(),
  29. deps: deps(),
  30. releases: releases()
  31. ]
  32. end
  33. defp deps() do
  34. # we need several overrides here because dependencies specify
  35. # other exact versions, and not ranges.
  36. [
  37. {:lc, github: "qzhuyan/lc", tag: "0.1.2"},
  38. {:redbug, "2.0.7"},
  39. {:typerefl, github: "ieQu1/typerefl", tag: "0.8.6", override: true},
  40. {:ehttpc, github: "emqx/ehttpc", tag: "0.1.12"},
  41. {:gproc, github: "uwiger/gproc", tag: "0.8.0", override: true},
  42. {:jiffy, github: "emqx/jiffy", tag: "1.0.5", override: true},
  43. {:cowboy, github: "emqx/cowboy", tag: "2.9.0", override: true},
  44. {:esockd, github: "emqx/esockd", tag: "5.9.1", override: true},
  45. {:mria, github: "emqx/mria", tag: "0.2.1", override: true},
  46. {:ekka, github: "emqx/ekka", tag: "0.12.1", override: true},
  47. {:gen_rpc, github: "emqx/gen_rpc", tag: "2.8.0", override: true},
  48. {:minirest, github: "emqx/minirest", tag: "1.2.12", override: true},
  49. {:ecpool, github: "emqx/ecpool", tag: "0.5.2"},
  50. {:replayq, "0.3.3", override: true},
  51. {:pbkdf2, github: "emqx/erlang-pbkdf2", tag: "2.0.4", override: true},
  52. {:emqtt, github: "emqx/emqtt", tag: "1.4.3", override: true},
  53. {:rulesql, github: "emqx/rulesql", tag: "0.1.4"},
  54. {:observer_cli, "1.7.1"},
  55. {:system_monitor, github: "ieQu1/system_monitor", tag: "3.0.2"},
  56. # in conflict by emqtt and hocon
  57. {:getopt, "1.0.2", override: true},
  58. {:snabbkaffe, github: "kafka4beam/snabbkaffe", tag: "0.18.0", override: true},
  59. {:hocon, github: "emqx/hocon", tag: "0.26.1", override: true},
  60. {:emqx_http_lib, github: "emqx/emqx_http_lib", tag: "0.4.1", override: true},
  61. {:esasl, github: "emqx/esasl", tag: "0.2.0"},
  62. {:jose, github: "potatosalad/erlang-jose", tag: "1.11.2"},
  63. # in conflict by ehttpc and emqtt
  64. {:gun, github: "emqx/gun", tag: "1.3.6", override: true},
  65. # in conflict by emqx_connectior and system_monitor
  66. {:epgsql, github: "emqx/epgsql", tag: "4.7-emqx.2", override: true},
  67. # in conflict by mongodb and eredis_cluster
  68. {:poolboy, github: "emqx/poolboy", tag: "1.5.2", override: true},
  69. # in conflict by emqx and observer_cli
  70. {:recon, github: "ferd/recon", tag: "2.5.1", override: true},
  71. {:jsx, github: "talentdeficit/jsx", tag: "v3.1.0", override: true},
  72. # dependencies of dependencies; we choose specific refs to match
  73. # what rebar3 chooses.
  74. # in conflict by gun and emqtt
  75. {:cowlib,
  76. github: "ninenines/cowlib", ref: "c6553f8308a2ca5dcd69d845f0a7d098c40c3363", override: true},
  77. # in conflict by cowboy_swagger and cowboy
  78. {:ranch,
  79. github: "ninenines/ranch", ref: "a692f44567034dacf5efcaa24a24183788594eb7", override: true},
  80. # in conflict by grpc and eetcd
  81. {:gpb, "4.11.2", override: true, runtime: false}
  82. ] ++ umbrella_apps() ++ bcrypt_dep() ++ quicer_dep()
  83. end
  84. defp umbrella_apps() do
  85. "apps/*"
  86. |> Path.wildcard()
  87. |> Enum.map(fn path ->
  88. app =
  89. path
  90. |> String.trim_leading("apps/")
  91. |> String.to_atom()
  92. {app, path: path, manager: :rebar3, override: true}
  93. end)
  94. end
  95. defp releases() do
  96. [
  97. emqx: fn ->
  98. %{
  99. release_type: release_type,
  100. package_type: package_type,
  101. edition_type: edition_type
  102. } = check_profile!()
  103. base_steps = [
  104. :assemble,
  105. &create_RELEASES/1,
  106. &copy_files(&1, release_type, package_type, edition_type),
  107. &copy_escript(&1, "nodetool"),
  108. &copy_escript(&1, "install_upgrade.escript")
  109. ]
  110. steps =
  111. if System.get_env("ELIXIR_MAKE_TAR") == "yes" do
  112. base_steps ++ [&prepare_tar_overlays/1, :tar]
  113. else
  114. base_steps
  115. end
  116. [
  117. applications: applications(release_type, edition_type),
  118. skip_mode_validation_for: [
  119. :emqx_gateway,
  120. :emqx_dashboard,
  121. :emqx_resource,
  122. :emqx_connector,
  123. :emqx_exhook,
  124. :emqx_bridge,
  125. :emqx_modules,
  126. :emqx_management,
  127. :emqx_statsd,
  128. :emqx_retainer,
  129. :emqx_prometheus,
  130. :emqx_plugins
  131. ],
  132. steps: steps,
  133. strip_beams: false
  134. ]
  135. end
  136. ]
  137. end
  138. def applications(release_type, edition_type) do
  139. [
  140. crypto: :permanent,
  141. public_key: :permanent,
  142. asn1: :permanent,
  143. syntax_tools: :permanent,
  144. ssl: :permanent,
  145. os_mon: :permanent,
  146. inets: :permanent,
  147. compiler: :permanent,
  148. runtime_tools: :permanent,
  149. redbug: :permanent,
  150. hocon: :load,
  151. emqx: :load,
  152. emqx_conf: :load,
  153. emqx_machine: :permanent,
  154. mria: :load,
  155. mnesia: :load,
  156. ekka: :load,
  157. emqx_plugin_libs: :load,
  158. esasl: :load,
  159. observer_cli: :permanent,
  160. system_monitor: :load,
  161. emqx_http_lib: :permanent,
  162. emqx_resource: :permanent,
  163. emqx_connector: :permanent,
  164. emqx_authn: :permanent,
  165. emqx_authz: :permanent,
  166. emqx_auto_subscribe: :permanent,
  167. emqx_gateway: :permanent,
  168. emqx_exhook: :permanent,
  169. emqx_bridge: :permanent,
  170. emqx_rule_engine: :permanent,
  171. emqx_modules: :permanent,
  172. emqx_management: :permanent,
  173. emqx_dashboard: :permanent,
  174. emqx_retainer: :permanent,
  175. emqx_statsd: :permanent,
  176. emqx_prometheus: :permanent,
  177. emqx_psk: :permanent,
  178. emqx_slow_subs: :permanent,
  179. emqx_plugins: :permanent,
  180. emqx_mix: :none
  181. ] ++
  182. if(enable_quicer?(), do: [quicer: :permanent], else: []) ++
  183. if(enable_bcrypt?(), do: [bcrypt: :permanent], else: []) ++
  184. if(edition_type == :enterprise,
  185. do: [
  186. emqx_enterprise_conf: :load,
  187. emqx_license: :permanent
  188. ],
  189. else: []
  190. ) ++
  191. if(release_type == :cloud,
  192. do: [xmerl: :permanent, observer: :load],
  193. else: []
  194. )
  195. end
  196. def emqx_machine_boot_apps(:community) do
  197. [
  198. :emqx_prometheus,
  199. :emqx_modules,
  200. :emqx_dashboard,
  201. :emqx_connector,
  202. :emqx_gateway,
  203. :emqx_statsd,
  204. :emqx_resource,
  205. :emqx_rule_engine,
  206. :emqx_bridge,
  207. :emqx_plugin_libs,
  208. :emqx_management,
  209. :emqx_retainer,
  210. :emqx_exhook,
  211. :emqx_authn,
  212. :emqx_authz,
  213. :emqx_plugin
  214. ]
  215. end
  216. def emqx_machine_boot_apps(:enterprise) do
  217. emqx_machine_boot_apps(:community) ++
  218. []
  219. end
  220. defp emqx_machine_boot_app_list(edition_type) do
  221. edition_type
  222. |> emqx_machine_boot_apps()
  223. |> Enum.map(&Atom.to_string/1)
  224. |> Enum.join(", ")
  225. end
  226. def check_profile!() do
  227. valid_envs = [
  228. :dev,
  229. :emqx,
  230. :"emqx-pkg",
  231. :"emqx-enterprise",
  232. :"emqx-enterprise-pkg",
  233. :"emqx-edge",
  234. :"emqx-edge-pkg"
  235. ]
  236. if Mix.env() not in valid_envs do
  237. formatted_envs =
  238. valid_envs
  239. |> Enum.map(&" * #{&1}")
  240. |> Enum.join("\n")
  241. Mix.raise("""
  242. Invalid env #{Mix.env()}. Valid options are:
  243. #{formatted_envs}
  244. """)
  245. end
  246. {
  247. release_type,
  248. package_type,
  249. edition_type
  250. } =
  251. case Mix.env() do
  252. :dev ->
  253. {:cloud, :bin, :community}
  254. :emqx ->
  255. {:cloud, :bin, :community}
  256. :"emqx-edge" ->
  257. {:edge, :bin, :community}
  258. :"emqx-enterprise" ->
  259. {:cloud, :bin, :enterprise}
  260. :"emqx-pkg" ->
  261. {:cloud, :pkg, :community}
  262. :"emqx-edge-pkg" ->
  263. {:edge, :pkg, :community}
  264. :"emqx-enterprise-pkg" ->
  265. {:cloud, :pkg, :enterprise}
  266. end
  267. normalize_env!()
  268. %{
  269. release_type: release_type,
  270. package_type: package_type,
  271. edition_type: edition_type
  272. }
  273. end
  274. #############################################################################
  275. # Custom Steps
  276. #############################################################################
  277. defp copy_files(release, release_type, package_type, edition_type) do
  278. overwrite? = Keyword.get(release.options, :overwrite, false)
  279. bin = Path.join(release.path, "bin")
  280. etc = Path.join(release.path, "etc")
  281. log = Path.join(release.path, "log")
  282. Mix.Generator.create_directory(bin)
  283. Mix.Generator.create_directory(etc)
  284. Mix.Generator.create_directory(log)
  285. Mix.Generator.create_directory(Path.join(etc, "certs"))
  286. Enum.each(
  287. ["mnesia", "configs", "patches", "scripts"],
  288. fn dir ->
  289. path = Path.join([release.path, "data", dir])
  290. Mix.Generator.create_directory(path)
  291. end
  292. )
  293. Mix.Generator.copy_file(
  294. "apps/emqx_authz/etc/acl.conf",
  295. Path.join(etc, "acl.conf"),
  296. force: overwrite?
  297. )
  298. # required by emqx_authz
  299. File.cp_r!(
  300. "apps/emqx/etc/certs",
  301. Path.join(etc, "certs")
  302. )
  303. # this is required by the produced escript / nodetool
  304. Mix.Generator.copy_file(
  305. Path.join(release.version_path, "start_clean.boot"),
  306. Path.join(bin, "no_dot_erlang.boot"),
  307. force: overwrite?
  308. )
  309. assigns = template_vars(release, release_type, package_type, edition_type)
  310. # This is generated by `scripts/merge-config.escript` or `make
  311. # conf-segs`. So, this should be run before the release.
  312. # TODO: run as a "compiler" step???
  313. render_template(
  314. "apps/emqx_conf/etc/emqx.conf.all",
  315. assigns,
  316. Path.join(etc, "emqx.conf")
  317. )
  318. if edition_type == :enterprise do
  319. render_template(
  320. "apps/emqx_conf/etc/emqx_enterprise.conf.all",
  321. assigns,
  322. Path.join(etc, "emqx_enterprise.conf")
  323. )
  324. end
  325. render_template(
  326. "rel/emqx_vars",
  327. assigns,
  328. Path.join([release.path, "releases", "emqx_vars"])
  329. )
  330. vm_args_template_path =
  331. case release_type do
  332. :cloud ->
  333. "apps/emqx/etc/emqx_cloud/vm.args"
  334. :edge ->
  335. "apps/emqx/etc/emqx_edge/vm.args"
  336. end
  337. render_template(
  338. vm_args_template_path,
  339. assigns,
  340. [
  341. Path.join(etc, "vm.args"),
  342. Path.join(release.version_path, "vm.args")
  343. ]
  344. )
  345. for name <- [
  346. "emqx",
  347. "emqx_ctl"
  348. ] do
  349. Mix.Generator.copy_file(
  350. "bin/#{name}",
  351. Path.join(bin, name),
  352. force: overwrite?
  353. )
  354. # Files with the version appended are expected by the release
  355. # upgrade script `install_upgrade.escript`
  356. Mix.Generator.copy_file(
  357. Path.join(bin, name),
  358. Path.join(bin, name <> "-#{release.version}"),
  359. force: overwrite?
  360. )
  361. end
  362. for base_name <- ["emqx", "emqx_ctl"],
  363. suffix <- ["", "-#{release.version}"] do
  364. name = base_name <> suffix
  365. File.chmod!(Path.join(bin, name), 0o755)
  366. end
  367. Mix.Generator.copy_file(
  368. "bin/node_dump",
  369. Path.join(bin, "node_dump"),
  370. force: overwrite?
  371. )
  372. File.chmod!(Path.join(bin, "node_dump"), 0o755)
  373. render_template(
  374. "rel/BUILD_INFO",
  375. assigns,
  376. Path.join(release.version_path, "BUILD_INFO")
  377. )
  378. release
  379. end
  380. defp render_template(template, assigns, target) when is_binary(target) do
  381. render_template(template, assigns, [target])
  382. end
  383. defp render_template(template, assigns, tartgets) when is_list(tartgets) do
  384. rendered =
  385. File.read!(template)
  386. |> from_rebar_to_eex_template()
  387. |> EEx.eval_string(assigns)
  388. for target <- tartgets do
  389. File.write!(target, rendered)
  390. end
  391. end
  392. # needed by nodetool and by release_handler
  393. defp create_RELEASES(release) do
  394. apps =
  395. Enum.map(release.applications, fn {app_name, app_props} ->
  396. app_vsn = Keyword.fetch!(app_props, :vsn)
  397. app_path =
  398. "./lib"
  399. |> Path.join("#{app_name}-#{app_vsn}")
  400. |> to_charlist()
  401. {app_name, app_vsn, app_path}
  402. end)
  403. release_entry = [
  404. {
  405. :release,
  406. to_charlist(release.name),
  407. to_charlist(release.version),
  408. release.erts_version,
  409. apps,
  410. :permanent
  411. }
  412. ]
  413. release.path
  414. |> Path.join("releases")
  415. |> Path.join("RELEASES")
  416. |> File.open!([:write, :utf8], fn handle ->
  417. IO.puts(handle, "%% coding: utf-8")
  418. :io.format(handle, '~tp.~n', [release_entry])
  419. end)
  420. release
  421. end
  422. defp copy_escript(release, escript_name) do
  423. [shebang, rest] =
  424. "bin/#{escript_name}"
  425. |> File.read!()
  426. |> String.split("\n", parts: 2)
  427. # the elixir version of escript + start.boot required the boot_var
  428. # RELEASE_LIB to be defined.
  429. boot_var = "%%!-boot_var RELEASE_LIB $RUNNER_ROOT_DIR/lib"
  430. # Files with the version appended are expected by the release
  431. # upgrade script `install_upgrade.escript`
  432. Enum.each(
  433. [escript_name, escript_name <> "-" <> release.version],
  434. fn name ->
  435. path = Path.join([release.path, "bin", name])
  436. File.write!(path, [shebang, "\n", boot_var, "\n", rest])
  437. end
  438. )
  439. release
  440. end
  441. # The `:tar` built-in step in Mix Release does not currently add the
  442. # `etc` directory into the resulting tarball. The workaround is to
  443. # add those to the `:overlays` key before running `:tar`.
  444. # See: https://hexdocs.pm/mix/1.13.3/Mix.Release.html#__struct__/0
  445. defp prepare_tar_overlays(release) do
  446. Map.update!(
  447. release,
  448. :overlays,
  449. &[
  450. "etc",
  451. "data",
  452. "bin/node_dump"
  453. | &1
  454. ]
  455. )
  456. end
  457. #############################################################################
  458. # Helper functions
  459. #############################################################################
  460. defp template_vars(release, release_type, :bin = _package_type, edition_type) do
  461. [
  462. platform_bin_dir: "bin",
  463. platform_data_dir: "data",
  464. platform_etc_dir: "etc",
  465. platform_lib_dir: "lib",
  466. platform_log_dir: "log",
  467. platform_plugins_dir: "plugins",
  468. runner_root_dir: "$(cd $(dirname $(readlink $0 || echo $0))/..; pwd -P)",
  469. runner_bin_dir: "$RUNNER_ROOT_DIR/bin",
  470. runner_etc_dir: "$RUNNER_ROOT_DIR/etc",
  471. runner_lib_dir: "$RUNNER_ROOT_DIR/lib",
  472. runner_log_dir: "$RUNNER_ROOT_DIR/log",
  473. runner_user: "",
  474. release_version: release.version,
  475. erts_vsn: release.erts_version,
  476. # FIXME: this is empty in `make emqx` ???
  477. erl_opts: "",
  478. emqx_description: emqx_description(release_type, edition_type),
  479. emqx_schema_mod: emqx_schema_mod(edition_type),
  480. emqx_machine_boot_apps: emqx_machine_boot_app_list(edition_type),
  481. is_elixir: "yes",
  482. is_enterprise: if(edition_type == :enterprise, do: "yes", else: "no")
  483. ] ++ build_info()
  484. end
  485. defp template_vars(release, release_type, :pkg = _package_type, edition_type) do
  486. [
  487. platform_bin_dir: "",
  488. platform_data_dir: "/var/lib/emqx",
  489. platform_etc_dir: "/etc/emqx",
  490. platform_lib_dir: "",
  491. platform_log_dir: "/var/log/emqx",
  492. platform_plugins_dir: "/var/lib/emqx/plugins",
  493. runner_root_dir: "/usr/lib/emqx",
  494. runner_bin_dir: "/usr/bin",
  495. runner_etc_dir: "/etc/emqx",
  496. runner_lib_dir: "$RUNNER_ROOT_DIR/lib",
  497. runner_log_dir: "/var/log/emqx",
  498. runner_user: "emqx",
  499. release_version: release.version,
  500. erts_vsn: release.erts_version,
  501. # FIXME: this is empty in `make emqx` ???
  502. erl_opts: "",
  503. emqx_description: emqx_description(release_type, edition_type),
  504. emqx_schema_mod: emqx_schema_mod(edition_type),
  505. emqx_machine_boot_apps: emqx_machine_boot_app_list(edition_type),
  506. is_elixir: "yes",
  507. is_enterprise: if(edition_type == :enterprise, do: "yes", else: "no")
  508. ] ++ build_info()
  509. end
  510. defp emqx_description(release_type, edition_type) do
  511. case {release_type, edition_type} do
  512. {:cloud, :enterprise} ->
  513. "EMQX Enterprise"
  514. {:cloud, :community} ->
  515. "EMQX"
  516. {:edge, :community} ->
  517. "EMQX Edge"
  518. end
  519. end
  520. defp emqx_schema_mod(:enterprise), do: :emqx_enterprise_conf_schema
  521. defp emqx_schema_mod(:community), do: :emqx_conf_schema
  522. defp bcrypt_dep() do
  523. if enable_bcrypt?(),
  524. do: [{:bcrypt, github: "emqx/erlang-bcrypt", tag: "0.6.0", override: true}],
  525. else: []
  526. end
  527. defp quicer_dep() do
  528. if enable_quicer?(),
  529. # in conflict with emqx and emqtt
  530. do: [{:quicer, github: "emqx/quic", tag: "0.0.9", override: true}],
  531. else: []
  532. end
  533. defp enable_bcrypt?() do
  534. not win32?()
  535. end
  536. defp enable_quicer?() do
  537. not Enum.any?([
  538. build_without_quic?(),
  539. win32?(),
  540. centos6?()
  541. ])
  542. end
  543. defp pkg_vsn() do
  544. %{edition_type: edition_type} = check_profile!()
  545. basedir = Path.dirname(__ENV__.file)
  546. script = Path.join(basedir, "pkg-vsn.sh")
  547. os_cmd(script, [Atom.to_string(edition_type)])
  548. end
  549. defp os_cmd(script, args) do
  550. {str, 0} = System.cmd("bash", [script | args])
  551. String.trim(str)
  552. end
  553. defp win32?(),
  554. do: match?({:win_32, _}, :os.type())
  555. defp centos6?() do
  556. case File.read("/etc/centos-release") do
  557. {:ok, "CentOS release 6" <> _} ->
  558. true
  559. _ ->
  560. false
  561. end
  562. end
  563. defp build_without_quic?() do
  564. opt = System.get_env("BUILD_WITHOUT_QUIC", "false")
  565. String.downcase(opt) != "false"
  566. end
  567. defp from_rebar_to_eex_template(str) do
  568. # we must not consider surrounding space in the template var name
  569. # because some help strings contain informative variables that
  570. # should not be interpolated, and those have no spaces.
  571. Regex.replace(
  572. ~r/\{\{ ([a-zA-Z0-9_]+) \}\}/,
  573. str,
  574. "<%= \\g{1} %>"
  575. )
  576. end
  577. defp build_info() do
  578. [
  579. build_info_arch: to_string(:erlang.system_info(:system_architecture)),
  580. build_info_wordsize: wordsize(),
  581. build_info_os: os_cmd("./scripts/get-distro.sh", []),
  582. build_info_erlang: otp_release(),
  583. build_info_elixir: System.version(),
  584. build_info_relform: System.get_env("EMQX_REL_FORM", "tgz")
  585. ]
  586. end
  587. # https://github.com/erlang/rebar3/blob/e3108ac187b88fff01eca6001a856283a3e0ec87/src/rebar_utils.erl#L142
  588. defp wordsize() do
  589. size =
  590. try do
  591. :erlang.system_info({:wordsize, :external})
  592. rescue
  593. ErlangError ->
  594. :erlang.system_info(:wordsize)
  595. end
  596. to_string(8 * size)
  597. end
  598. defp normalize_env!() do
  599. env =
  600. case Mix.env() do
  601. :dev ->
  602. :emqx
  603. env ->
  604. env
  605. end
  606. Mix.env(env)
  607. end
  608. # As from Erlang/OTP 17, the OTP release number corresponds to the
  609. # major OTP version number. No erlang:system_info() argument gives
  610. # the exact OTP version.
  611. # https://www.erlang.org/doc/man/erlang.html#system_info_otp_release
  612. # https://github.com/erlang/rebar3/blob/e3108ac187b88fff01eca6001a856283a3e0ec87/src/rebar_utils.erl#L572-L577
  613. defp otp_release() do
  614. major_version = System.otp_release()
  615. root_dir = to_string(:code.root_dir())
  616. [root_dir, "releases", major_version, "OTP_VERSION"]
  617. |> Path.join()
  618. |> File.read()
  619. |> case do
  620. {:error, _} ->
  621. major_version
  622. {:ok, version} ->
  623. version
  624. |> String.trim()
  625. |> String.split("**")
  626. |> List.first()
  627. end
  628. end
  629. end