igniter/lib/deps.ex
2024-06-12 18:22:08 -04:00

118 lines
3.8 KiB
Elixir

defmodule Igniter.Deps do
@moduledoc "Codemods and utilities for managing dependencies declared in mix.exs"
require Igniter.Code.Common
alias Igniter.Code.Common
alias Sourceror.Zipper
def add_dependency(igniter, name, version, yes? \\ false) do
if name in List.wrap(igniter.assigns[:manually_installed]) do
igniter
else
case get_dependency_declaration(igniter, name) do
nil ->
do_add_dependency(igniter, name, version)
current ->
desired = "`{#{inspect(name)}, #{inspect(version)}}`"
current = "`#{current}`"
if desired == current do
igniter
else
if yes? ||
Mix.shell().yes?("""
Dependency #{name} is already in mix.exs. Should we replace it?
Desired: #{desired}
Found: #{current}
""") do
igniter
|> remove_dependency(name)
|> do_add_dependency(name, version)
else
igniter
end
end
end
end
end
def get_dependency_declaration(igniter, name) do
zipper =
igniter
|> Igniter.include_existing_elixir_file("mix.exs")
|> Map.get(:rewrite)
|> Rewrite.source!("mix.exs")
|> Rewrite.Source.get(:quoted)
|> Zipper.zip()
with {:ok, zipper} <- Igniter.Code.Module.move_to_module_using(zipper, Mix.Project),
{:ok, zipper} <- Igniter.Code.Module.move_to_defp(zipper, :deps, 0),
true <- Common.node_matches_pattern?(zipper, value when is_list(value)),
{:ok, current_declaration} <-
Igniter.Code.List.move_to_list_item(zipper, fn item ->
if Igniter.Code.Tuple.tuple?(item) do
case Igniter.Code.Tuple.tuple_elem(item, 0) do
{:ok, first_elem} ->
Common.node_matches_pattern?(first_elem, ^name)
:error ->
false
end
end
end) do
current_declaration
|> Zipper.subtree()
|> Zipper.node()
|> Sourceror.to_string()
else
_ ->
nil
end
end
defp remove_dependency(igniter, name) do
igniter
|> Igniter.update_elixir_file("mix.exs", fn zipper ->
with {:ok, zipper} <- Igniter.Code.Module.move_to_module_using(zipper, Mix.Project),
{:ok, zipper} <- Igniter.Code.Module.move_to_defp(zipper, :deps, 0),
true <- Common.node_matches_pattern?(zipper, value when is_list(value)),
current_declaration_index when not is_nil(current_declaration_index) <-
Igniter.Code.List.find_list_item_index(zipper, fn item ->
if Igniter.Code.Tuple.tuple?(item) do
case Igniter.Code.Tuple.tuple_elem(item, 0) do
{:ok, first_elem} ->
Common.node_matches_pattern?(first_elem, ^name)
:error ->
false
end
end
end) do
Igniter.Code.List.remove_index(zipper, current_declaration_index)
else
_ ->
{:error, "Failed to remove dependency #{inspect(name)}"}
end
end)
end
defp do_add_dependency(igniter, name, version) do
igniter
|> Igniter.update_elixir_file("mix.exs", fn zipper ->
with {:ok, zipper} <- Igniter.Code.Module.move_to_module_using(zipper, Mix.Project),
{:ok, zipper} <- Igniter.Code.Module.move_to_defp(zipper, :deps, 0),
true <- Common.node_matches_pattern?(zipper, value when is_list(value)) do
quoted =
quote do
{unquote(name), unquote(version)}
end
Igniter.Code.List.prepend_to_list(zipper, quoted)
else
_ ->
{:error, "Failed to add dependency #{inspect({inspect(name), inspect(version)})}"}
end
end)
end
end