diff options
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/mix/tasks/pleroma/docs.ex | 42 | ||||
| -rw-r--r-- | lib/pleroma/docs/generator.ex | 73 | ||||
| -rw-r--r-- | lib/pleroma/docs/json.ex | 20 | ||||
| -rw-r--r-- | lib/pleroma/docs/markdown.ex | 78 | ||||
| -rw-r--r-- | lib/pleroma/user.ex | 9 | ||||
| -rw-r--r-- | lib/pleroma/web/admin_api/config.ex | 15 | ||||
| -rw-r--r-- | lib/pleroma/web/router.ex | 1 | ||||
| -rw-r--r-- | lib/pleroma/web/twitter_api/controllers/util_controller.ex | 19 | 
8 files changed, 254 insertions, 3 deletions
| diff --git a/lib/mix/tasks/pleroma/docs.ex b/lib/mix/tasks/pleroma/docs.ex new file mode 100644 index 000000000..0d2663648 --- /dev/null +++ b/lib/mix/tasks/pleroma/docs.ex @@ -0,0 +1,42 @@ +defmodule Mix.Tasks.Pleroma.Docs do +  use Mix.Task +  import Mix.Pleroma + +  @shortdoc "Generates docs from descriptions.exs" +  @moduledoc """ +  Generates docs from `descriptions.exs`. + +  Supports two formats: `markdown` and `json`. + +  ## Generate Markdown docs + +  `mix pleroma.docs` + +  ## Generate JSON docs + +  `mix pleroma.docs json` +  """ + +  def run(["json"]) do +    do_run(Pleroma.Docs.JSON) +  end + +  def run(_) do +    do_run(Pleroma.Docs.Markdown) +  end + +  defp do_run(implementation) do +    start_pleroma() + +    with {descriptions, _paths} <- Mix.Config.eval!("config/description.exs"), +         {:ok, file_path} <- +           Pleroma.Docs.Generator.process( +             implementation, +             descriptions[:pleroma][:config_description] +           ) do +      type = if implementation == Pleroma.Docs.Markdown, do: "Markdown", else: "JSON" + +      Mix.shell().info([:green, "#{type} docs successfully generated to #{file_path}."]) +    end +  end +end diff --git a/lib/pleroma/docs/generator.ex b/lib/pleroma/docs/generator.ex new file mode 100644 index 000000000..aa578eee2 --- /dev/null +++ b/lib/pleroma/docs/generator.ex @@ -0,0 +1,73 @@ +defmodule Pleroma.Docs.Generator do +  @callback process(keyword()) :: {:ok, String.t()} + +  @spec process(module(), keyword()) :: {:ok, String.t()} +  def process(implementation, descriptions) do +    implementation.process(descriptions) +  end + +  @spec uploaders_list() :: [module()] +  def uploaders_list do +    {:ok, modules} = :application.get_key(:pleroma, :modules) + +    Enum.filter(modules, fn module -> +      name_as_list = Module.split(module) + +      List.starts_with?(name_as_list, ["Pleroma", "Uploaders"]) and +        List.last(name_as_list) != "Uploader" +    end) +  end + +  @spec filters_list() :: [module()] +  def filters_list do +    {:ok, modules} = :application.get_key(:pleroma, :modules) + +    Enum.filter(modules, fn module -> +      name_as_list = Module.split(module) + +      List.starts_with?(name_as_list, ["Pleroma", "Upload", "Filter"]) +    end) +  end + +  @spec mrf_list() :: [module()] +  def mrf_list do +    {:ok, modules} = :application.get_key(:pleroma, :modules) + +    Enum.filter(modules, fn module -> +      name_as_list = Module.split(module) + +      List.starts_with?(name_as_list, ["Pleroma", "Web", "ActivityPub", "MRF"]) and +        length(name_as_list) > 4 +    end) +  end + +  @spec richmedia_parsers() :: [module()] +  def richmedia_parsers do +    {:ok, modules} = :application.get_key(:pleroma, :modules) + +    Enum.filter(modules, fn module -> +      name_as_list = Module.split(module) + +      List.starts_with?(name_as_list, ["Pleroma", "Web", "RichMedia", "Parsers"]) and +        length(name_as_list) == 5 +    end) +  end +end + +defimpl Jason.Encoder, for: Tuple do +  def encode(tuple, opts) do +    Jason.Encode.list(Tuple.to_list(tuple), opts) +  end +end + +defimpl Jason.Encoder, for: [Regex, Function] do +  def encode(term, opts) do +    Jason.Encode.string(inspect(term), opts) +  end +end + +defimpl String.Chars, for: Regex do +  def to_string(term) do +    inspect(term) +  end +end diff --git a/lib/pleroma/docs/json.ex b/lib/pleroma/docs/json.ex new file mode 100644 index 000000000..18ba01d58 --- /dev/null +++ b/lib/pleroma/docs/json.ex @@ -0,0 +1,20 @@ +defmodule Pleroma.Docs.JSON do +  @behaviour Pleroma.Docs.Generator + +  @spec process(keyword()) :: {:ok, String.t()} +  def process(descriptions) do +    config_path = "docs/generate_config.json" + +    with {:ok, file} <- File.open(config_path, [:write]), +         json <- generate_json(descriptions), +         :ok <- IO.write(file, json), +         :ok <- File.close(file) do +      {:ok, config_path} +    end +  end + +  @spec generate_json([keyword()]) :: String.t() +  def generate_json(descriptions) do +    Jason.encode!(descriptions) +  end +end diff --git a/lib/pleroma/docs/markdown.ex b/lib/pleroma/docs/markdown.ex new file mode 100644 index 000000000..24930cc9f --- /dev/null +++ b/lib/pleroma/docs/markdown.ex @@ -0,0 +1,78 @@ +defmodule Pleroma.Docs.Markdown do +  @behaviour Pleroma.Docs.Generator + +  @spec process(keyword()) :: {:ok, String.t()} +  def process(descriptions) do +    config_path = "docs/config.md" +    {:ok, file} = File.open(config_path, [:utf8, :write]) +    IO.write(file, "# Configuration\n") +    IO.write(file, "Date of generation: #{Date.utc_today()}\n\n") + +    IO.write( +      file, +      "This file describe the configuration, it is recommended to edit the relevant `*.secret.exs` file instead of the others founds in the ``config`` directory.\n\n" <> +        "If you run Pleroma with ``MIX_ENV=prod`` the file is ``prod.secret.exs``, otherwise it is ``dev.secret.exs``.\n\n" +    ) + +    for group <- descriptions do +      if is_nil(group[:key]) do +        IO.write(file, "## #{inspect(group[:group])}\n") +      else +        IO.write(file, "## #{inspect(group[:key])}\n") +      end + +      IO.write(file, "#{group[:description]}\n") + +      for child <- group[:children] do +        print_child_header(file, child) + +        print_suggestions(file, child[:suggestions]) + +        if child[:children] do +          for subchild <- child[:children] do +            print_child_header(file, subchild) + +            print_suggestions(file, subchild[:suggestions]) +          end +        end +      end + +      IO.write(file, "\n") +    end + +    :ok = File.close(file) +    {:ok, config_path} +  end + +  defp print_suggestion(file, suggestion) when is_list(suggestion) do +    IO.write(file, "  `#{inspect(suggestion)}`\n") +  end + +  defp print_suggestion(file, suggestion) when is_function(suggestion) do +    IO.write(file, "  `#{inspect(suggestion.())}`\n") +  end + +  defp print_suggestion(file, suggestion, as_list \\ false) do +    list_mark = if as_list, do: "- ", else: "" +    IO.write(file, "  #{list_mark}`#{inspect(suggestion)}`\n") +  end + +  defp print_suggestions(_file, nil), do: nil + +  defp print_suggestions(file, suggestions) do +    IO.write(file, "Suggestions:\n") + +    if length(suggestions) > 1 do +      for suggestion <- suggestions do +        print_suggestion(file, suggestion, true) +      end +    else +      print_suggestion(file, List.first(suggestions)) +    end +  end + +  defp print_child_header(file, child) do +    IO.write(file, "- `#{inspect(child[:key])}` -`#{inspect(child[:type])}`  \n") +    IO.write(file, "#{child[:description]}  \n") +  end +end diff --git a/lib/pleroma/user.ex b/lib/pleroma/user.ex index ceca11def..5e10ac25f 100644 --- a/lib/pleroma/user.ex +++ b/lib/pleroma/user.ex @@ -1624,4 +1624,13 @@ defmodule Pleroma.User do    def is_internal_user?(%User{nickname: nil}), do: true    def is_internal_user?(%User{local: true, nickname: "internal." <> _}), do: true    def is_internal_user?(_), do: false + +  def change_email(user, email) do +    user +    |> cast(%{email: email}, [:email]) +    |> validate_required([:email]) +    |> unique_constraint(:email) +    |> validate_format(:email, @email_regex) +    |> update_and_set_cache() +  end  end diff --git a/lib/pleroma/web/admin_api/config.ex b/lib/pleroma/web/admin_api/config.ex index a10cc779b..1917a5580 100644 --- a/lib/pleroma/web/admin_api/config.ex +++ b/lib/pleroma/web/admin_api/config.ex @@ -90,6 +90,8 @@ defmodule Pleroma.Web.AdminAPI.Config do      for v <- entity, into: [], do: do_convert(v)    end +  defp do_convert(%Regex{} = entity), do: inspect(entity) +    defp do_convert(entity) when is_map(entity) do      for {k, v} <- entity, into: %{}, do: {do_convert(k), do_convert(v)}    end @@ -122,7 +124,7 @@ defmodule Pleroma.Web.AdminAPI.Config do    def transform(entity), do: :erlang.term_to_binary(entity) -  defp do_transform(%Regex{} = entity) when is_map(entity), do: entity +  defp do_transform(%Regex{} = entity), do: entity    defp do_transform(%{"tuple" => [":dispatch", [entity]]}) do      {dispatch_settings, []} = do_eval(entity) @@ -154,8 +156,15 @@ defmodule Pleroma.Web.AdminAPI.Config do    defp do_transform(entity), do: entity    defp do_transform_string("~r/" <> pattern) do -    pattern = String.trim_trailing(pattern, "/") -    ~r/#{pattern}/ +    modificator = String.split(pattern, "/") |> List.last() +    pattern = String.trim_trailing(pattern, "/" <> modificator) + +    case modificator do +      "" -> ~r/#{pattern}/ +      "i" -> ~r/#{pattern}/i +      "u" -> ~r/#{pattern}/u +      "s" -> ~r/#{pattern}/s +    end    end    defp do_transform_string(":" <> atom), do: String.to_atom(atom) diff --git a/lib/pleroma/web/router.ex b/lib/pleroma/web/router.ex index 7cd59acb2..b0464037e 100644 --- a/lib/pleroma/web/router.ex +++ b/lib/pleroma/web/router.ex @@ -224,6 +224,7 @@ defmodule Pleroma.Web.Router do      scope [] do        pipe_through(:oauth_write) +      post("/change_email", UtilController, :change_email)        post("/change_password", UtilController, :change_password)        post("/delete_account", UtilController, :delete_account)        put("/notification_settings", UtilController, :update_notificaton_settings) diff --git a/lib/pleroma/web/twitter_api/controllers/util_controller.ex b/lib/pleroma/web/twitter_api/controllers/util_controller.ex index 3405bd3b7..867787c57 100644 --- a/lib/pleroma/web/twitter_api/controllers/util_controller.ex +++ b/lib/pleroma/web/twitter_api/controllers/util_controller.ex @@ -314,6 +314,25 @@ defmodule Pleroma.Web.TwitterAPI.UtilController do      end    end +  def change_email(%{assigns: %{user: user}} = conn, params) do +    case CommonAPI.Utils.confirm_current_password(user, params["password"]) do +      {:ok, user} -> +        with {:ok, _user} <- User.change_email(user, params["email"]) do +          json(conn, %{status: "success"}) +        else +          {:error, changeset} -> +            {_, {error, _}} = Enum.at(changeset.errors, 0) +            json(conn, %{error: "Email #{error}."}) + +          _ -> +            json(conn, %{error: "Unable to change email."}) +        end + +      {:error, msg} -> +        json(conn, %{error: msg}) +    end +  end +    def delete_account(%{assigns: %{user: user}} = conn, params) do      case CommonAPI.Utils.confirm_current_password(user, params["password"]) do        {:ok, user} -> | 
