Skip to content

Commit 421dc7c

Browse files
author
Rodrigo Álvarez
committed
Add basic Gleam integration with Mix
1 parent 8e43008 commit 421dc7c

File tree

7 files changed

+76
-11
lines changed

7 files changed

+76
-11
lines changed

lib/mix/lib/mix/dep.ex

+9-2
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ defmodule Mix.Dep do
2727
* `top_level` - true if dependency was defined in the top-level project
2828
2929
* `manager` - the project management, possible values:
30-
`:rebar3` | `:mix` | `:make` | `nil`
30+
`:rebar3` | `:mix` | `:make` | `:gleam' | `nil`
3131
3232
* `from` - path to the file where the dependency was defined
3333
@@ -73,7 +73,7 @@ defmodule Mix.Dep do
7373
status: {:ok, String.t() | nil} | atom | tuple,
7474
opts: keyword,
7575
top_level: boolean,
76-
manager: :rebar3 | :mix | :make | nil,
76+
manager: :rebar3 | :mix | :make | :gleam | nil,
7777
from: String.t(),
7878
extra: term,
7979
system_env: keyword
@@ -535,6 +535,13 @@ defmodule Mix.Dep do
535535
manager == :make
536536
end
537537

538+
@doc """
539+
Returns `true` if dependency is a Gleam project.
540+
"""
541+
def gleam?(%Mix.Dep{manager: manager}) do
542+
manager == :gleam
543+
end
544+
538545
## Helpers
539546

540547
defp mix_env_var do

lib/mix/lib/mix/dep/converger.ex

+1-1
Original file line numberDiff line numberDiff line change
@@ -426,7 +426,7 @@ defmodule Mix.Dep.Converger do
426426
%{other | manager: sort_manager(other_manager, manager, in_upper?)}
427427
end
428428

429-
@managers [:mix, :rebar3, :make]
429+
@managers [:mix, :rebar3, :make, :gleam]
430430

431431
defp sort_manager(other_manager, manager, true) do
432432
other_manager || manager

lib/mix/lib/mix/dep/loader.ex

+23-4
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88
defmodule Mix.Dep.Loader do
99
@moduledoc false
1010

11-
import Mix.Dep, only: [ok?: 1, mix?: 1, rebar?: 1, make?: 1]
11+
import Mix.Dep, only: [ok?: 1, mix?: 1, rebar?: 1, make?: 1, gleam?: 1]
1212

1313
@doc """
1414
Gets all direct children of the current `Mix.Project`
@@ -84,9 +84,9 @@ defmodule Mix.Dep.Loader do
8484
def load(%Mix.Dep{manager: manager, scm: scm, opts: opts} = dep, children, locked?) do
8585
# The manager for a child dependency is set based on the following rules:
8686
# 1. Set in dependency definition
87-
# 2. From SCM, so that Hex dependencies of a rebar project can be compiled with mix
87+
# 2. From SCM, so that Hex dependencies of a rebar/gleam project can be compiled with mix
8888
# 3. From the parent dependency, used for rebar dependencies from git
89-
# 4. Inferred from files in dependency (mix.exs, rebar.config, Makefile)
89+
# 4. Inferred from files in dependency (mix.exs, rebar.config, Makefile, gleam.toml)
9090
manager = opts[:manager] || scm_manager(scm, opts) || manager || infer_manager(opts[:dest])
9191
dep = %{dep | manager: manager, status: scm_status(scm, opts)}
9292

@@ -106,6 +106,9 @@ defmodule Mix.Dep.Loader do
106106
make?(dep) ->
107107
make_dep(dep)
108108

109+
gleam?(dep) ->
110+
gleam_dep(dep, children, locked?)
111+
109112
true ->
110113
{dep, []}
111114
end
@@ -220,7 +223,7 @@ defmodule Mix.Dep.Loader do
220223

221224
# Note that we ignore Make dependencies because the
222225
# file based heuristic will always figure it out.
223-
@scm_managers ~w(mix rebar3)a
226+
@scm_managers ~w(mix rebar3 gleam)a
224227

225228
defp scm_manager(scm, opts) do
226229
managers = scm.managers(opts)
@@ -246,6 +249,9 @@ defmodule Mix.Dep.Loader do
246249
any_of?(dest, ["Makefile", "Makefile.win"]) ->
247250
:make
248251

252+
any_of?(dest, ["gleam.toml"]) ->
253+
:gleam
254+
249255
true ->
250256
nil
251257
end
@@ -361,6 +367,19 @@ defmodule Mix.Dep.Loader do
361367
{dep, []}
362368
end
363369

370+
defp gleam_dep(%Mix.Dep{opts: opts} = dep, children, locked?) do
371+
deps =
372+
if children do
373+
Enum.map(children, &to_dep(&1, opts[:dest], _manager = nil, locked?))
374+
else
375+
config = File.cd!(opts[:dest], fn -> Mix.Gleam.load_config(".") end)
376+
from = Path.join(opts[:dest], "gleam.toml")
377+
Enum.map(config[:deps], &to_dep(&1, from, _manager = nil, locked?))
378+
end
379+
380+
{dep, deps}
381+
end
382+
364383
defp mix_children(config, locked?, opts) do
365384
from = Mix.Project.project_file()
366385

lib/mix/lib/mix/task.compiler.ex

+1-1
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,7 @@ defmodule Mix.Task.Compiler do
8080
* `:scm` - the SCM module of the dependency.
8181
8282
* `:manager` - the dependency project management, possible values:
83-
`:rebar3`, `:mix`, `:make`, `nil`.
83+
`:rebar3`, `:mix`, `:make`, `:gleam`, `nil`.
8484
8585
* `:os_pid` - the operating system PID of the process that run
8686
the compilation. The value is a string and it can be compared

lib/mix/lib/mix/tasks/deps.compile.ex

+18-1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ defmodule Mix.Tasks.Deps.Compile do
2222
* `Makefile.win`- invokes `nmake /F Makefile.win` (only on Windows)
2323
* `Makefile` - invokes `gmake` on DragonFlyBSD, FreeBSD, NetBSD, and OpenBSD,
2424
invokes `make` on any other operating system (except on Windows)
25+
* `gleam.toml` - invokes `gleam export`
2526
2627
The compilation can be customized by passing a `compile` option
2728
in the dependency:
@@ -101,9 +102,12 @@ defmodule Mix.Tasks.Deps.Compile do
101102
dep.manager == :rebar3 ->
102103
do_rebar3(dep, config)
103104

105+
dep.manager == :gleam ->
106+
do_gleam(dep, config)
107+
104108
true ->
105109
shell.error(
106-
"Could not compile #{inspect(app)}, no \"mix.exs\", \"rebar.config\" or \"Makefile\" " <>
110+
"Could not compile #{inspect(app)}, no \"mix.exs\", \"rebar.config\", \"Makefile\" or \"gleam.toml\" " <>
107111
"(pass :compile as an option to customize compilation, set it to \"false\" to do nothing)"
108112
)
109113

@@ -296,6 +300,19 @@ defmodule Mix.Tasks.Deps.Compile do
296300
true
297301
end
298302

303+
defp do_gleam(%Mix.Dep{opts: opts} = dep, config) do
304+
lib = Path.join(Mix.Project.build_path(), "lib")
305+
out = opts[:build]
306+
package = opts[:dest]
307+
308+
command =
309+
"gleam compile-package --target erlang " <>
310+
"--package \"#{package}\" --out \"#{out}\" --lib \"#{lib}\""
311+
312+
shell_cmd!(dep, config, command)
313+
Code.prepend_path(Path.join(out, "ebin"), cache: true)
314+
end
315+
299316
defp make_command(dep) do
300317
makefile_win? = makefile_win?(dep)
301318

lib/mix/lib/mix/tasks/deps.ex

+2-2
Original file line numberDiff line numberDiff line change
@@ -101,10 +101,10 @@ defmodule Mix.Tasks.Deps do
101101
* `:override` - if set to `true` the dependency will override any other
102102
definitions of itself by other dependencies
103103
104-
* `:manager` - Mix can also compile Rebar3 and makefile projects
104+
* `:manager` - Mix can also compile Rebar3, makefile and gleam projects
105105
and can fetch sub dependencies of Rebar3 projects. Mix will
106106
try to infer the type of project but it can be overridden with this
107-
option by setting it to `:mix`, `:rebar3`, or `:make`. In case
107+
option by setting it to `:mix`, `:rebar3`, `:make` or `:gleam`. In case
108108
there are conflicting definitions, the first manager in the list above
109109
will be picked up. For example, if a dependency is found with `:rebar3`
110110
as a manager in different part of the trees, `:rebar3` will be automatically

lib/mix/test/mix/gleam_test.exs

+22
Original file line numberDiff line numberDiff line change
@@ -33,4 +33,26 @@ defmodule Mix.GleamTest do
3333
]
3434
end
3535
end
36+
37+
describe "integration with Mix" do
38+
test "gets and compiles dependencies" do
39+
in_tmp("get and compile dependencies", fn ->
40+
Mix.Project.push(GleamAsDep)
41+
42+
Mix.Tasks.Deps.Get.run([])
43+
assert_received {:mix_shell, :info, ["* Getting gleam_stdlib " <> _]}
44+
assert_received {:mix_shell, :info, ["* Getting gleam_otp " <> _]}
45+
assert_received {:mix_shell, :info, ["* Getting gleeunit " <> _]}
46+
47+
Mix.Tasks.Deps.Compile.run([])
48+
assert :gleam_dep.main()
49+
assert :gleam@int.to_string(1) == "1"
50+
51+
assert Enum.any?(load_paths, &String.ends_with?(&1, "gleam_dep/ebin"))
52+
assert Enum.any?(load_paths, &String.ends_with?(&1, "gleam_stdlib/ebin"))
53+
# Dep of a dep
54+
assert Enum.any?(load_paths, &String.ends_with?(&1, "gleam_erlang/ebin"))
55+
end)
56+
end
57+
end
3658
end

0 commit comments

Comments
 (0)