2020-01-26 20:36:50 +00:00
|
|
|
defmodule Mobilizon.Web.JsonLD.ObjectView do
|
|
|
|
use Mobilizon.Web, :view
|
2019-03-04 17:38:30 +00:00
|
|
|
|
|
|
|
alias Mobilizon.Actors.Actor
|
|
|
|
alias Mobilizon.Addresses.Address
|
2022-04-18 12:38:57 +00:00
|
|
|
alias Mobilizon.Events.{Event, EventOptions, Participant}
|
2020-07-09 15:24:28 +00:00
|
|
|
alias Mobilizon.Posts.Post
|
2020-12-15 16:17:42 +00:00
|
|
|
alias Mobilizon.Web.Endpoint
|
2020-01-26 20:36:50 +00:00
|
|
|
alias Mobilizon.Web.JsonLD.ObjectView
|
2021-10-13 10:57:54 +00:00
|
|
|
alias Mobilizon.Web.Router.Helpers, as: Routes
|
2019-03-04 17:38:30 +00:00
|
|
|
|
2021-11-15 11:11:29 +00:00
|
|
|
import Mobilizon.Service.Metadata.Utils,
|
|
|
|
only: [process_description: 3]
|
|
|
|
|
2021-09-27 07:41:36 +00:00
|
|
|
@spec render(String.t(), map()) :: map()
|
2020-09-02 08:00:39 +00:00
|
|
|
def render("group.json", %{group: %Actor{} = group}) do
|
2021-10-21 15:55:16 +00:00
|
|
|
res = %{
|
2020-09-02 08:00:39 +00:00
|
|
|
"@context" => "http://schema.org",
|
|
|
|
"@type" => "Organization",
|
|
|
|
"url" => group.url,
|
|
|
|
"name" => group.name || group.preferred_username,
|
|
|
|
"address" => render_address(group)
|
|
|
|
}
|
2021-10-21 15:55:16 +00:00
|
|
|
|
|
|
|
res =
|
|
|
|
if group.banner do
|
|
|
|
Map.put(res, "image", group.banner.url)
|
|
|
|
else
|
|
|
|
res
|
|
|
|
end
|
|
|
|
|
|
|
|
if group.physical_address do
|
|
|
|
Map.put(
|
|
|
|
res,
|
|
|
|
"address",
|
|
|
|
render_one(group.physical_address, ObjectView, "address.json", as: :address)
|
|
|
|
)
|
|
|
|
else
|
|
|
|
res
|
|
|
|
end
|
2020-09-02 08:00:39 +00:00
|
|
|
end
|
|
|
|
|
2019-03-04 17:38:30 +00:00
|
|
|
def render("event.json", %{event: %Event{} = event}) do
|
2020-07-09 15:24:28 +00:00
|
|
|
organizer = %{
|
|
|
|
"@type" => if(event.organizer_actor.type == :Group, do: "Organization", else: "Person"),
|
|
|
|
"name" => Actor.display_name(event.organizer_actor)
|
|
|
|
}
|
2019-05-22 12:12:11 +00:00
|
|
|
|
2021-10-13 10:57:54 +00:00
|
|
|
organizer =
|
|
|
|
if event.organizer_actor.avatar do
|
|
|
|
Map.put(organizer, "image", event.organizer_actor.avatar.url)
|
|
|
|
else
|
|
|
|
organizer
|
|
|
|
end
|
|
|
|
|
2022-02-08 14:27:53 +00:00
|
|
|
participant_count = Mobilizon.Events.count_participant_participants(event.id)
|
|
|
|
|
2019-03-04 17:38:30 +00:00
|
|
|
json_ld = %{
|
|
|
|
"@context" => "https://schema.org",
|
|
|
|
"@type" => "Event",
|
|
|
|
"name" => event.title,
|
2021-11-15 11:11:29 +00:00
|
|
|
"description" => process_description(event.description, "en", nil),
|
2020-07-09 15:24:28 +00:00
|
|
|
# We assume for now performer == organizer
|
|
|
|
"performer" => organizer,
|
|
|
|
"organizer" => organizer,
|
2021-11-15 11:11:29 +00:00
|
|
|
"location" => render_all_locations(event),
|
|
|
|
"eventAttendanceMode" => event |> attendance_mode() |> event_attendance_mode(),
|
2022-02-08 14:27:53 +00:00
|
|
|
"maximumAttendeeCapacity" => event.options.maximum_attendee_capacity,
|
|
|
|
"remainingAttendeeCapacity" =>
|
|
|
|
remaining_attendee_capacity(event.options, participant_count),
|
2020-07-09 15:24:28 +00:00
|
|
|
"eventStatus" =>
|
|
|
|
if(event.status == :cancelled,
|
|
|
|
do: "https://schema.org/EventCancelled",
|
|
|
|
else: "https://schema.org/EventScheduled"
|
2020-09-02 15:42:17 +00:00
|
|
|
),
|
|
|
|
"image" =>
|
|
|
|
if(event.picture,
|
|
|
|
do: [
|
2020-12-15 16:17:42 +00:00
|
|
|
event.picture.file.url
|
2020-09-02 15:42:17 +00:00
|
|
|
],
|
|
|
|
else: ["#{Endpoint.url()}/img/mobilizon_default_card.png"]
|
2020-07-09 15:24:28 +00:00
|
|
|
)
|
2019-03-04 17:38:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
json_ld =
|
|
|
|
if event.begins_on,
|
|
|
|
do: Map.put(json_ld, "startDate", DateTime.to_iso8601(event.begins_on)),
|
|
|
|
else: json_ld
|
|
|
|
|
|
|
|
json_ld =
|
|
|
|
if event.ends_on,
|
|
|
|
do: Map.put(json_ld, "endDate", DateTime.to_iso8601(event.ends_on)),
|
|
|
|
else: json_ld
|
|
|
|
|
|
|
|
json_ld
|
|
|
|
end
|
|
|
|
|
|
|
|
def render("place.json", %{address: %Address{} = address}) do
|
|
|
|
%{
|
|
|
|
"@type" => "Place",
|
|
|
|
"name" => address.description,
|
2020-09-02 08:00:39 +00:00
|
|
|
"address" => render_one(address, ObjectView, "address.json", as: :address)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def render("address.json", %{address: %Address{} = address}) do
|
|
|
|
%{
|
|
|
|
"@type" => "PostalAddress",
|
|
|
|
"streetAddress" => address.street,
|
|
|
|
"addressLocality" => address.locality,
|
|
|
|
"postalCode" => address.postal_code,
|
|
|
|
"addressRegion" => address.region,
|
|
|
|
"addressCountry" => address.country
|
2019-03-04 17:38:30 +00:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2020-07-09 15:24:28 +00:00
|
|
|
def render("post.json", %{post: %Post{} = post}) do
|
|
|
|
%{
|
|
|
|
"@context" => "https://schema.org",
|
|
|
|
"@type" => "Article",
|
|
|
|
"name" => post.title,
|
2021-10-21 15:55:16 +00:00
|
|
|
"headline" => post.title,
|
2020-07-09 15:24:28 +00:00
|
|
|
"author" => %{
|
|
|
|
"@type" => "Organization",
|
2021-10-21 15:55:16 +00:00
|
|
|
"name" => Actor.display_name(post.attributed_to),
|
|
|
|
"url" =>
|
|
|
|
Endpoint
|
|
|
|
|> Routes.page_url(
|
|
|
|
:actor,
|
|
|
|
Actor.preferred_username_and_domain(post.attributed_to)
|
|
|
|
)
|
|
|
|
|> URI.decode()
|
2020-07-09 15:24:28 +00:00
|
|
|
},
|
|
|
|
"datePublished" => post.publish_at,
|
2021-10-21 15:55:16 +00:00
|
|
|
"dateModified" => post.updated_at,
|
|
|
|
"image" =>
|
|
|
|
if(post.picture,
|
|
|
|
do: [
|
|
|
|
post.picture.file.url
|
|
|
|
],
|
|
|
|
else: ["#{Endpoint.url()}/img/mobilizon_default_card.png"]
|
|
|
|
)
|
2020-07-09 15:24:28 +00:00
|
|
|
}
|
|
|
|
end
|
2020-08-10 14:22:15 +00:00
|
|
|
|
2021-10-13 10:57:54 +00:00
|
|
|
def render("participation.json", %{
|
|
|
|
participant: %Participant{} = participant
|
|
|
|
}) do
|
|
|
|
res = %{
|
|
|
|
"@context" => "http://schema.org",
|
|
|
|
"@type" => "EventReservation",
|
|
|
|
"underName" => %{
|
|
|
|
"@type" => "Person",
|
|
|
|
"name" => participant.actor.name || participant.actor.preferred_username
|
|
|
|
},
|
|
|
|
"reservationFor" => render("event.json", %{event: participant.event}),
|
|
|
|
"reservationStatus" => reservation_status(participant.role),
|
|
|
|
"modifiedTime" => participant.updated_at,
|
|
|
|
"modifyReservationUrl" => Routes.page_url(Endpoint, :event, participant.event.uuid)
|
|
|
|
}
|
|
|
|
|
|
|
|
if participant.code do
|
|
|
|
Map.put(res, "reservationNumber", participant.code)
|
|
|
|
else
|
|
|
|
res
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-04-18 12:38:57 +00:00
|
|
|
@spec reservation_status(atom()) :: String.t()
|
2021-10-13 10:57:54 +00:00
|
|
|
defp reservation_status(:rejected), do: "https://schema.org/ReservationCancelled"
|
|
|
|
defp reservation_status(:not_confirmed), do: "https://schema.org/ReservationPending"
|
|
|
|
defp reservation_status(:not_approved), do: "https://schema.org/ReservationHold"
|
|
|
|
defp reservation_status(_), do: "https://schema.org/ReservationConfirmed"
|
|
|
|
|
2021-11-15 11:11:29 +00:00
|
|
|
defp render_all_locations(%Event{} = event) do
|
|
|
|
[]
|
|
|
|
|> render_location(event)
|
|
|
|
|> render_virtual_location(event)
|
2022-08-23 08:09:10 +00:00
|
|
|
|> maybe_render_single_element()
|
2021-11-15 11:11:29 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
@spec render_location(list(), map()) :: list()
|
|
|
|
defp render_location(locations, %{physical_address: %Address{} = address}),
|
|
|
|
do: locations ++ [render_one(address, ObjectView, "place.json", as: :address)]
|
|
|
|
|
|
|
|
defp render_location(locations, _), do: locations
|
2020-08-10 14:22:15 +00:00
|
|
|
|
|
|
|
# For now the Virtual Location of an event is it's own URL,
|
|
|
|
# but in the future it will be a special field
|
2021-11-15 11:11:29 +00:00
|
|
|
defp render_virtual_location(locations, %Event{
|
|
|
|
url: event_url,
|
|
|
|
metadata: metadata,
|
|
|
|
options: %EventOptions{is_online: is_online}
|
|
|
|
}) do
|
|
|
|
links = virtual_location_links(metadata)
|
|
|
|
fallback_links = if is_online, do: [event_url], else: []
|
|
|
|
links = if length(links) > 0, do: Enum.map(links, & &1.value), else: fallback_links
|
|
|
|
|
|
|
|
locations ++
|
|
|
|
Enum.map(
|
|
|
|
links,
|
|
|
|
&%{
|
|
|
|
"@type" => "VirtualLocation",
|
|
|
|
"url" => &1
|
|
|
|
}
|
|
|
|
)
|
2020-08-10 14:22:15 +00:00
|
|
|
end
|
2020-09-02 08:00:39 +00:00
|
|
|
|
2021-11-15 11:11:29 +00:00
|
|
|
defp render_virtual_location(locations, _), do: locations
|
2020-09-02 08:00:39 +00:00
|
|
|
|
2022-08-23 08:09:10 +00:00
|
|
|
@spec maybe_render_single_element(list(map())) :: list(map()) | map()
|
|
|
|
defp maybe_render_single_element([location]), do: location
|
|
|
|
defp maybe_render_single_element(locations), do: locations
|
|
|
|
|
2020-09-02 08:00:39 +00:00
|
|
|
defp render_address(%{physical_address: %Address{} = address}),
|
|
|
|
do: render_one(address, ObjectView, "address.json", as: :address)
|
|
|
|
|
|
|
|
defp render_address(_), do: nil
|
2021-11-15 10:08:18 +00:00
|
|
|
|
2021-11-15 11:11:29 +00:00
|
|
|
defp event_attendance_mode(:online), do: "https://schema.org/OnlineEventAttendanceMode"
|
|
|
|
defp event_attendance_mode(:offline), do: "https://schema.org/OfflineEventAttendanceMode"
|
|
|
|
defp event_attendance_mode(:mixed), do: "https://schema.org/MixedEventAttendanceMode"
|
2021-11-15 10:08:18 +00:00
|
|
|
|
2021-11-15 11:11:29 +00:00
|
|
|
defp attendance_mode(%Event{options: %EventOptions{is_online: true}}),
|
|
|
|
do: :online
|
2021-11-15 10:08:18 +00:00
|
|
|
|
2021-11-15 11:11:29 +00:00
|
|
|
defp attendance_mode(%Event{physical_address: %Address{}, metadata: metadata}) do
|
|
|
|
if metadata |> virtual_location_links() |> length() > 0 do
|
|
|
|
:mixed
|
2021-11-15 10:08:18 +00:00
|
|
|
else
|
2021-11-15 11:11:29 +00:00
|
|
|
:offline
|
2021-11-15 10:08:18 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-11-15 11:11:29 +00:00
|
|
|
defp attendance_mode(%Event{}),
|
|
|
|
do: :offline
|
|
|
|
|
|
|
|
@livestream_keys ["mz:live", "mz:visio"]
|
|
|
|
@spec virtual_location_links(list()) :: list()
|
|
|
|
defp virtual_location_links(metadata),
|
|
|
|
do: Enum.filter(metadata, &String.contains?(&1.key, @livestream_keys))
|
2022-02-08 14:27:53 +00:00
|
|
|
|
|
|
|
# TODO: Make this in common with Mobilizon.Federation.ActivityStream.Converter.Event
|
|
|
|
@spec remaining_attendee_capacity(map(), integer()) :: integer() | nil
|
|
|
|
defp remaining_attendee_capacity(
|
|
|
|
%{maximum_attendee_capacity: maximum_attendee_capacity},
|
|
|
|
participant_count
|
|
|
|
)
|
|
|
|
when is_integer(maximum_attendee_capacity) and maximum_attendee_capacity > 0 do
|
|
|
|
maximum_attendee_capacity - participant_count
|
|
|
|
end
|
|
|
|
|
|
|
|
defp remaining_attendee_capacity(
|
|
|
|
%{maximum_attendee_capacity: _},
|
|
|
|
_participant_count
|
|
|
|
),
|
|
|
|
do: nil
|
2019-03-04 17:38:30 +00:00
|
|
|
end
|