mobilizon/lib/federation/activity_stream/converter/resource.ex

86 lines
2.3 KiB
Elixir

defmodule Mobilizon.Federation.ActivityStream.Converter.Resource do
@moduledoc """
Resource converter.
This module allows to convert resources from ActivityStream format to our own
internal one, and back.
"""
alias Mobilizon.Actors.Actor
alias Mobilizon.Federation.ActivityPub
alias Mobilizon.Federation.ActivityStream.{Converter, Convertible}
alias Mobilizon.Resources
alias Mobilizon.Resources.Resource
@behaviour Converter
defimpl Convertible, for: Resource do
alias Mobilizon.Federation.ActivityStream.Converter.Resource, as: ResourceConverter
defdelegate model_to_as(resource), to: ResourceConverter
end
@doc """
Convert an resource struct to an ActivityStream representation
"""
@impl Converter
@spec model_to_as(Resource.t()) :: map
def model_to_as(%Resource{actor: %Actor{url: actor_url}} = resource) do
res = %{
"type" => "Resource",
"actor" => actor_url,
"id" => resource.url,
"name" => resource.title,
"summary" => resource.summary
# TODO add resource_url
}
# if parent do
# Map.put(res, "parent", parent.url)
# else
# res
# end
end
@doc """
Converts an AP object data to our internal data structure.
"""
@impl Converter
@spec as_to_model_data(map) :: {:ok, map} | {:error, any()}
def as_to_model_data(%{"type" => "Resource", "actor" => actor_url} = object) do
with {:ok, %Actor{id: actor_id} = actor} <-
ActivityPub.get_or_fetch_actor_by_url(actor_url) do
data = %{
title: object["name"],
summary: object["summary"],
url: object["id"],
actor_id: actor_id
# parent_id: get_eventual_parent(object)
# TODO add resource_url
}
{:ok, data}
end
end
defp get_eventual_parent_id(%{"parent" => parent_url})
when not is_nil(parent_url) do
with %Resource{id: resource_id} = _resource <-
Resources.get_resource_by_url(parent_url) do
resource_id
else
nil ->
with {:ok, %Resource{id: resource_id} = _resource} <-
ActivityPub.fetch_object_from_url(parent_url) do
resource_id
else
_ -> nil
end
_ ->
nil
end
end
defp get_eventual_parent_id(_), do: nil
end