mix.exs 20 KB

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