Skip to content

Commit 0bb305a

Browse files
Add environment variable for reusing Mix.install/2 installation (#13378)
1 parent 7291154 commit 0bb305a

File tree

3 files changed

+185
-43
lines changed

3 files changed

+185
-43
lines changed

lib/iex/lib/iex/helpers.ex

+8
Original file line numberDiff line numberDiff line change
@@ -105,6 +105,14 @@ defmodule IEx.Helpers do
105105

106106
Mix.installed?() ->
107107
Mix.in_install_project(fn ->
108+
# TODO: remove this once Mix requires Hex with the fix from
109+
# https://github.com/hexpm/hex/pull/1015
110+
# Context: Mix.install/1 starts :hex if necessary and stops
111+
# it afterwards. Calling compile here may require hex to be
112+
# started and that should happen automatically, but because
113+
# of a bug it is not (fixed in the linked PR).
114+
_ = Application.ensure_all_started(:hex)
115+
108116
do_recompile(options)
109117
# Just as with Mix.install/2 we clear all task invocations,
110118
# so that we can recompile the dependencies again next time

lib/mix/lib/mix.ex

+63-11
Original file line numberDiff line numberDiff line change
@@ -661,6 +661,13 @@ defmodule Mix do
661661
This function can only be called outside of a Mix project and only with the
662662
same dependencies in the given VM.
663663
664+
The `MIX_INSTALL_RESTORE_PROJECT_DIR` environment variable may be specified.
665+
It should point to a previous installation directory, which can be obtained
666+
with `Mix.install_project_dir/0` (after calling `Mix.install/2`). Using a
667+
restore dir may speed up the installation, since matching dependencies do
668+
not need be refetched nor recompiled. This environment variable is ignored
669+
if `:force` is enabled.
670+
664671
## Options
665672
666673
* `:force` - if `true`, runs with empty install cache. This is useful when you want
@@ -845,14 +852,14 @@ defmodule Mix do
845852
Application.put_all_env(config, persistent: true)
846853
System.put_env(system_env)
847854

848-
install_dir = install_dir(id)
855+
install_project_dir = install_project_dir(id)
849856

850857
if Keyword.fetch!(opts, :verbose) do
851-
Mix.shell().info("Mix.install/2 using #{install_dir}")
858+
Mix.shell().info("Mix.install/2 using #{install_project_dir}")
852859
end
853860

854861
if force? do
855-
File.rm_rf!(install_dir)
862+
File.rm_rf!(install_project_dir)
856863
end
857864

858865
dynamic_config = [
@@ -865,21 +872,28 @@ defmodule Mix do
865872

866873
started_apps = Application.started_applications()
867874
:ok = Mix.ProjectStack.push(@mix_install_project, config, "nofile")
868-
build_dir = Path.join(install_dir, "_build")
875+
build_dir = Path.join(install_project_dir, "_build")
869876
external_lockfile = expand_path(opts[:lockfile], deps, :lockfile, "mix.lock")
870877

871878
try do
872879
first_build? = not File.dir?(build_dir)
873-
File.mkdir_p!(install_dir)
874880

875-
File.cd!(install_dir, fn ->
881+
restore_dir = System.get_env("MIX_INSTALL_RESTORE_PROJECT_DIR")
882+
883+
if first_build? and restore_dir != nil and not force? do
884+
File.cp_r(restore_dir, install_project_dir)
885+
end
886+
887+
File.mkdir_p!(install_project_dir)
888+
889+
File.cd!(install_project_dir, fn ->
876890
if config_path do
877891
Mix.Task.rerun("loadconfig")
878892
end
879893

880894
cond do
881895
external_lockfile ->
882-
md5_path = Path.join(install_dir, "merge.lock.md5")
896+
md5_path = Path.join(install_project_dir, "merge.lock.md5")
883897

884898
old_md5 =
885899
case File.read(md5_path) do
@@ -890,7 +904,7 @@ defmodule Mix do
890904
new_md5 = external_lockfile |> File.read!() |> :erlang.md5()
891905

892906
if old_md5 != new_md5 do
893-
lockfile = Path.join(install_dir, "mix.lock")
907+
lockfile = Path.join(install_project_dir, "mix.lock")
894908
old_lock = Mix.Dep.Lock.read(lockfile)
895909
new_lock = Mix.Dep.Lock.read(external_lockfile)
896910
Mix.Dep.Lock.write(Map.merge(old_lock, new_lock), file: lockfile)
@@ -928,6 +942,10 @@ defmodule Mix do
928942
end
929943
end
930944

945+
if restore_dir do
946+
remove_leftover_deps(install_project_dir)
947+
end
948+
931949
Mix.State.put(:installed, {id, dynamic_config})
932950
:ok
933951
after
@@ -965,7 +983,29 @@ defmodule Mix do
965983
Path.join(app_dir, relative_path)
966984
end
967985

968-
defp install_dir(cache_id) do
986+
defp remove_leftover_deps(install_project_dir) do
987+
build_lib_dir = Path.join([install_project_dir, "_build", "dev", "lib"])
988+
deps_dir = Path.join(install_project_dir, "deps")
989+
990+
deps = File.ls!(build_lib_dir)
991+
992+
loaded_deps =
993+
for {app, _description, _version} <- Application.loaded_applications(),
994+
into: MapSet.new(),
995+
do: Atom.to_string(app)
996+
997+
# We want to keep :mix_install, but it has no application
998+
loaded_deps = MapSet.put(loaded_deps, "mix_install")
999+
1000+
for dep <- deps, not MapSet.member?(loaded_deps, dep) do
1001+
build_path = Path.join(build_lib_dir, dep)
1002+
File.rm_rf(build_path)
1003+
dep_path = Path.join(deps_dir, dep)
1004+
File.rm_rf(dep_path)
1005+
end
1006+
end
1007+
1008+
defp install_project_dir(cache_id) do
9691009
install_root =
9701010
System.get_env("MIX_INSTALL_DIR") ||
9711011
Path.join(Mix.Utils.mix_cache(), "installs")
@@ -996,9 +1036,9 @@ defmodule Mix do
9961036
{id, dynamic_config} ->
9971037
config = install_project_config(dynamic_config)
9981038

999-
install_dir = install_dir(id)
1039+
install_project_dir = install_project_dir(id)
10001040

1001-
File.cd!(install_dir, fn ->
1041+
File.cd!(install_project_dir, fn ->
10021042
:ok = Mix.ProjectStack.push(@mix_install_project, config, "nofile")
10031043

10041044
try do
@@ -1013,6 +1053,18 @@ defmodule Mix do
10131053
end
10141054
end
10151055

1056+
@doc """
1057+
Returns the directory where the current `Mix.install/2` project
1058+
resides.
1059+
"""
1060+
@spec install_project_dir() :: Path.t()
1061+
def install_project_dir() do
1062+
case Mix.State.get(:installed) do
1063+
{id, _dynamic_config} -> install_project_dir(id)
1064+
nil -> nil
1065+
end
1066+
end
1067+
10161068
@doc """
10171069
Returns whether `Mix.install/2` was called in the current node.
10181070
"""

lib/mix/test/mix_test.exs

+114-32
Original file line numberDiff line numberDiff line change
@@ -263,30 +263,26 @@ defmodule MixTest do
263263
[
264264
{:git_repo, git: fixture_path("git_repo")}
265265
],
266-
lockfile: lockfile,
267-
verbose: true
266+
lockfile: lockfile
268267
)
269268

270269
assert_received {:mix_shell, :info, ["* Getting git_repo " <> _]}
271-
assert_received {:mix_shell, :info, ["Mix.install/2 using " <> install_dir]}
272-
assert File.read!(Path.join(install_dir, "mix.lock")) =~ rev
273-
after
274-
purge([GitRepo, GitRepo.MixProject])
270+
271+
install_project_dir = Mix.install_project_dir()
272+
assert File.read!(Path.join(install_project_dir, "mix.lock")) =~ rev
275273
end
276274

277275
test ":lockfile merging", %{tmp_dir: tmp_dir} do
278276
[rev1, rev2 | _] = get_git_repo_revs("git_repo")
279277

280-
Mix.install(
281-
[
282-
{:git_repo, git: fixture_path("git_repo")}
283-
],
284-
verbose: true
285-
)
278+
Mix.install([
279+
{:git_repo, git: fixture_path("git_repo")}
280+
])
286281

287282
assert_received {:mix_shell, :info, ["* Getting git_repo " <> _]}
288-
assert_received {:mix_shell, :info, ["Mix.install/2 using " <> install_dir]}
289-
assert File.read!(Path.join(install_dir, "mix.lock")) =~ rev1
283+
284+
install_project_dir = Mix.install_project_dir()
285+
assert File.read!(Path.join(install_project_dir, "mix.lock")) =~ rev1
290286

291287
Mix.Project.push(GitApp)
292288
lockfile = Path.join(tmp_dir, "lock")
@@ -300,9 +296,7 @@ defmodule MixTest do
300296
lockfile: lockfile
301297
)
302298

303-
assert File.read!(Path.join(install_dir, "mix.lock")) =~ rev1
304-
after
305-
purge([GitRepo, GitRepo.MixProject])
299+
assert File.read!(Path.join(install_project_dir, "mix.lock")) =~ rev1
306300
end
307301

308302
test ":lockfile with application name", %{tmp_dir: tmp_dir} do
@@ -318,15 +312,12 @@ defmodule MixTest do
318312
{:install_test, path: Path.join(tmp_dir, "install_test")},
319313
{:git_repo, git: fixture_path("git_repo")}
320314
],
321-
lockfile: :install_test,
322-
verbose: true
315+
lockfile: :install_test
323316
)
324317

325318
assert_received {:mix_shell, :info, ["* Getting git_repo " <> _]}
326-
assert_received {:mix_shell, :info, ["Mix.install/2 using " <> install_dir]}
327-
assert File.read!(Path.join(install_dir, "mix.lock")) =~ rev
328-
after
329-
purge([GitRepo, GitRepo.MixProject])
319+
install_project_dir = Mix.install_project_dir()
320+
assert File.read!(Path.join(install_project_dir, "mix.lock")) =~ rev
330321
end
331322

332323
test ":lockfile that does not exist" do
@@ -335,6 +326,73 @@ defmodule MixTest do
335326
end
336327
end
337328

329+
test "restore dir", %{tmp_dir: tmp_dir} do
330+
with_cleanup(fn ->
331+
Mix.install([
332+
{:git_repo, git: fixture_path("git_repo")}
333+
])
334+
335+
assert_received {:mix_shell, :info, ["* Getting git_repo " <> _]}
336+
assert_received {:mix_shell, :info, ["==> git_repo"]}
337+
assert_received {:mix_shell, :info, ["Compiling 1 file (.ex)"]}
338+
assert_received {:mix_shell, :info, ["Generated git_repo app"]}
339+
refute_received _
340+
341+
install_project_dir = Mix.install_project_dir()
342+
build_lib_path = Path.join([install_project_dir, "_build", "dev", "lib"])
343+
deps_path = Path.join([install_project_dir, "deps"])
344+
345+
assert File.ls!(build_lib_path) |> Enum.sort() == ["git_repo", "mix_install"]
346+
assert File.ls!(deps_path) == ["git_repo"]
347+
348+
System.put_env("MIX_INSTALL_RESTORE_PROJECT_DIR", install_project_dir)
349+
end)
350+
351+
# Adding a dependency
352+
353+
with_cleanup(fn ->
354+
Mix.install([
355+
{:git_repo, git: fixture_path("git_repo")},
356+
{:install_test, path: Path.join(tmp_dir, "install_test")}
357+
])
358+
359+
assert_received {:mix_shell, :info, ["==> install_test"]}
360+
assert_received {:mix_shell, :info, ["Compiling 2 files (.ex)"]}
361+
assert_received {:mix_shell, :info, ["Generated install_test app"]}
362+
refute_received _
363+
364+
install_project_dir = Mix.install_project_dir()
365+
build_lib_path = Path.join([install_project_dir, "_build", "dev", "lib"])
366+
deps_path = Path.join([install_project_dir, "deps"])
367+
368+
assert File.ls!(build_lib_path) |> Enum.sort() ==
369+
["git_repo", "install_test", "mix_install"]
370+
371+
assert File.ls!(deps_path) == ["git_repo"]
372+
373+
System.put_env("MIX_INSTALL_RESTORE_PROJECT_DIR", install_project_dir)
374+
end)
375+
376+
# Removing a dependency
377+
378+
with_cleanup(fn ->
379+
Mix.install([
380+
{:install_test, path: Path.join(tmp_dir, "install_test")}
381+
])
382+
383+
refute_received _
384+
385+
install_project_dir = Mix.install_project_dir()
386+
build_lib_path = Path.join([install_project_dir, "_build", "dev", "lib"])
387+
deps_path = Path.join([install_project_dir, "deps"])
388+
389+
assert File.ls!(build_lib_path) |> Enum.sort() == ["install_test", "mix_install"]
390+
assert File.ls!(deps_path) == []
391+
end)
392+
after
393+
System.delete_env("MIX_INSTALL_RESTORE_PROJECT_DIR")
394+
end
395+
338396
test "installed?", %{tmp_dir: tmp_dir} do
339397
refute Mix.installed?()
340398

@@ -380,15 +438,7 @@ defmodule MixTest do
380438

381439
on_exit(fn ->
382440
:code.set_path(path)
383-
purge([InstallTest, InstallTest.MixProject, InstallTest.Protocol])
384-
385-
ExUnit.CaptureLog.capture_log(fn ->
386-
Application.stop(:git_repo)
387-
Application.unload(:git_repo)
388-
389-
Application.stop(:install_test)
390-
Application.unload(:install_test)
391-
end)
441+
cleanup_deps()
392442
end)
393443

394444
Mix.State.put(:installed, nil)
@@ -424,5 +474,37 @@ defmodule MixTest do
424474

425475
[tmp_dir: tmp_dir]
426476
end
477+
478+
defp with_cleanup(fun) do
479+
path = :code.get_path()
480+
481+
try do
482+
fun.()
483+
after
484+
:code.set_path(path)
485+
cleanup_deps()
486+
487+
Mix.State.clear_cache()
488+
Mix.State.put(:installed, nil)
489+
end
490+
end
491+
492+
defp cleanup_deps() do
493+
purge([
494+
GitRepo,
495+
GitRepo.MixProject,
496+
InstallTest,
497+
InstallTest.MixProject,
498+
InstallTest.Protocol
499+
])
500+
501+
ExUnit.CaptureLog.capture_log(fn ->
502+
Application.stop(:git_repo)
503+
Application.unload(:git_repo)
504+
505+
Application.stop(:install_test)
506+
Application.unload(:install_test)
507+
end)
508+
end
427509
end
428510
end

0 commit comments

Comments
 (0)