diff options
Diffstat (limited to 'lib')
34 files changed, 353 insertions, 293 deletions
| diff --git a/lib/mix/tasks/pleroma/config.ex b/lib/mix/tasks/pleroma/config.ex index 5c9ef6904..f1b3a8766 100644 --- a/lib/mix/tasks/pleroma/config.ex +++ b/lib/mix/tasks/pleroma/config.ex @@ -72,8 +72,7 @@ defmodule Mix.Tasks.Pleroma.Config do      group      |> Pleroma.Config.Loader.filter_group(settings)      |> Enum.each(fn {key, value} -> -      key = inspect(key) -      {:ok, _} = ConfigDB.update_or_create(%{group: inspect(group), key: key, value: value}) +      {:ok, _} = ConfigDB.update_or_create(%{group: group, key: key, value: value})        shell_info("Settings for key #{key} migrated.")      end) @@ -131,12 +130,9 @@ defmodule Mix.Tasks.Pleroma.Config do    end    defp write(config, file) do -    value = -      config.value -      |> ConfigDB.from_binary() -      |> inspect(limit: :infinity) +    value = inspect(config.value, limit: :infinity) -    IO.write(file, "config #{config.group}, #{config.key}, #{value}\r\n\r\n") +    IO.write(file, "config #{inspect(config.group)}, #{inspect(config.key)}, #{value}\r\n\r\n")      config    end diff --git a/lib/pleroma/config/config_db.ex b/lib/pleroma/config/config_db.ex index 2b43d4c36..2f4eb8581 100644 --- a/lib/pleroma/config/config_db.ex +++ b/lib/pleroma/config/config_db.ex @@ -6,7 +6,7 @@ defmodule Pleroma.ConfigDB do    use Ecto.Schema    import Ecto.Changeset -  import Ecto.Query +  import Ecto.Query, only: [select: 3]    import Pleroma.Web.Gettext    alias __MODULE__ @@ -14,16 +14,6 @@ defmodule Pleroma.ConfigDB do    @type t :: %__MODULE__{} -  @full_key_update [ -    {:pleroma, :ecto_repos}, -    {:quack, :meta}, -    {:mime, :types}, -    {:cors_plug, [:max_age, :methods, :expose, :headers]}, -    {:auto_linker, :opts}, -    {:swarm, :node_blacklist}, -    {:logger, :backends} -  ] -    @full_subkey_update [      {:pleroma, :assets, :mascots},      {:pleroma, :emoji, :groups}, @@ -32,14 +22,10 @@ defmodule Pleroma.ConfigDB do      {:pleroma, :mrf_keyword, :replace}    ] -  @regex ~r/^~r(?'delimiter'[\/|"'([{<]{1})(?'pattern'.+)[\/|"')\]}>]{1}(?'modifier'[uismxfU]*)/u - -  @delimiters ["/", "|", "\"", "'", {"(", ")"}, {"[", "]"}, {"{", "}"}, {"<", ">"}] -    schema "config" do -    field(:key, :string) -    field(:group, :string) -    field(:value, :binary) +    field(:key, Pleroma.EctoType.Config.Atom) +    field(:group, Pleroma.EctoType.Config.Atom) +    field(:value, Pleroma.EctoType.Config.BinaryValue)      field(:db, {:array, :string}, virtual: true, default: [])      timestamps() @@ -51,10 +37,6 @@ defmodule Pleroma.ConfigDB do      |> select([c], {c.group, c.key, c.value})      |> Repo.all()      |> Enum.reduce([], fn {group, key, value}, acc -> -      group = ConfigDB.from_string(group) -      key = ConfigDB.from_string(key) -      value = from_binary(value) -        Keyword.update(acc, group, [{key, value}], &Keyword.merge(&1, [{key, value}]))      end)    end @@ -64,50 +46,41 @@ defmodule Pleroma.ConfigDB do    @spec changeset(ConfigDB.t(), map()) :: Changeset.t()    def changeset(config, params \\ %{}) do -    params = Map.put(params, :value, transform(params[:value])) -      config      |> cast(params, [:key, :group, :value])      |> validate_required([:key, :group, :value])      |> unique_constraint(:key, name: :config_group_key_index)    end -  @spec create(map()) :: {:ok, ConfigDB.t()} | {:error, Changeset.t()} -  def create(params) do +  defp create(params) do      %ConfigDB{}      |> changeset(params)      |> Repo.insert()    end -  @spec update(ConfigDB.t(), map()) :: {:ok, ConfigDB.t()} | {:error, Changeset.t()} -  def update(%ConfigDB{} = config, %{value: value}) do +  defp update(%ConfigDB{} = config, %{value: value}) do      config      |> changeset(%{value: value})      |> Repo.update()    end -  @spec get_db_keys(ConfigDB.t()) :: [String.t()] -  def get_db_keys(%ConfigDB{} = config) do -    config.value -    |> ConfigDB.from_binary() -    |> get_db_keys(config.key) -  end -    @spec get_db_keys(keyword(), any()) :: [String.t()]    def get_db_keys(value, key) do -    if Keyword.keyword?(value) do -      value |> Keyword.keys() |> Enum.map(&convert(&1)) -    else -      [convert(key)] -    end +    keys = +      if Keyword.keyword?(value) do +        Keyword.keys(value) +      else +        [key] +      end + +    Enum.map(keys, &to_json_types(&1))    end    @spec merge_group(atom(), atom(), keyword(), keyword()) :: keyword()    def merge_group(group, key, old_value, new_value) do -    new_keys = to_map_set(new_value) +    new_keys = to_mapset(new_value) -    intersect_keys = -      old_value |> to_map_set() |> MapSet.intersection(new_keys) |> MapSet.to_list() +    intersect_keys = old_value |> to_mapset() |> MapSet.intersection(new_keys) |> MapSet.to_list()      merged_value = ConfigDB.merge(old_value, new_value) @@ -120,12 +93,10 @@ defmodule Pleroma.ConfigDB do          []      end)      |> List.flatten() -    |> Enum.reduce(merged_value, fn subkey, acc -> -      Keyword.put(acc, subkey, new_value[subkey]) -    end) +    |> Enum.reduce(merged_value, &Keyword.put(&2, &1, new_value[&1]))    end -  defp to_map_set(keyword) do +  defp to_mapset(keyword) do      keyword      |> Keyword.keys()      |> MapSet.new() @@ -159,43 +130,40 @@ defmodule Pleroma.ConfigDB do    @spec update_or_create(map()) :: {:ok, ConfigDB.t()} | {:error, Changeset.t()}    def update_or_create(params) do +    params = Map.put(params, :value, to_elixir_types(params[:value]))      search_opts = Map.take(params, [:group, :key])      with %ConfigDB{} = config <- ConfigDB.get_by_params(search_opts), -         {:partial_update, true, config} <- -           {:partial_update, can_be_partially_updated?(config), config}, -         old_value <- from_binary(config.value), -         transformed_value <- do_transform(params[:value]), -         {:can_be_merged, true, config} <- {:can_be_merged, is_list(transformed_value), config}, -         new_value <- -           merge_group( -             ConfigDB.from_string(config.group), -             ConfigDB.from_string(config.key), -             old_value, -             transformed_value -           ) do -      ConfigDB.update(config, %{value: new_value}) +         {_, true, config} <- {:partial_update, can_be_partially_updated?(config), config}, +         {_, true, config} <- +           {:can_be_merged, is_list(params[:value]) and is_list(config.value), config} do +      new_value = merge_group(config.group, config.key, config.value, params[:value]) +      update(config, %{value: new_value})      else        {reason, false, config} when reason in [:partial_update, :can_be_merged] -> -        ConfigDB.update(config, params) +        update(config, params)        nil -> -        ConfigDB.create(params) +        create(params)      end    end    defp can_be_partially_updated?(%ConfigDB{} = config), do: not only_full_update?(config) -  defp only_full_update?(%ConfigDB{} = config) do -    config_group = ConfigDB.from_string(config.group) -    config_key = ConfigDB.from_string(config.key) - -    Enum.any?(@full_key_update, fn -      {group, key} when is_list(key) -> -        config_group == group and config_key in key - -      {group, key} -> -        config_group == group and config_key == key +  defp only_full_update?(%ConfigDB{group: group, key: key}) do +    full_key_update = [ +      {:pleroma, :ecto_repos}, +      {:quack, :meta}, +      {:mime, :types}, +      {:cors_plug, [:max_age, :methods, :expose, :headers]}, +      {:auto_linker, :opts}, +      {:swarm, :node_blacklist}, +      {:logger, :backends} +    ] + +    Enum.any?(full_key_update, fn +      {s_group, s_key} -> +        group == s_group and ((is_list(s_key) and key in s_key) or key == s_key)      end)    end @@ -205,11 +173,10 @@ defmodule Pleroma.ConfigDB do      with %ConfigDB{} = config <- ConfigDB.get_by_params(search_opts),           {config, sub_keys} when is_list(sub_keys) <- {config, params[:subkeys]}, -         old_value <- from_binary(config.value), -         keys <- Enum.map(sub_keys, &do_transform_string(&1)), -         {:partial_remove, config, new_value} when new_value != [] <- -           {:partial_remove, config, Keyword.drop(old_value, keys)} do -      ConfigDB.update(config, %{value: new_value}) +         keys <- Enum.map(sub_keys, &string_to_elixir_types(&1)), +         {_, config, new_value} when new_value != [] <- +           {:partial_remove, config, Keyword.drop(config.value, keys)} do +      update(config, %{value: new_value})      else        {:partial_remove, config, []} ->          Repo.delete(config) @@ -225,37 +192,32 @@ defmodule Pleroma.ConfigDB do      end    end -  @spec from_binary(binary()) :: term() -  def from_binary(binary), do: :erlang.binary_to_term(binary) - -  @spec from_binary_with_convert(binary()) :: any() -  def from_binary_with_convert(binary) do -    binary -    |> from_binary() -    |> do_convert() +  @spec to_json_types(term()) :: map() | list() | boolean() | String.t() +  def to_json_types(entity) when is_list(entity) do +    Enum.map(entity, &to_json_types/1)    end -  @spec from_string(String.t()) :: atom() | no_return() -  def from_string(string), do: do_transform_string(string) +  def to_json_types(%Regex{} = entity), do: inspect(entity) -  @spec convert(any()) :: any() -  def convert(entity), do: do_convert(entity) - -  defp do_convert(entity) when is_list(entity) do -    for v <- entity, into: [], do: do_convert(v) +  def to_json_types(entity) when is_map(entity) do +    Map.new(entity, fn {k, v} -> {to_json_types(k), to_json_types(v)} end)    end -  defp do_convert(%Regex{} = entity), do: inspect(entity) +  def to_json_types({:args, args}) when is_list(args) do +    arguments = +      Enum.map(args, fn +        arg when is_tuple(arg) -> inspect(arg) +        arg -> to_json_types(arg) +      end) -  defp do_convert(entity) when is_map(entity) do -    for {k, v} <- entity, into: %{}, do: {do_convert(k), do_convert(v)} +    %{"tuple" => [":args", arguments]}    end -  defp do_convert({:proxy_url, {type, :localhost, port}}) do -    %{"tuple" => [":proxy_url", %{"tuple" => [do_convert(type), "localhost", port]}]} +  def to_json_types({:proxy_url, {type, :localhost, port}}) do +    %{"tuple" => [":proxy_url", %{"tuple" => [to_json_types(type), "localhost", port]}]}    end -  defp do_convert({:proxy_url, {type, host, port}}) when is_tuple(host) do +  def to_json_types({:proxy_url, {type, host, port}}) when is_tuple(host) do      ip =        host        |> :inet_parse.ntoa() @@ -264,66 +226,64 @@ defmodule Pleroma.ConfigDB do      %{        "tuple" => [          ":proxy_url", -        %{"tuple" => [do_convert(type), ip, port]} +        %{"tuple" => [to_json_types(type), ip, port]}        ]      }    end -  defp do_convert({:proxy_url, {type, host, port}}) do +  def to_json_types({:proxy_url, {type, host, port}}) do      %{        "tuple" => [          ":proxy_url", -        %{"tuple" => [do_convert(type), to_string(host), port]} +        %{"tuple" => [to_json_types(type), to_string(host), port]}        ]      }    end -  defp do_convert({:partial_chain, entity}), do: %{"tuple" => [":partial_chain", inspect(entity)]} +  def to_json_types({:partial_chain, entity}), +    do: %{"tuple" => [":partial_chain", inspect(entity)]} -  defp do_convert(entity) when is_tuple(entity) do +  def to_json_types(entity) when is_tuple(entity) do      value =        entity        |> Tuple.to_list() -      |> do_convert() +      |> to_json_types()      %{"tuple" => value}    end -  defp do_convert(entity) when is_boolean(entity) or is_number(entity) or is_nil(entity) do +  def to_json_types(entity) when is_binary(entity), do: entity + +  def to_json_types(entity) when is_boolean(entity) or is_number(entity) or is_nil(entity) do      entity    end -  defp do_convert(entity) -       when is_atom(entity) and entity in [:"tlsv1.1", :"tlsv1.2", :"tlsv1.3"] do +  def to_json_types(entity) when entity in [:"tlsv1.1", :"tlsv1.2", :"tlsv1.3"] do      ":#{entity}"    end -  defp do_convert(entity) when is_atom(entity), do: inspect(entity) +  def to_json_types(entity) when is_atom(entity), do: inspect(entity) -  defp do_convert(entity) when is_binary(entity), do: entity +  @spec to_elixir_types(boolean() | String.t() | map() | list()) :: term() +  def to_elixir_types(%{"tuple" => [":args", args]}) when is_list(args) do +    arguments = +      Enum.map(args, fn arg -> +        if String.contains?(arg, ["{", "}"]) do +          {elem, []} = Code.eval_string(arg) +          elem +        else +          to_elixir_types(arg) +        end +      end) -  @spec transform(any()) :: binary() | no_return() -  def transform(entity) when is_binary(entity) or is_map(entity) or is_list(entity) do -    entity -    |> do_transform() -    |> to_binary() +    {:args, arguments}    end -  def transform(entity), do: to_binary(entity) - -  @spec transform_with_out_binary(any()) :: any() -  def transform_with_out_binary(entity), do: do_transform(entity) - -  @spec to_binary(any()) :: binary() -  def to_binary(entity), do: :erlang.term_to_binary(entity) - -  defp do_transform(%Regex{} = entity), do: entity - -  defp do_transform(%{"tuple" => [":proxy_url", %{"tuple" => [type, host, port]}]}) do -    {:proxy_url, {do_transform_string(type), parse_host(host), port}} +  def to_elixir_types(%{"tuple" => [":proxy_url", %{"tuple" => [type, host, port]}]}) do +    {:proxy_url, {string_to_elixir_types(type), parse_host(host), port}}    end -  defp do_transform(%{"tuple" => [":partial_chain", entity]}) do +  def to_elixir_types(%{"tuple" => [":partial_chain", entity]}) do      {partial_chain, []} =        entity        |> String.replace(~r/[^\w|^{:,[|^,|^[|^\]^}|^\/|^\.|^"]^\s/, "") @@ -332,25 +292,51 @@ defmodule Pleroma.ConfigDB do      {:partial_chain, partial_chain}    end -  defp do_transform(%{"tuple" => entity}) do -    Enum.reduce(entity, {}, fn val, acc -> Tuple.append(acc, do_transform(val)) end) +  def to_elixir_types(%{"tuple" => entity}) do +    Enum.reduce(entity, {}, &Tuple.append(&2, to_elixir_types(&1)))    end -  defp do_transform(entity) when is_map(entity) do -    for {k, v} <- entity, into: %{}, do: {do_transform(k), do_transform(v)} +  def to_elixir_types(entity) when is_map(entity) do +    Map.new(entity, fn {k, v} -> {to_elixir_types(k), to_elixir_types(v)} end)    end -  defp do_transform(entity) when is_list(entity) do -    for v <- entity, into: [], do: do_transform(v) +  def to_elixir_types(entity) when is_list(entity) do +    Enum.map(entity, &to_elixir_types/1)    end -  defp do_transform(entity) when is_binary(entity) do +  def to_elixir_types(entity) when is_binary(entity) do      entity      |> String.trim() -    |> do_transform_string() +    |> string_to_elixir_types() +  end + +  def to_elixir_types(entity), do: entity + +  @spec string_to_elixir_types(String.t()) :: +          atom() | Regex.t() | module() | String.t() | no_return() +  def string_to_elixir_types("~r" <> _pattern = regex) do +    pattern = +      ~r/^~r(?'delimiter'[\/|"'([{<]{1})(?'pattern'.+)[\/|"')\]}>]{1}(?'modifier'[uismxfU]*)/u + +    delimiters = ["/", "|", "\"", "'", {"(", ")"}, {"[", "]"}, {"{", "}"}, {"<", ">"}] + +    with %{"modifier" => modifier, "pattern" => pattern, "delimiter" => regex_delimiter} <- +           Regex.named_captures(pattern, regex), +         {:ok, {leading, closing}} <- find_valid_delimiter(delimiters, pattern, regex_delimiter), +         {result, _} <- Code.eval_string("~r#{leading}#{pattern}#{closing}#{modifier}") do +      result +    end    end -  defp do_transform(entity), do: entity +  def string_to_elixir_types(":" <> atom), do: String.to_atom(atom) + +  def string_to_elixir_types(value) do +    if module_name?(value) do +      String.to_existing_atom("Elixir." <> value) +    else +      value +    end +  end    defp parse_host("localhost"), do: :localhost @@ -387,27 +373,8 @@ defmodule Pleroma.ConfigDB do      end    end -  defp do_transform_string("~r" <> _pattern = regex) do -    with %{"modifier" => modifier, "pattern" => pattern, "delimiter" => regex_delimiter} <- -           Regex.named_captures(@regex, regex), -         {:ok, {leading, closing}} <- find_valid_delimiter(@delimiters, pattern, regex_delimiter), -         {result, _} <- Code.eval_string("~r#{leading}#{pattern}#{closing}#{modifier}") do -      result -    end -  end - -  defp do_transform_string(":" <> atom), do: String.to_atom(atom) - -  defp do_transform_string(value) do -    if is_module_name?(value) do -      String.to_existing_atom("Elixir." <> value) -    else -      value -    end -  end - -  @spec is_module_name?(String.t()) :: boolean() -  def is_module_name?(string) do +  @spec module_name?(String.t()) :: boolean() +  def module_name?(string) do      Regex.match?(~r/^(Pleroma|Phoenix|Tesla|Quack|Ueberauth|Swoosh)\./, string) or        string in ["Oban", "Ueberauth", "ExSyslogger"]    end diff --git a/lib/pleroma/config/transfer_task.ex b/lib/pleroma/config/transfer_task.ex index c02b70e96..eb86b8ff4 100644 --- a/lib/pleroma/config/transfer_task.ex +++ b/lib/pleroma/config/transfer_task.ex @@ -28,10 +28,6 @@ defmodule Pleroma.Config.TransferTask do      {:pleroma, Pleroma.Captcha, [:seconds_valid]},      {:pleroma, Pleroma.Upload, [:proxy_remote]},      {:pleroma, :instance, [:upload_limit]}, -    {:pleroma, :email_notifications, [:digest]}, -    {:pleroma, :oauth2, [:clean_expired_tokens]}, -    {:pleroma, Pleroma.ActivityExpiration, [:enabled]}, -    {:pleroma, Pleroma.ScheduledActivity, [:enabled]},      {:pleroma, :gopher, [:enabled]}    ] @@ -48,7 +44,7 @@ defmodule Pleroma.Config.TransferTask do        {logger, other} =          (Repo.all(ConfigDB) ++ deleted_settings) -        |> Enum.map(&transform_and_merge/1) +        |> Enum.map(&merge_with_default/1)          |> Enum.split_with(fn {group, _, _, _} -> group in [:logger, :quack] end)        logger @@ -92,11 +88,7 @@ defmodule Pleroma.Config.TransferTask do      end    end -  defp transform_and_merge(%{group: group, key: key, value: value} = setting) do -    group = ConfigDB.from_string(group) -    key = ConfigDB.from_string(key) -    value = ConfigDB.from_binary(value) - +  defp merge_with_default(%{group: group, key: key, value: value} = setting) do      default = Config.Holder.default_config(group, key)      merged = diff --git a/lib/pleroma/conversation/participation.ex b/lib/pleroma/conversation/participation.ex index ce7bd2396..8bc3e85d6 100644 --- a/lib/pleroma/conversation/participation.ex +++ b/lib/pleroma/conversation/participation.ex @@ -162,10 +162,13 @@ defmodule Pleroma.Conversation.Participation do      for_user(user, params)      |> Enum.map(fn participation ->        activity_id = -        ActivityPub.fetch_latest_activity_id_for_context(participation.conversation.ap_id, %{ -          user: user, -          blocking_user: user -        }) +        ActivityPub.fetch_latest_direct_activity_id_for_context( +          participation.conversation.ap_id, +          %{ +            user: user, +            blocking_user: user +          } +        )        %{          participation diff --git a/lib/pleroma/web/activity_pub/object_validators/types/date_time.ex b/lib/pleroma/ecto_type/activity_pub/object_validators/date_time.ex index 4f412fcde..d852c0abd 100644 --- a/lib/pleroma/web/activity_pub/object_validators/types/date_time.ex +++ b/lib/pleroma/ecto_type/activity_pub/object_validators/date_time.ex @@ -1,4 +1,8 @@ -defmodule Pleroma.Web.ActivityPub.ObjectValidators.Types.DateTime do +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.EctoType.ActivityPub.ObjectValidators.DateTime do    @moduledoc """    The AP standard defines the date fields in AP as xsd:DateTime. Elixir's    DateTime can't parse this, but it can parse the related iso8601. This diff --git a/lib/pleroma/web/activity_pub/object_validators/types/object_id.ex b/lib/pleroma/ecto_type/activity_pub/object_validators/object_id.ex index f71f76370..8034235b0 100644 --- a/lib/pleroma/web/activity_pub/object_validators/types/object_id.ex +++ b/lib/pleroma/ecto_type/activity_pub/object_validators/object_id.ex @@ -1,4 +1,8 @@ -defmodule Pleroma.Web.ActivityPub.ObjectValidators.Types.ObjectID do +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.EctoType.ActivityPub.ObjectValidators.ObjectID do    use Ecto.Type    def type, do: :string diff --git a/lib/pleroma/web/activity_pub/object_validators/types/recipients.ex b/lib/pleroma/ecto_type/activity_pub/object_validators/recipients.ex index 408e0f6ee..205527a96 100644 --- a/lib/pleroma/web/activity_pub/object_validators/types/recipients.ex +++ b/lib/pleroma/ecto_type/activity_pub/object_validators/recipients.ex @@ -1,7 +1,11 @@ -defmodule Pleroma.Web.ActivityPub.ObjectValidators.Types.Recipients do +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.EctoType.ActivityPub.ObjectValidators.Recipients do    use Ecto.Type -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types.ObjectID +  alias Pleroma.EctoType.ActivityPub.ObjectValidators.ObjectID    def type, do: {:array, ObjectID} diff --git a/lib/pleroma/web/activity_pub/object_validators/types/safe_text.ex b/lib/pleroma/ecto_type/activity_pub/object_validators/safe_text.ex index 95c948123..7f0405c7b 100644 --- a/lib/pleroma/web/activity_pub/object_validators/types/safe_text.ex +++ b/lib/pleroma/ecto_type/activity_pub/object_validators/safe_text.ex @@ -2,7 +2,7 @@  # Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/>  # SPDX-License-Identifier: AGPL-3.0-only -defmodule Pleroma.Web.ActivityPub.ObjectValidators.Types.SafeText do +defmodule Pleroma.EctoType.ActivityPub.ObjectValidators.SafeText do    use Ecto.Type    alias Pleroma.HTML diff --git a/lib/pleroma/web/activity_pub/object_validators/types/uri.ex b/lib/pleroma/ecto_type/activity_pub/object_validators/uri.ex index 24845bcc0..2054c26be 100644 --- a/lib/pleroma/web/activity_pub/object_validators/types/uri.ex +++ b/lib/pleroma/ecto_type/activity_pub/object_validators/uri.ex @@ -1,4 +1,8 @@ -defmodule Pleroma.Web.ActivityPub.ObjectValidators.Types.Uri do +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.EctoType.ActivityPub.ObjectValidators.Uri do    use Ecto.Type    def type, do: :string diff --git a/lib/pleroma/ecto_type/config/atom.ex b/lib/pleroma/ecto_type/config/atom.ex new file mode 100644 index 000000000..df565d432 --- /dev/null +++ b/lib/pleroma/ecto_type/config/atom.ex @@ -0,0 +1,26 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.EctoType.Config.Atom do +  use Ecto.Type + +  def type, do: :atom + +  def cast(key) when is_atom(key) do +    {:ok, key} +  end + +  def cast(key) when is_binary(key) do +    {:ok, Pleroma.ConfigDB.string_to_elixir_types(key)} +  end + +  def cast(_), do: :error + +  def load(key) do +    {:ok, Pleroma.ConfigDB.string_to_elixir_types(key)} +  end + +  def dump(key) when is_atom(key), do: {:ok, inspect(key)} +  def dump(_), do: :error +end diff --git a/lib/pleroma/ecto_type/config/binary_value.ex b/lib/pleroma/ecto_type/config/binary_value.ex new file mode 100644 index 000000000..bbd2608c5 --- /dev/null +++ b/lib/pleroma/ecto_type/config/binary_value.ex @@ -0,0 +1,27 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.EctoType.Config.BinaryValue do +  use Ecto.Type + +  def type, do: :term + +  def cast(value) when is_binary(value) do +    if String.valid?(value) do +      {:ok, value} +    else +      {:ok, :erlang.binary_to_term(value)} +    end +  end + +  def cast(value), do: {:ok, value} + +  def load(value) when is_binary(value) do +    {:ok, :erlang.binary_to_term(value)} +  end + +  def dump(value) do +    {:ok, :erlang.term_to_binary(value)} +  end +end diff --git a/lib/pleroma/migration_helper/notification_backfill.ex b/lib/pleroma/migration_helper/notification_backfill.ex index 09647d12a..b3770307a 100644 --- a/lib/pleroma/migration_helper/notification_backfill.ex +++ b/lib/pleroma/migration_helper/notification_backfill.ex @@ -18,7 +18,7 @@ defmodule Pleroma.MigrationHelper.NotificationBackfill do        )      query -    |> Repo.all() +    |> Repo.chunk_stream(100)      |> Enum.each(fn notification ->        type =          notification.activity diff --git a/lib/pleroma/repo.ex b/lib/pleroma/repo.ex index f62138466..6d85d70bc 100644 --- a/lib/pleroma/repo.ex +++ b/lib/pleroma/repo.ex @@ -8,6 +8,7 @@ defmodule Pleroma.Repo do      adapter: Ecto.Adapters.Postgres,      migration_timestamps: [type: :naive_datetime_usec] +  import Ecto.Query    require Logger    defmodule Instrumenter do @@ -78,6 +79,33 @@ defmodule Pleroma.Repo do        :ok      end    end + +  def chunk_stream(query, chunk_size) do +    # We don't actually need start and end funcitons of resource streaming, +    # but it seems to be the only way to not fetch records one-by-one and +    # have individual records be the elements of the stream, instead of +    # lists of records +    Stream.resource( +      fn -> 0 end, +      fn +        last_id -> +          query +          |> order_by(asc: :id) +          |> where([r], r.id > ^last_id) +          |> limit(^chunk_size) +          |> all() +          |> case do +            [] -> +              {:halt, last_id} + +            records -> +              last_id = List.last(records).id +              {records, last_id} +          end +      end, +      fn _ -> :ok end +    ) +  end  end  defmodule Pleroma.Repo.UnappliedMigrationsError do diff --git a/lib/pleroma/signature.ex b/lib/pleroma/signature.ex index d01728361..3aa6909d2 100644 --- a/lib/pleroma/signature.ex +++ b/lib/pleroma/signature.ex @@ -5,10 +5,10 @@  defmodule Pleroma.Signature do    @behaviour HTTPSignatures.Adapter +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.Keys    alias Pleroma.User    alias Pleroma.Web.ActivityPub.ActivityPub -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    def key_id_to_actor_id(key_id) do      uri = @@ -24,7 +24,7 @@ defmodule Pleroma.Signature do      maybe_ap_id = URI.to_string(uri) -    case Types.ObjectID.cast(maybe_ap_id) do +    case ObjectValidators.ObjectID.cast(maybe_ap_id) do        {:ok, ap_id} ->          {:ok, ap_id} diff --git a/lib/pleroma/user.ex b/lib/pleroma/user.ex index 52ac9052b..19ce9fb56 100644 --- a/lib/pleroma/user.ex +++ b/lib/pleroma/user.ex @@ -14,6 +14,7 @@ defmodule Pleroma.User do    alias Pleroma.Config    alias Pleroma.Conversation.Participation    alias Pleroma.Delivery +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.Emoji    alias Pleroma.FollowingRelationship    alias Pleroma.Formatter @@ -30,7 +31,6 @@ defmodule Pleroma.User do    alias Pleroma.Web    alias Pleroma.Web.ActivityPub.ActivityPub    alias Pleroma.Web.ActivityPub.Builder -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    alias Pleroma.Web.ActivityPub.Pipeline    alias Pleroma.Web.ActivityPub.Utils    alias Pleroma.Web.CommonAPI @@ -79,6 +79,7 @@ defmodule Pleroma.User do    schema "users" do      field(:bio, :string) +    field(:raw_bio, :string)      field(:email, :string)      field(:name, :string)      field(:nickname, :string) @@ -115,7 +116,7 @@ defmodule Pleroma.User do      field(:is_admin, :boolean, default: false)      field(:show_role, :boolean, default: true)      field(:settings, :map, default: nil) -    field(:uri, Types.Uri, default: nil) +    field(:uri, ObjectValidators.Uri, default: nil)      field(:hide_followers_count, :boolean, default: false)      field(:hide_follows_count, :boolean, default: false)      field(:hide_followers, :boolean, default: false) @@ -432,6 +433,7 @@ defmodule Pleroma.User do        params,        [          :bio, +        :raw_bio,          :name,          :emoji,          :avatar, @@ -607,7 +609,16 @@ defmodule Pleroma.User do      struct      |> confirmation_changeset(need_confirmation: need_confirmation?) -    |> cast(params, [:bio, :email, :name, :nickname, :password, :password_confirmation, :emoji]) +    |> cast(params, [ +      :bio, +      :raw_bio, +      :email, +      :name, +      :nickname, +      :password, +      :password_confirmation, +      :emoji +    ])      |> validate_required([:name, :nickname, :password, :password_confirmation])      |> validate_confirmation(:password)      |> unique_constraint(:email) diff --git a/lib/pleroma/web/activity_pub/activity_pub.ex b/lib/pleroma/web/activity_pub/activity_pub.ex index c9dc6135c..3e4f3ad30 100644 --- a/lib/pleroma/web/activity_pub/activity_pub.ex +++ b/lib/pleroma/web/activity_pub/activity_pub.ex @@ -210,7 +210,7 @@ defmodule Pleroma.Web.ActivityPub.ActivityPub do        conversation = Repo.preload(conversation, :participations)        last_activity_id = -        fetch_latest_activity_id_for_context(conversation.ap_id, %{ +        fetch_latest_direct_activity_id_for_context(conversation.ap_id, %{            user: user,            blocking_user: user          }) @@ -517,11 +517,12 @@ defmodule Pleroma.Web.ActivityPub.ActivityPub do      |> Repo.all()    end -  @spec fetch_latest_activity_id_for_context(String.t(), keyword() | map()) :: +  @spec fetch_latest_direct_activity_id_for_context(String.t(), keyword() | map()) ::            FlakeId.Ecto.CompatType.t() | nil -  def fetch_latest_activity_id_for_context(context, opts \\ %{}) do +  def fetch_latest_direct_activity_id_for_context(context, opts \\ %{}) do      context      |> fetch_activities_for_context_query(Map.merge(%{skip_preload: true}, opts)) +    |> restrict_visibility(%{visibility: "direct"})      |> limit(1)      |> select([a], a.id)      |> Repo.one() diff --git a/lib/pleroma/web/activity_pub/object_validator.ex b/lib/pleroma/web/activity_pub/object_validator.ex index c01c5f780..6a83a2c33 100644 --- a/lib/pleroma/web/activity_pub/object_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validator.ex @@ -9,6 +9,7 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidator do    the system.    """ +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.Object    alias Pleroma.User    alias Pleroma.Web.ActivityPub.ObjectValidators.AnnounceValidator @@ -17,7 +18,6 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidator do    alias Pleroma.Web.ActivityPub.ObjectValidators.DeleteValidator    alias Pleroma.Web.ActivityPub.ObjectValidators.EmojiReactValidator    alias Pleroma.Web.ActivityPub.ObjectValidators.LikeValidator -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    alias Pleroma.Web.ActivityPub.ObjectValidators.UndoValidator    @spec validate(map(), keyword()) :: {:ok, map(), keyword()} | {:error, any()} @@ -120,7 +120,7 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidator do    def stringify_keys(object), do: object    def fetch_actor(object) do -    with {:ok, actor} <- Types.ObjectID.cast(object["actor"]) do +    with {:ok, actor} <- ObjectValidators.ObjectID.cast(object["actor"]) do        User.get_or_fetch_by_ap_id(actor)      end    end diff --git a/lib/pleroma/web/activity_pub/object_validators/announce_validator.ex b/lib/pleroma/web/activity_pub/object_validators/announce_validator.ex index 40f861f47..6f757f49c 100644 --- a/lib/pleroma/web/activity_pub/object_validators/announce_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/announce_validator.ex @@ -5,9 +5,9 @@  defmodule Pleroma.Web.ActivityPub.ObjectValidators.AnnounceValidator do    use Ecto.Schema +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.Object    alias Pleroma.User -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    alias Pleroma.Web.ActivityPub.Utils    alias Pleroma.Web.ActivityPub.Visibility @@ -19,14 +19,14 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.AnnounceValidator do    @primary_key false    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) +    field(:id, ObjectValidators.ObjectID, primary_key: true)      field(:type, :string) -    field(:object, Types.ObjectID) -    field(:actor, Types.ObjectID) +    field(:object, ObjectValidators.ObjectID) +    field(:actor, ObjectValidators.ObjectID)      field(:context, :string, autogenerate: {Utils, :generate_context_id, []}) -    field(:to, Types.Recipients, default: []) -    field(:cc, Types.Recipients, default: []) -    field(:published, Types.DateTime) +    field(:to, ObjectValidators.Recipients, default: []) +    field(:cc, ObjectValidators.Recipients, default: []) +    field(:published, ObjectValidators.DateTime)    end    def cast_and_validate(data) do diff --git a/lib/pleroma/web/activity_pub/object_validators/chat_message_validator.ex b/lib/pleroma/web/activity_pub/object_validators/chat_message_validator.ex index 138736f23..c481d79e0 100644 --- a/lib/pleroma/web/activity_pub/object_validators/chat_message_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/chat_message_validator.ex @@ -5,9 +5,9 @@  defmodule Pleroma.Web.ActivityPub.ObjectValidators.ChatMessageValidator do    use Ecto.Schema +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.User    alias Pleroma.Web.ActivityPub.ObjectValidators.AttachmentValidator -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    import Ecto.Changeset    import Pleroma.Web.ActivityPub.Transmogrifier, only: [fix_emoji: 1] @@ -16,12 +16,12 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.ChatMessageValidator do    @derive Jason.Encoder    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) -    field(:to, Types.Recipients, default: []) +    field(:id, ObjectValidators.ObjectID, primary_key: true) +    field(:to, ObjectValidators.Recipients, default: [])      field(:type, :string) -    field(:content, Types.SafeText) -    field(:actor, Types.ObjectID) -    field(:published, Types.DateTime) +    field(:content, ObjectValidators.SafeText) +    field(:actor, ObjectValidators.ObjectID) +    field(:published, ObjectValidators.DateTime)      field(:emoji, :map, default: %{})      embeds_one(:attachment, AttachmentValidator) diff --git a/lib/pleroma/web/activity_pub/object_validators/create_chat_message_validator.ex b/lib/pleroma/web/activity_pub/object_validators/create_chat_message_validator.ex index fc582400b..7269f9ff0 100644 --- a/lib/pleroma/web/activity_pub/object_validators/create_chat_message_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/create_chat_message_validator.ex @@ -7,9 +7,9 @@  # - doesn't embed, will only get the object id  defmodule Pleroma.Web.ActivityPub.ObjectValidators.CreateChatMessageValidator do    use Ecto.Schema +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.Object -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    import Ecto.Changeset    import Pleroma.Web.ActivityPub.ObjectValidators.CommonValidations @@ -17,11 +17,11 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.CreateChatMessageValidator do    @primary_key false    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) -    field(:actor, Types.ObjectID) +    field(:id, ObjectValidators.ObjectID, primary_key: true) +    field(:actor, ObjectValidators.ObjectID)      field(:type, :string) -    field(:to, Types.Recipients, default: []) -    field(:object, Types.ObjectID) +    field(:to, ObjectValidators.Recipients, default: []) +    field(:object, ObjectValidators.ObjectID)    end    def cast_and_apply(data) do diff --git a/lib/pleroma/web/activity_pub/object_validators/create_note_validator.ex b/lib/pleroma/web/activity_pub/object_validators/create_note_validator.ex index 926804ce7..316bd0c07 100644 --- a/lib/pleroma/web/activity_pub/object_validators/create_note_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/create_note_validator.ex @@ -5,16 +5,16 @@  defmodule Pleroma.Web.ActivityPub.ObjectValidators.CreateNoteValidator do    use Ecto.Schema +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.Web.ActivityPub.ObjectValidators.NoteValidator -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    import Ecto.Changeset    @primary_key false    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) -    field(:actor, Types.ObjectID) +    field(:id, ObjectValidators.ObjectID, primary_key: true) +    field(:actor, ObjectValidators.ObjectID)      field(:type, :string)      field(:to, {:array, :string})      field(:cc, {:array, :string}) diff --git a/lib/pleroma/web/activity_pub/object_validators/delete_validator.ex b/lib/pleroma/web/activity_pub/object_validators/delete_validator.ex index e5d08eb5c..93a7b0e0b 100644 --- a/lib/pleroma/web/activity_pub/object_validators/delete_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/delete_validator.ex @@ -6,8 +6,8 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.DeleteValidator do    use Ecto.Schema    alias Pleroma.Activity +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.User -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    import Ecto.Changeset    import Pleroma.Web.ActivityPub.ObjectValidators.CommonValidations @@ -15,13 +15,13 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.DeleteValidator do    @primary_key false    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) +    field(:id, ObjectValidators.ObjectID, primary_key: true)      field(:type, :string) -    field(:actor, Types.ObjectID) -    field(:to, Types.Recipients, default: []) -    field(:cc, Types.Recipients, default: []) -    field(:deleted_activity_id, Types.ObjectID) -    field(:object, Types.ObjectID) +    field(:actor, ObjectValidators.ObjectID) +    field(:to, ObjectValidators.Recipients, default: []) +    field(:cc, ObjectValidators.Recipients, default: []) +    field(:deleted_activity_id, ObjectValidators.ObjectID) +    field(:object, ObjectValidators.ObjectID)    end    def cast_data(data) do diff --git a/lib/pleroma/web/activity_pub/object_validators/emoji_react_validator.ex b/lib/pleroma/web/activity_pub/object_validators/emoji_react_validator.ex index e87519c59..a543af1f8 100644 --- a/lib/pleroma/web/activity_pub/object_validators/emoji_react_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/emoji_react_validator.ex @@ -5,8 +5,8 @@  defmodule Pleroma.Web.ActivityPub.ObjectValidators.EmojiReactValidator do    use Ecto.Schema +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.Object -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    import Ecto.Changeset    import Pleroma.Web.ActivityPub.ObjectValidators.CommonValidations @@ -14,10 +14,10 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.EmojiReactValidator do    @primary_key false    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) +    field(:id, ObjectValidators.ObjectID, primary_key: true)      field(:type, :string) -    field(:object, Types.ObjectID) -    field(:actor, Types.ObjectID) +    field(:object, ObjectValidators.ObjectID) +    field(:actor, ObjectValidators.ObjectID)      field(:context, :string)      field(:content, :string)      field(:to, {:array, :string}, default: []) diff --git a/lib/pleroma/web/activity_pub/object_validators/like_validator.ex b/lib/pleroma/web/activity_pub/object_validators/like_validator.ex index 034f25492..493e4c247 100644 --- a/lib/pleroma/web/activity_pub/object_validators/like_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/like_validator.ex @@ -5,8 +5,8 @@  defmodule Pleroma.Web.ActivityPub.ObjectValidators.LikeValidator do    use Ecto.Schema +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.Object -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    alias Pleroma.Web.ActivityPub.Utils    import Ecto.Changeset @@ -15,13 +15,13 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.LikeValidator do    @primary_key false    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) +    field(:id, ObjectValidators.ObjectID, primary_key: true)      field(:type, :string) -    field(:object, Types.ObjectID) -    field(:actor, Types.ObjectID) +    field(:object, ObjectValidators.ObjectID) +    field(:actor, ObjectValidators.ObjectID)      field(:context, :string) -    field(:to, Types.Recipients, default: []) -    field(:cc, Types.Recipients, default: []) +    field(:to, ObjectValidators.Recipients, default: []) +    field(:cc, ObjectValidators.Recipients, default: [])    end    def cast_and_validate(data) do @@ -67,7 +67,7 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.LikeValidator do      with {[], []} <- {to, cc},           %Object{data: %{"actor" => actor}} <- Object.get_cached_by_ap_id(object), -         {:ok, actor} <- Types.ObjectID.cast(actor) do +         {:ok, actor} <- ObjectValidators.ObjectID.cast(actor) do        cng        |> put_change(:to, [actor])      else diff --git a/lib/pleroma/web/activity_pub/object_validators/note_validator.ex b/lib/pleroma/web/activity_pub/object_validators/note_validator.ex index 462a5620a..a10728ac6 100644 --- a/lib/pleroma/web/activity_pub/object_validators/note_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/note_validator.ex @@ -5,14 +5,14 @@  defmodule Pleroma.Web.ActivityPub.ObjectValidators.NoteValidator do    use Ecto.Schema -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    import Ecto.Changeset    @primary_key false    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) +    field(:id, ObjectValidators.ObjectID, primary_key: true)      field(:to, {:array, :string}, default: [])      field(:cc, {:array, :string}, default: [])      field(:bto, {:array, :string}, default: []) @@ -22,10 +22,10 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.NoteValidator do      field(:type, :string)      field(:content, :string)      field(:context, :string) -    field(:actor, Types.ObjectID) -    field(:attributedTo, Types.ObjectID) +    field(:actor, ObjectValidators.ObjectID) +    field(:attributedTo, ObjectValidators.ObjectID)      field(:summary, :string) -    field(:published, Types.DateTime) +    field(:published, ObjectValidators.DateTime)      # TODO: Write type      field(:emoji, :map, default: %{})      field(:sensitive, :boolean, default: false) @@ -35,7 +35,7 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.NoteValidator do      field(:like_count, :integer, default: 0)      field(:announcement_count, :integer, default: 0)      field(:inRepyTo, :string) -    field(:uri, Types.Uri) +    field(:uri, ObjectValidators.Uri)      field(:likes, {:array, :string}, default: [])      field(:announcements, {:array, :string}, default: []) diff --git a/lib/pleroma/web/activity_pub/object_validators/undo_validator.ex b/lib/pleroma/web/activity_pub/object_validators/undo_validator.ex index d0ba418e8..e8d2d39c1 100644 --- a/lib/pleroma/web/activity_pub/object_validators/undo_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/undo_validator.ex @@ -6,7 +6,7 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.UndoValidator do    use Ecto.Schema    alias Pleroma.Activity -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    import Ecto.Changeset    import Pleroma.Web.ActivityPub.ObjectValidators.CommonValidations @@ -14,10 +14,10 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.UndoValidator do    @primary_key false    embedded_schema do -    field(:id, Types.ObjectID, primary_key: true) +    field(:id, ObjectValidators.ObjectID, primary_key: true)      field(:type, :string) -    field(:object, Types.ObjectID) -    field(:actor, Types.ObjectID) +    field(:object, ObjectValidators.ObjectID) +    field(:actor, ObjectValidators.ObjectID)      field(:to, {:array, :string}, default: [])      field(:cc, {:array, :string}, default: [])    end diff --git a/lib/pleroma/web/activity_pub/object_validators/url_object_validator.ex b/lib/pleroma/web/activity_pub/object_validators/url_object_validator.ex index 47e231150..f64fac46d 100644 --- a/lib/pleroma/web/activity_pub/object_validators/url_object_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/url_object_validator.ex @@ -1,14 +1,18 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only +  defmodule Pleroma.Web.ActivityPub.ObjectValidators.UrlObjectValidator do    use Ecto.Schema -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    import Ecto.Changeset    @primary_key false    embedded_schema do      field(:type, :string) -    field(:href, Types.Uri) +    field(:href, ObjectValidators.Uri)      field(:mediaType, :string)    end diff --git a/lib/pleroma/web/activity_pub/transmogrifier.ex b/lib/pleroma/web/activity_pub/transmogrifier.ex index 985921aa0..851f474b8 100644 --- a/lib/pleroma/web/activity_pub/transmogrifier.ex +++ b/lib/pleroma/web/activity_pub/transmogrifier.ex @@ -8,6 +8,7 @@ defmodule Pleroma.Web.ActivityPub.Transmogrifier do    """    alias Pleroma.Activity    alias Pleroma.EarmarkRenderer +  alias Pleroma.EctoType.ActivityPub.ObjectValidators    alias Pleroma.FollowingRelationship    alias Pleroma.Maps    alias Pleroma.Notification @@ -18,7 +19,6 @@ defmodule Pleroma.Web.ActivityPub.Transmogrifier do    alias Pleroma.Web.ActivityPub.ActivityPub    alias Pleroma.Web.ActivityPub.Builder    alias Pleroma.Web.ActivityPub.ObjectValidator -  alias Pleroma.Web.ActivityPub.ObjectValidators.Types    alias Pleroma.Web.ActivityPub.Pipeline    alias Pleroma.Web.ActivityPub.Utils    alias Pleroma.Web.ActivityPub.Visibility @@ -725,7 +725,7 @@ defmodule Pleroma.Web.ActivityPub.Transmogrifier do      else        {:error, {:validate_object, _}} = e ->          # Check if we have a create activity for this -        with {:ok, object_id} <- Types.ObjectID.cast(data["object"]), +        with {:ok, object_id} <- ObjectValidators.ObjectID.cast(data["object"]),               %Activity{data: %{"actor" => actor}} <-                 Activity.create_by_object_ap_id(object_id) |> Repo.one(),               # We have one, insert a tombstone and retry diff --git a/lib/pleroma/web/admin_api/controllers/config_controller.ex b/lib/pleroma/web/admin_api/controllers/config_controller.ex index d6e2019bc..7f60470cb 100644 --- a/lib/pleroma/web/admin_api/controllers/config_controller.ex +++ b/lib/pleroma/web/admin_api/controllers/config_controller.ex @@ -33,7 +33,11 @@ defmodule Pleroma.Web.AdminAPI.ConfigController do    def show(conn, %{only_db: true}) do      with :ok <- configurable_from_database() do        configs = Pleroma.Repo.all(ConfigDB) -      render(conn, "index.json", %{configs: configs}) + +      render(conn, "index.json", %{ +        configs: configs, +        need_reboot: Restarter.Pleroma.need_reboot?() +      })      end    end @@ -61,17 +65,20 @@ defmodule Pleroma.Web.AdminAPI.ConfigController do                  value                end -            %{ -              group: ConfigDB.convert(group), -              key: ConfigDB.convert(key), -              value: ConfigDB.convert(merged_value) +            %ConfigDB{ +              group: group, +              key: key, +              value: merged_value              }              |> Pleroma.Maps.put_if_present(:db, db)            end)          end)          |> List.flatten() -      json(conn, %{configs: merged, need_reboot: Restarter.Pleroma.need_reboot?()}) +      render(conn, "index.json", %{ +        configs: merged, +        need_reboot: Restarter.Pleroma.need_reboot?() +      })      end    end @@ -91,24 +98,17 @@ defmodule Pleroma.Web.AdminAPI.ConfigController do        {deleted, updated} =          results -        |> Enum.map(fn {:ok, config} -> -          Map.put(config, :db, ConfigDB.get_db_keys(config)) -        end) -        |> Enum.split_with(fn config -> -          Ecto.get_meta(config, :state) == :deleted +        |> Enum.map(fn {:ok, %{key: key, value: value} = config} -> +          Map.put(config, :db, ConfigDB.get_db_keys(value, key))          end) +        |> Enum.split_with(&(Ecto.get_meta(&1, :state) == :deleted))        Config.TransferTask.load_and_update_env(deleted, false)        if not Restarter.Pleroma.need_reboot?() do          changed_reboot_settings? =            (updated ++ deleted) -          |> Enum.any?(fn config -> -            group = ConfigDB.from_string(config.group) -            key = ConfigDB.from_string(config.key) -            value = ConfigDB.from_binary(config.value) -            Config.TransferTask.pleroma_need_restart?(group, key, value) -          end) +          |> Enum.any?(&Config.TransferTask.pleroma_need_restart?(&1.group, &1.key, &1.value))          if changed_reboot_settings?, do: Restarter.Pleroma.need_reboot()        end diff --git a/lib/pleroma/web/admin_api/views/config_view.ex b/lib/pleroma/web/admin_api/views/config_view.ex index 587ef760e..d2d8b5907 100644 --- a/lib/pleroma/web/admin_api/views/config_view.ex +++ b/lib/pleroma/web/admin_api/views/config_view.ex @@ -5,23 +5,20 @@  defmodule Pleroma.Web.AdminAPI.ConfigView do    use Pleroma.Web, :view +  alias Pleroma.ConfigDB +    def render("index.json", %{configs: configs} = params) do -    map = %{ -      configs: render_many(configs, __MODULE__, "show.json", as: :config) +    %{ +      configs: render_many(configs, __MODULE__, "show.json", as: :config), +      need_reboot: params[:need_reboot]      } - -    if params[:need_reboot] do -      Map.put(map, :need_reboot, true) -    else -      map -    end    end    def render("show.json", %{config: config}) do      map = %{ -      key: config.key, -      group: config.group, -      value: Pleroma.ConfigDB.from_binary_with_convert(config.value) +      key: ConfigDB.to_json_types(config.key), +      group: ConfigDB.to_json_types(config.group), +      value: ConfigDB.to_json_types(config.value)      }      if config.db != [] do diff --git a/lib/pleroma/web/api_spec/operations/notification_operation.ex b/lib/pleroma/web/api_spec/operations/notification_operation.ex index c966b553a..41328b5f2 100644 --- a/lib/pleroma/web/api_spec/operations/notification_operation.ex +++ b/lib/pleroma/web/api_spec/operations/notification_operation.ex @@ -183,7 +183,6 @@ defmodule Pleroma.Web.ApiSpec.NotificationOperation do          "favourite",          "reblog",          "mention", -        "poll",          "pleroma:emoji_reaction",          "pleroma:chat_mention",          "move", diff --git a/lib/pleroma/web/mastodon_api/controllers/account_controller.ex b/lib/pleroma/web/mastodon_api/controllers/account_controller.ex index 7cdd8f458..c38c2b895 100644 --- a/lib/pleroma/web/mastodon_api/controllers/account_controller.ex +++ b/lib/pleroma/web/mastodon_api/controllers/account_controller.ex @@ -165,6 +165,7 @@ defmodule Pleroma.Web.MastodonAPI.AccountController do        end)        |> Maps.put_if_present(:name, params[:display_name])        |> Maps.put_if_present(:bio, params[:note]) +      |> Maps.put_if_present(:raw_bio, params[:note])        |> Maps.put_if_present(:avatar, params[:avatar])        |> Maps.put_if_present(:banner, params[:header])        |> Maps.put_if_present(:background, params[:pleroma_background_image]) diff --git a/lib/pleroma/web/mastodon_api/views/account_view.ex b/lib/pleroma/web/mastodon_api/views/account_view.ex index 9fc06bf9d..68beb69b8 100644 --- a/lib/pleroma/web/mastodon_api/views/account_view.ex +++ b/lib/pleroma/web/mastodon_api/views/account_view.ex @@ -224,7 +224,7 @@ defmodule Pleroma.Web.MastodonAPI.AccountView do        fields: user.fields,        bot: bot,        source: %{ -        note: prepare_user_bio(user), +        note: user.raw_bio || "",          sensitive: false,          fields: user.raw_fields,          pleroma: %{ @@ -260,17 +260,6 @@ defmodule Pleroma.Web.MastodonAPI.AccountView do      |> maybe_put_unread_notification_count(user, opts[:for])    end -  defp prepare_user_bio(%User{bio: ""}), do: "" - -  defp prepare_user_bio(%User{bio: bio}) when is_binary(bio) do -    bio -    |> String.replace(~r(<br */?>), "\n") -    |> Pleroma.HTML.strip_tags() -    |> HtmlEntities.decode() -  end - -  defp prepare_user_bio(_), do: "" -    defp username_from_nickname(string) when is_binary(string) do      hd(String.split(string, "@"))    end diff --git a/lib/pleroma/web/mastodon_api/views/conversation_view.ex b/lib/pleroma/web/mastodon_api/views/conversation_view.ex index fbe618377..06f0c1728 100644 --- a/lib/pleroma/web/mastodon_api/views/conversation_view.ex +++ b/lib/pleroma/web/mastodon_api/views/conversation_view.ex @@ -23,10 +23,13 @@ defmodule Pleroma.Web.MastodonAPI.ConversationView do      last_activity_id =        with nil <- participation.last_activity_id do -        ActivityPub.fetch_latest_activity_id_for_context(participation.conversation.ap_id, %{ -          user: user, -          blocking_user: user -        }) +        ActivityPub.fetch_latest_direct_activity_id_for_context( +          participation.conversation.ap_id, +          %{ +            user: user, +            blocking_user: user +          } +        )        end      activity = Activity.get_by_id_with_object(last_activity_id) | 
