2019-07-09 22:13:23 -07:00
|
|
|
# Pleroma: A lightweight social networking server
|
2021-01-12 22:49:20 -08:00
|
|
|
# Copyright © 2017-2021 Pleroma Authors <https://pleroma.social/>
|
2019-07-09 22:13:23 -07:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2019-06-14 08:45:05 -07:00
|
|
|
defmodule Mix.Tasks.Pleroma.Config do
|
|
|
|
use Mix.Task
|
2020-01-17 00:45:44 -08:00
|
|
|
|
2020-12-06 07:02:30 -08:00
|
|
|
import Ecto.Query
|
2019-06-19 16:05:19 -07:00
|
|
|
import Mix.Pleroma
|
2020-01-17 00:45:44 -08:00
|
|
|
|
2020-01-15 21:50:27 -08:00
|
|
|
alias Pleroma.ConfigDB
|
2019-06-14 08:45:05 -07:00
|
|
|
alias Pleroma.Repo
|
2020-01-17 00:45:44 -08:00
|
|
|
|
2019-06-14 08:45:05 -07:00
|
|
|
@shortdoc "Manages the location of the config"
|
2019-10-03 04:12:57 -07:00
|
|
|
@moduledoc File.read!("docs/administration/CLI_tasks/config.md")
|
2019-09-29 01:17:38 -07:00
|
|
|
|
2019-06-14 08:45:05 -07:00
|
|
|
def run(["migrate_to_db"]) do
|
2020-12-02 10:33:34 -08:00
|
|
|
check_configdb(fn ->
|
|
|
|
start_pleroma()
|
|
|
|
migrate_to_db()
|
|
|
|
end)
|
2019-06-14 08:45:05 -07:00
|
|
|
end
|
|
|
|
|
2019-09-29 01:17:38 -07:00
|
|
|
def run(["migrate_from_db" | options]) do
|
2020-12-02 10:33:34 -08:00
|
|
|
check_configdb(fn ->
|
|
|
|
start_pleroma()
|
2019-06-14 08:45:05 -07:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
{opts, _} =
|
|
|
|
OptionParser.parse!(options,
|
|
|
|
strict: [env: :string, delete: :boolean],
|
|
|
|
aliases: [d: :delete]
|
|
|
|
)
|
2019-06-14 08:45:05 -07:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
migrate_from_db(opts)
|
|
|
|
end)
|
2020-01-18 05:55:33 -08:00
|
|
|
end
|
|
|
|
|
2020-11-25 10:44:06 -08:00
|
|
|
def run(["dump"]) do
|
2020-12-02 10:33:34 -08:00
|
|
|
check_configdb(fn ->
|
|
|
|
start_pleroma()
|
2020-11-25 10:44:06 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
header = config_header()
|
2020-11-25 10:44:06 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
settings =
|
|
|
|
ConfigDB
|
|
|
|
|> Repo.all()
|
|
|
|
|> Enum.sort()
|
2020-11-25 10:44:06 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
unless settings == [] do
|
|
|
|
shell_info("#{header}")
|
2020-11-27 12:44:05 -08:00
|
|
|
|
2020-12-06 07:02:30 -08:00
|
|
|
Enum.each(settings, &dump(&1))
|
2020-12-02 10:33:34 -08:00
|
|
|
else
|
|
|
|
shell_error("No settings in ConfigDB.")
|
|
|
|
end
|
|
|
|
end)
|
2020-11-25 10:44:06 -08:00
|
|
|
end
|
|
|
|
|
2020-11-27 11:55:43 -08:00
|
|
|
def run(["dump", group, key]) do
|
2020-12-02 10:33:34 -08:00
|
|
|
check_configdb(fn ->
|
|
|
|
start_pleroma()
|
2020-11-25 10:44:06 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
group = maybe_atomize(group)
|
|
|
|
key = maybe_atomize(key)
|
2020-11-27 12:44:05 -08:00
|
|
|
|
2020-12-03 08:34:23 -08:00
|
|
|
group
|
2020-12-06 07:02:30 -08:00
|
|
|
|> ConfigDB.get_by_group_and_key(key)
|
|
|
|
|> dump()
|
2020-12-02 10:33:34 -08:00
|
|
|
end)
|
2020-11-27 11:55:43 -08:00
|
|
|
end
|
|
|
|
|
|
|
|
def run(["dump", group]) do
|
2020-12-02 10:33:34 -08:00
|
|
|
check_configdb(fn ->
|
|
|
|
start_pleroma()
|
2020-11-27 11:55:43 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
group = maybe_atomize(group)
|
2020-11-27 12:44:05 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
dump_group(group)
|
|
|
|
end)
|
2020-11-25 10:44:06 -08:00
|
|
|
end
|
|
|
|
|
2020-11-25 15:49:36 -08:00
|
|
|
def run(["groups"]) do
|
2020-12-02 10:33:34 -08:00
|
|
|
check_configdb(fn ->
|
|
|
|
start_pleroma()
|
2020-11-29 11:29:36 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
groups =
|
|
|
|
ConfigDB
|
2020-12-06 07:02:30 -08:00
|
|
|
|> distinct([c], true)
|
|
|
|
|> select([c], c.group)
|
2020-12-02 10:33:34 -08:00
|
|
|
|> Repo.all()
|
|
|
|
|
|
|
|
if length(groups) > 0 do
|
|
|
|
shell_info("The following configuration groups are set in ConfigDB:\r\n")
|
|
|
|
groups |> Enum.each(fn x -> shell_info("- #{x}") end)
|
|
|
|
shell_info("\r\n")
|
|
|
|
end
|
|
|
|
end)
|
2020-11-27 10:33:55 -08:00
|
|
|
end
|
|
|
|
|
2020-12-02 14:34:23 -08:00
|
|
|
def run(["reset", "--force"]) do
|
|
|
|
check_configdb(fn ->
|
|
|
|
start_pleroma()
|
|
|
|
truncatedb()
|
|
|
|
shell_info("The ConfigDB settings have been removed from the database.")
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
def run(["reset"]) do
|
2020-12-02 10:33:34 -08:00
|
|
|
check_configdb(fn ->
|
|
|
|
start_pleroma()
|
2020-11-25 15:31:44 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
shell_info("The following settings will be permanently removed:")
|
2020-11-27 14:32:46 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
ConfigDB
|
|
|
|
|> Repo.all()
|
|
|
|
|> Enum.sort()
|
|
|
|
|> Enum.each(&dump(&1))
|
2020-11-27 14:32:46 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
shell_error("\nTHIS CANNOT BE UNDONE!")
|
2020-11-27 14:32:46 -08:00
|
|
|
|
2020-12-02 14:34:23 -08:00
|
|
|
if shell_prompt("Are you sure you want to continue?", "n") in ~w(Yn Y y) do
|
|
|
|
truncatedb()
|
2020-11-25 15:31:44 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
shell_info("The ConfigDB settings have been removed from the database.")
|
|
|
|
else
|
|
|
|
shell_error("No changes made.")
|
|
|
|
end
|
|
|
|
end)
|
2020-11-25 15:31:44 -08:00
|
|
|
end
|
|
|
|
|
2020-12-02 14:24:32 -08:00
|
|
|
def run(["delete", "--force", group, key]) do
|
|
|
|
start_pleroma()
|
2020-11-27 12:44:05 -08:00
|
|
|
|
2020-12-02 14:24:32 -08:00
|
|
|
group = maybe_atomize(group)
|
|
|
|
key = maybe_atomize(key)
|
2020-11-27 12:02:45 -08:00
|
|
|
|
2020-12-07 09:45:56 -08:00
|
|
|
with true <- key_exists?(group, key) do
|
|
|
|
shell_info("The following settings will be removed from ConfigDB:\n")
|
|
|
|
|
|
|
|
group
|
|
|
|
|> ConfigDB.get_by_group_and_key(key)
|
|
|
|
|> dump()
|
|
|
|
|
|
|
|
delete_key(group, key)
|
|
|
|
else
|
|
|
|
_ ->
|
|
|
|
shell_error("No settings in ConfigDB for #{inspect(group)}, #{inspect(key)}. Aborting.")
|
|
|
|
end
|
2020-12-02 14:24:32 -08:00
|
|
|
end
|
2020-11-27 12:02:45 -08:00
|
|
|
|
2020-12-02 14:24:32 -08:00
|
|
|
def run(["delete", "--force", group]) do
|
|
|
|
start_pleroma()
|
2020-12-02 10:33:34 -08:00
|
|
|
|
2020-12-02 14:24:32 -08:00
|
|
|
group = maybe_atomize(group)
|
|
|
|
|
2020-12-04 09:30:48 -08:00
|
|
|
with true <- group_exists?(group) do
|
|
|
|
shell_info("The following settings will be removed from ConfigDB:\n")
|
|
|
|
dump_group(group)
|
|
|
|
delete_group(group)
|
|
|
|
else
|
|
|
|
_ -> shell_error("No settings in ConfigDB for #{inspect(group)}. Aborting.")
|
|
|
|
end
|
2020-12-02 10:33:34 -08:00
|
|
|
end
|
|
|
|
|
|
|
|
def run(["delete", group, key]) do
|
2020-12-02 14:24:32 -08:00
|
|
|
start_pleroma()
|
2020-12-02 10:33:34 -08:00
|
|
|
|
2020-12-02 14:24:32 -08:00
|
|
|
group = maybe_atomize(group)
|
|
|
|
key = maybe_atomize(key)
|
2020-11-25 10:44:06 -08:00
|
|
|
|
2020-12-07 09:45:56 -08:00
|
|
|
with true <- key_exists?(group, key) do
|
|
|
|
shell_info("The following settings will be removed from ConfigDB:\n")
|
|
|
|
|
|
|
|
group
|
|
|
|
|> ConfigDB.get_by_group_and_key(key)
|
|
|
|
|> dump()
|
|
|
|
|
|
|
|
if shell_prompt("Are you sure you want to continue?", "n") in ~w(Yn Y y) do
|
|
|
|
delete_key(group, key)
|
|
|
|
else
|
|
|
|
shell_error("No changes made.")
|
|
|
|
end
|
2020-12-02 14:24:32 -08:00
|
|
|
else
|
2020-12-07 09:45:56 -08:00
|
|
|
_ ->
|
|
|
|
shell_error("No settings in ConfigDB for #{inspect(group)}, #{inspect(key)}. Aborting.")
|
2020-12-02 14:24:32 -08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def run(["delete", group]) do
|
|
|
|
start_pleroma()
|
|
|
|
|
|
|
|
group = maybe_atomize(group)
|
|
|
|
|
2020-12-04 09:30:48 -08:00
|
|
|
with true <- group_exists?(group) do
|
|
|
|
shell_info("The following settings will be removed from ConfigDB:\n")
|
|
|
|
dump_group(group)
|
|
|
|
|
|
|
|
if shell_prompt("Are you sure you want to continue?", "n") in ~w(Yn Y y) do
|
|
|
|
delete_group(group)
|
|
|
|
else
|
|
|
|
shell_error("No changes made.")
|
|
|
|
end
|
2020-12-02 14:24:32 -08:00
|
|
|
else
|
2020-12-04 09:30:48 -08:00
|
|
|
_ -> shell_error("No settings in ConfigDB for #{inspect(group)}. Aborting.")
|
2020-12-02 14:24:32 -08:00
|
|
|
end
|
2020-11-25 10:44:06 -08:00
|
|
|
end
|
|
|
|
|
2020-01-17 00:45:44 -08:00
|
|
|
@spec migrate_to_db(Path.t() | nil) :: any()
|
|
|
|
def migrate_to_db(file_path \\ nil) do
|
2020-11-29 11:29:36 -08:00
|
|
|
with :ok <- Pleroma.Config.DeprecationWarnings.warn() do
|
2020-01-21 06:49:22 -08:00
|
|
|
config_file =
|
|
|
|
if file_path do
|
|
|
|
file_path
|
|
|
|
else
|
|
|
|
if Pleroma.Config.get(:release) do
|
|
|
|
Pleroma.Config.get(:config_path)
|
|
|
|
else
|
|
|
|
"config/#{Pleroma.Config.get(:env)}.secret.exs"
|
|
|
|
end
|
|
|
|
end
|
2020-01-18 05:55:33 -08:00
|
|
|
|
2020-01-17 00:45:44 -08:00
|
|
|
do_migrate_to_db(config_file)
|
|
|
|
else
|
2020-12-02 11:00:07 -08:00
|
|
|
_ ->
|
|
|
|
shell_error("Migration is not allowed until all deprecation warnings have been resolved.")
|
2020-01-17 00:45:44 -08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp do_migrate_to_db(config_file) do
|
|
|
|
if File.exists?(config_file) do
|
2020-06-20 02:34:34 -07:00
|
|
|
shell_info("Migrating settings from file: #{Path.expand(config_file)}")
|
2020-12-02 14:34:23 -08:00
|
|
|
truncatedb()
|
2020-01-23 06:23:02 -08:00
|
|
|
|
2020-01-21 06:49:22 -08:00
|
|
|
custom_config =
|
|
|
|
config_file
|
|
|
|
|> read_file()
|
|
|
|
|> elem(0)
|
2020-01-17 00:45:44 -08:00
|
|
|
|
|
|
|
custom_config
|
|
|
|
|> Keyword.keys()
|
2020-01-21 06:49:22 -08:00
|
|
|
|> Enum.each(&create(&1, custom_config))
|
2020-01-17 00:45:44 -08:00
|
|
|
else
|
2020-01-18 05:55:33 -08:00
|
|
|
shell_info("To migrate settings, you must define custom settings in #{config_file}.")
|
2020-01-17 00:45:44 -08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp create(group, settings) do
|
2020-01-21 06:49:22 -08:00
|
|
|
group
|
|
|
|
|> Pleroma.Config.Loader.filter_group(settings)
|
2019-09-29 01:17:38 -07:00
|
|
|
|> Enum.each(fn {key, value} ->
|
2020-05-31 00:46:02 -07:00
|
|
|
{:ok, _} = ConfigDB.update_or_create(%{group: group, key: key, value: value})
|
2019-09-29 01:17:38 -07:00
|
|
|
|
2020-01-18 05:55:33 -08:00
|
|
|
shell_info("Settings for key #{key} migrated.")
|
2019-09-29 01:17:38 -07:00
|
|
|
end)
|
|
|
|
|
2020-11-28 10:24:37 -08:00
|
|
|
shell_info("Settings for group #{inspect(group)} migrated.")
|
2019-09-29 01:17:38 -07:00
|
|
|
end
|
|
|
|
|
2020-01-21 06:49:22 -08:00
|
|
|
defp migrate_from_db(opts) do
|
2020-11-29 11:29:36 -08:00
|
|
|
env = opts[:env] || Pleroma.Config.get(:env)
|
2020-01-21 06:49:22 -08:00
|
|
|
|
2020-11-29 11:29:36 -08:00
|
|
|
config_path =
|
|
|
|
if Pleroma.Config.get(:release) do
|
|
|
|
:config_path
|
|
|
|
|> Pleroma.Config.get()
|
|
|
|
|> Path.dirname()
|
|
|
|
else
|
|
|
|
"config"
|
|
|
|
end
|
|
|
|
|> Path.join("#{env}.exported_from_db.secret.exs")
|
2020-01-21 06:49:22 -08:00
|
|
|
|
2020-11-29 11:29:36 -08:00
|
|
|
file = File.open!(config_path, [:write, :utf8])
|
2020-01-21 06:49:22 -08:00
|
|
|
|
2020-11-29 11:29:36 -08:00
|
|
|
IO.write(file, config_header())
|
2019-09-29 01:17:38 -07:00
|
|
|
|
2020-11-29 11:29:36 -08:00
|
|
|
ConfigDB
|
|
|
|
|> Repo.all()
|
|
|
|
|> Enum.each(&write_and_delete(&1, file, opts[:delete]))
|
2020-07-13 08:32:17 -07:00
|
|
|
|
2020-11-29 11:29:36 -08:00
|
|
|
:ok = File.close(file)
|
|
|
|
System.cmd("mix", ["format", config_path])
|
2020-01-21 06:49:22 -08:00
|
|
|
|
2020-11-29 11:29:36 -08:00
|
|
|
shell_info(
|
|
|
|
"Database configuration settings have been exported to config/#{env}.exported_from_db.secret.exs"
|
2020-01-21 06:49:22 -08:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
if Code.ensure_loaded?(Config.Reader) do
|
|
|
|
defp config_header, do: "import Config\r\n\r\n"
|
|
|
|
defp read_file(config_file), do: Config.Reader.read_imports!(config_file)
|
|
|
|
else
|
|
|
|
defp config_header, do: "use Mix.Config\r\n\r\n"
|
|
|
|
defp read_file(config_file), do: Mix.Config.eval!(config_file)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp write_and_delete(config, file, delete?) do
|
|
|
|
config
|
|
|
|
|> write(file)
|
|
|
|
|> delete(delete?)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp write(config, file) do
|
2020-05-31 00:46:02 -07:00
|
|
|
value = inspect(config.value, limit: :infinity)
|
2020-01-21 06:49:22 -08:00
|
|
|
|
2020-05-31 00:46:02 -07:00
|
|
|
IO.write(file, "config #{inspect(config.group)}, #{inspect(config.key)}, #{value}\r\n\r\n")
|
2020-01-21 06:49:22 -08:00
|
|
|
|
|
|
|
config
|
|
|
|
end
|
|
|
|
|
|
|
|
defp delete(config, true) do
|
|
|
|
{:ok, _} = Repo.delete(config)
|
2020-11-27 12:44:05 -08:00
|
|
|
|
|
|
|
shell_info(
|
2020-12-02 14:24:32 -08:00
|
|
|
"config #{inspect(config.group)}, #{inspect(config.key)} was deleted from the ConfigDB."
|
2020-11-27 12:44:05 -08:00
|
|
|
)
|
2020-01-21 06:49:22 -08:00
|
|
|
end
|
|
|
|
|
|
|
|
defp delete(_config, _), do: :ok
|
2020-11-25 10:44:06 -08:00
|
|
|
|
2020-12-06 07:02:30 -08:00
|
|
|
defp dump(%ConfigDB{} = config) do
|
2020-11-25 10:44:06 -08:00
|
|
|
value = inspect(config.value, limit: :infinity)
|
|
|
|
|
|
|
|
shell_info("config #{inspect(config.group)}, #{inspect(config.key)}, #{value}\r\n\r\n")
|
|
|
|
end
|
|
|
|
|
2020-12-06 07:02:30 -08:00
|
|
|
defp dump(_), do: :noop
|
|
|
|
|
2020-11-27 12:44:05 -08:00
|
|
|
defp dump_group(group) when is_atom(group) do
|
2020-12-03 08:34:23 -08:00
|
|
|
group
|
|
|
|
|> ConfigDB.get_all_by_group()
|
2020-12-03 08:03:14 -08:00
|
|
|
|> Enum.each(&dump/1)
|
2020-11-27 10:33:55 -08:00
|
|
|
end
|
2020-11-27 12:44:05 -08:00
|
|
|
|
2020-12-03 08:03:14 -08:00
|
|
|
defp group_exists?(group) do
|
2020-12-03 08:34:23 -08:00
|
|
|
group
|
|
|
|
|> ConfigDB.get_all_by_group()
|
2020-12-04 09:04:53 -08:00
|
|
|
|> Enum.any?()
|
2020-11-27 12:44:05 -08:00
|
|
|
end
|
|
|
|
|
2020-12-07 09:45:56 -08:00
|
|
|
defp key_exists?(group, key) do
|
|
|
|
group
|
|
|
|
|> ConfigDB.get_by_group_and_key(key)
|
|
|
|
|> is_nil
|
|
|
|
|> Kernel.!()
|
|
|
|
end
|
|
|
|
|
2020-11-29 10:59:03 -08:00
|
|
|
defp maybe_atomize(arg) when is_atom(arg), do: arg
|
2020-12-07 09:22:07 -08:00
|
|
|
|
|
|
|
defp maybe_atomize(":" <> arg), do: maybe_atomize(arg)
|
2020-11-27 14:20:28 -08:00
|
|
|
|
2020-11-29 10:59:03 -08:00
|
|
|
defp maybe_atomize(arg) when is_binary(arg) do
|
2020-12-06 07:02:30 -08:00
|
|
|
if ConfigDB.module_name?(arg) do
|
2020-12-03 08:03:14 -08:00
|
|
|
String.to_existing_atom("Elixir." <> arg)
|
2020-11-27 14:20:28 -08:00
|
|
|
else
|
|
|
|
String.to_atom(arg)
|
2020-12-03 08:34:23 -08:00
|
|
|
end
|
2020-11-27 14:20:28 -08:00
|
|
|
end
|
2020-11-29 11:29:36 -08:00
|
|
|
|
2020-12-02 10:33:34 -08:00
|
|
|
defp check_configdb(callback) do
|
2020-11-29 11:29:36 -08:00
|
|
|
with true <- Pleroma.Config.get([:configurable_from_database]) do
|
2020-12-02 10:33:34 -08:00
|
|
|
callback.()
|
2020-11-29 11:29:36 -08:00
|
|
|
else
|
2020-12-02 11:00:07 -08:00
|
|
|
_ ->
|
|
|
|
shell_error(
|
|
|
|
"ConfigDB not enabled. Please check the value of :configurable_from_database in your configuration."
|
|
|
|
)
|
2020-11-29 11:29:36 -08:00
|
|
|
end
|
|
|
|
end
|
2020-12-02 14:24:32 -08:00
|
|
|
|
|
|
|
defp delete_key(group, key) do
|
|
|
|
check_configdb(fn ->
|
2020-12-06 07:02:30 -08:00
|
|
|
ConfigDB.delete(%{group: group, key: key})
|
2020-12-02 14:24:32 -08:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp delete_group(group) do
|
|
|
|
check_configdb(fn ->
|
2020-12-04 09:30:48 -08:00
|
|
|
group
|
2020-12-06 07:02:30 -08:00
|
|
|
|> ConfigDB.get_all_by_group()
|
2020-12-04 09:37:49 -08:00
|
|
|
|> Enum.each(&ConfigDB.delete/1)
|
2020-12-02 14:24:32 -08:00
|
|
|
end)
|
|
|
|
end
|
2020-12-02 14:34:23 -08:00
|
|
|
|
2020-12-03 07:58:24 -08:00
|
|
|
defp truncatedb do
|
2020-12-02 14:34:23 -08:00
|
|
|
Ecto.Adapters.SQL.query!(Repo, "TRUNCATE config;")
|
|
|
|
Ecto.Adapters.SQL.query!(Repo, "ALTER SEQUENCE config_id_seq RESTART;")
|
|
|
|
end
|
2019-06-14 08:45:05 -07:00
|
|
|
end
|