igniter/lib/config.ex

171 lines
5 KiB
Elixir
Raw Normal View History

2024-05-28 15:30:41 +12:00
defmodule Igniter.Config do
@moduledoc "Codemods and utilities for configuring Elixir applications."
2024-05-28 15:30:41 +12:00
require Igniter.Common
alias Igniter.Common
alias Sourceror.Zipper
2024-06-11 01:58:20 +12:00
def configure_new(igniter, file_path, app_name, config_path, value) do
configure(igniter, file_path, app_name, config_path, value, & &1)
end
2024-05-28 15:30:41 +12:00
def configure(igniter, file_path, app_name, config_path, value, updater \\ nil) do
2024-06-11 01:58:20 +12:00
file_contents =
if file_path == "config.exs" do
"""
import Config
# Import environment specific config. This must remain at the bottom
# of this file so it overrides the configuration defined above.
import_config "\#{config_env()}.exs"
"""
else
"import Config\n"
end
2024-05-28 15:30:41 +12:00
file_path = Path.join("config", file_path)
config_path = List.wrap(config_path)
2024-06-11 01:58:20 +12:00
value =
case value do
{:code, value} -> value
value -> Macro.escape(value)
end
2024-05-28 15:30:41 +12:00
updater = updater || fn zipper -> Zipper.replace(zipper, value) end
igniter
2024-06-11 01:58:20 +12:00
|> Igniter.include_or_create_elixir_file(file_path, file_contents)
2024-06-04 05:13:49 +12:00
|> Igniter.update_elixir_file(file_path, fn zipper ->
2024-06-11 01:58:20 +12:00
case Zipper.find(zipper, fn
{:import, _, [Config]} ->
true
2024-05-28 15:30:41 +12:00
2024-06-11 01:58:20 +12:00
{:import, _, [{:__aliases__, _, [:Config]}]} ->
true
2024-06-11 01:58:20 +12:00
_ ->
false
end) do
nil ->
{:error, "No call to `import Config` found in configuration file"}
2024-05-28 15:30:41 +12:00
2024-06-11 01:58:20 +12:00
zipper ->
modify_configuration_code(zipper, config_path, app_name, value, updater)
2024-05-28 15:30:41 +12:00
end
end)
end
2024-06-11 01:58:20 +12:00
def modify_configuration_code(zipper, config_path, app_name, value, updater \\ nil) do
updater = updater || fn zipper -> Zipper.replace(zipper, value) end
case try_update_three_arg(zipper, config_path, app_name, updater) do
{:ok, zipper} ->
zipper
:error ->
case try_update_two_arg(zipper, config_path, app_name, value, updater) do
{:ok, zipper} ->
zipper
:error ->
[first | rest] = config_path
config =
{:config, [], [app_name, [{first, Igniter.Common.keywordify(rest, value)}]]}
case Common.move_to_function_call_in_current_scope(
zipper,
:import,
1,
fn function_call ->
Common.argument_matches_predicate?(
function_call,
0,
&Common.equal_modules?(&1, Config)
)
end
) do
{:ok, zipper} ->
zipper
|> Zipper.right()
|> case do
nil ->
Igniter.Common.add_code(zipper, config)
zipper ->
Igniter.Common.add_code(zipper, config, :before)
end
end
end
end
end
def configures?(zipper, config_path, app_name) do
if Enum.count(config_path) == 1 do
config_item = Enum.at(config_path, 0)
case Common.move_to_function_call_in_current_scope(zipper, :config, 3, fn function_call ->
Common.argument_matches_pattern?(function_call, 0, ^app_name) &&
Common.argument_matches_pattern?(function_call, 1, ^config_item)
end) do
:error ->
false
{:ok, _zipper} ->
true
end
else
case Common.move_to_function_call_in_current_scope(zipper, :config, 2, fn function_call ->
Common.argument_matches_pattern?(function_call, 0, ^app_name)
end) do
:error ->
:error
{:ok, zipper} ->
Common.argument_matches_predicate?(zipper, 1, fn zipper ->
Igniter.Common.keyword_has_path?(zipper, config_path)
end)
end
end
end
2024-05-28 15:30:41 +12:00
defp try_update_three_arg(zipper, config_path, app_name, updater) do
if Enum.count(config_path) == 1 do
config_item = Enum.at(config_path, 0)
case Common.move_to_function_call_in_current_scope(zipper, :config, 3, fn function_call ->
Common.argument_matches_pattern?(function_call, 0, ^app_name) &&
Common.argument_matches_pattern?(function_call, 1, ^config_item)
2024-05-28 15:30:41 +12:00
end) do
:error ->
2024-05-28 15:30:41 +12:00
:error
{:ok, zipper} ->
Common.update_nth_argument(zipper, 2, updater)
2024-05-28 15:30:41 +12:00
end
else
:error
end
end
defp try_update_two_arg(zipper, config_path, app_name, value, updater) do
case Common.move_to_function_call_in_current_scope(zipper, :config, 2, fn function_call ->
2024-05-28 15:30:41 +12:00
Common.argument_matches_pattern?(function_call, 0, ^app_name)
end) do
:error ->
2024-05-28 15:30:41 +12:00
:error
{:ok, zipper} ->
2024-05-28 15:30:41 +12:00
Common.update_nth_argument(zipper, 1, fn zipper ->
case Igniter.Common.put_in_keyword(zipper, config_path, value, updater) do
{:ok, new_zipper} ->
new_zipper
2024-05-28 15:30:41 +12:00
_ ->
zipper
end
end)
2024-05-28 15:30:41 +12:00
end
end
end