Issue #1567 : Add a long_event computed field to an event

The actual long_event implementation is only done for search and long_event is a parameter of the search request.

This change is needed for the the front-end to know if an event is a long_event everywhere an event is received.

The computed field (Ecto virtual field) is set after the Ecto request with the function with_virtual_fields().

with_virtual_fields() handles cases where there is an event, a list of events and a paginated list of events returned.
This commit is contained in:
Massedil 2024-11-21 19:52:57 +01:00 committed by setop
parent 9796304a9a
commit 752da9e641
4 changed files with 76 additions and 1 deletions

View file

@ -32,6 +32,11 @@ defmodule Mobilizon.GraphQL.Schema.EventType do
field(:description, :string, description: "The event's description") field(:description, :string, description: "The event's description")
field(:begins_on, :datetime, description: "Datetime for when the event begins") field(:begins_on, :datetime, description: "Datetime for when the event begins")
field(:ends_on, :datetime, description: "Datetime for when the event ends") field(:ends_on, :datetime, description: "Datetime for when the event ends")
field(:long_event, :boolean,
description: "Whether the event is a long event (activity) or not"
)
field(:status, :event_status, description: "Status of the event") field(:status, :event_status, description: "Status of the event")
field(:visibility, :event_visibility, description: "The event's visibility") field(:visibility, :event_visibility, description: "The event's visibility")
field(:join_options, :event_join_options, description: "The event's visibility") field(:join_options, :event_join_options, description: "The event's visibility")

View file

@ -17,6 +17,11 @@ defmodule Mobilizon.GraphQL.Schema.SearchType do
field(:title, :string, description: "The event's title") field(:title, :string, description: "The event's title")
field(:begins_on, :datetime, description: "Datetime for when the event begins") field(:begins_on, :datetime, description: "Datetime for when the event begins")
field(:ends_on, :datetime, description: "Datetime for when the event ends") field(:ends_on, :datetime, description: "Datetime for when the event ends")
field(:long_event, :boolean,
description: "Whether the event is a long event (activity) or not"
)
field(:status, :event_status, description: "Status of the event") field(:status, :event_status, description: "Status of the event")
field(:picture, :media, description: "The event's picture") field(:picture, :media, description: "The event's picture")
field(:physical_address, :address, description: "The event's physical address") field(:physical_address, :address, description: "The event's physical address")
@ -52,6 +57,11 @@ defmodule Mobilizon.GraphQL.Schema.SearchType do
field(:title, :string, description: "The event's title") field(:title, :string, description: "The event's title")
field(:begins_on, :datetime, description: "Datetime for when the event begins") field(:begins_on, :datetime, description: "Datetime for when the event begins")
field(:ends_on, :datetime, description: "Datetime for when the event ends") field(:ends_on, :datetime, description: "Datetime for when the event ends")
field(:long_event, :boolean,
description: "Whether the event is a long event (activity) or not"
)
field(:status, :event_status, description: "Status of the event") field(:status, :event_status, description: "Status of the event")
field(:picture, :media, description: "The event's picture") field(:picture, :media, description: "The event's picture")
field(:physical_address, :address, description: "The event's physical address") field(:physical_address, :address, description: "The event's physical address")

View file

@ -66,6 +66,7 @@ defmodule Mobilizon.Events.Event do
participants: [Actor.t()], participants: [Actor.t()],
contacts: [Actor.t()], contacts: [Actor.t()],
language: String.t(), language: String.t(),
long_event: boolean,
metadata: [EventMetadata.t()] metadata: [EventMetadata.t()]
} }
@ -89,7 +90,8 @@ defmodule Mobilizon.Events.Event do
:picture_id, :picture_id,
:physical_address_id, :physical_address_id,
:attributed_to_id, :attributed_to_id,
:language :language,
:long_event
] ]
@attrs @required_attrs ++ @optional_attrs @attrs @required_attrs ++ @optional_attrs
@ -102,6 +104,7 @@ defmodule Mobilizon.Events.Event do
field(:slug, :string) field(:slug, :string)
field(:description, :string) field(:description, :string)
field(:ends_on, :utc_datetime) field(:ends_on, :utc_datetime)
field(:long_event, :boolean, virtual: true, default: nil)
field(:title, :string) field(:title, :string)
field(:status, EventStatus, default: :confirmed) field(:status, EventStatus, default: :confirmed)
field(:draft, :boolean, default: false) field(:draft, :boolean, default: false)

View file

@ -12,6 +12,8 @@ defmodule Mobilizon.Events do
import Mobilizon.Storage.Ecto import Mobilizon.Storage.Ecto
import Mobilizon.Events.Utils, only: [calculate_notification_time: 1] import Mobilizon.Events.Utils, only: [calculate_notification_time: 1]
require Logger
alias Ecto.{Changeset, Multi} alias Ecto.{Changeset, Multi}
alias Mobilizon.Actors.{Actor, Follower} alias Mobilizon.Actors.{Actor, Follower}
@ -141,6 +143,7 @@ defmodule Mobilizon.Events do
url url
|> event_by_url_query() |> event_by_url_query()
|> Repo.one() |> Repo.one()
|> with_virtual_fields()
end end
@doc """ @doc """
@ -153,6 +156,7 @@ defmodule Mobilizon.Events do
|> event_by_url_query() |> event_by_url_query()
|> preload_for_event() |> preload_for_event()
|> Repo.one!() |> Repo.one!()
|> with_virtual_fields()
end end
@doc """ @doc """
@ -167,6 +171,7 @@ defmodule Mobilizon.Events do
|> filter_draft() |> filter_draft()
|> preload_for_event() |> preload_for_event()
|> Repo.one!() |> Repo.one!()
|> with_virtual_fields()
end end
@doc """ @doc """
@ -180,6 +185,7 @@ defmodule Mobilizon.Events do
|> filter_draft() |> filter_draft()
|> preload_for_event() |> preload_for_event()
|> Repo.one() |> Repo.one()
|> with_virtual_fields()
end end
@spec check_if_event_has_instance_follow(String.t(), integer()) :: boolean() @spec check_if_event_has_instance_follow(String.t(), integer()) :: boolean()
@ -199,6 +205,7 @@ defmodule Mobilizon.Events do
|> event_by_uuid_query() |> event_by_uuid_query()
|> preload_for_event() |> preload_for_event()
|> Repo.one() |> Repo.one()
|> with_virtual_fields()
end end
@doc """ @doc """
@ -212,6 +219,7 @@ defmodule Mobilizon.Events do
|> filter_not_event_uuid(not_event_uuid) |> filter_not_event_uuid(not_event_uuid)
|> filter_draft() |> filter_draft()
|> Repo.one() |> Repo.one()
|> with_virtual_fields()
end end
@doc """ @doc """
@ -392,6 +400,7 @@ defmodule Mobilizon.Events do
|> filter_cancelled_events() |> filter_cancelled_events()
|> filter_local_or_from_followed_instances_events() |> filter_local_or_from_followed_instances_events()
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@spec stream_events_for_sitemap :: Enum.t() @spec stream_events_for_sitemap :: Enum.t()
@ -413,6 +422,7 @@ defmodule Mobilizon.Events do
|> preload_for_event() |> preload_for_event()
|> event_order_by(sort, direction) |> event_order_by(sort, direction)
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@doc """ @doc """
@ -425,6 +435,7 @@ defmodule Mobilizon.Events do
|> events_by_tags_query(limit) |> events_by_tags_query(limit)
|> filter_draft() |> filter_draft()
|> Repo.all() |> Repo.all()
|> with_virtual_fields()
end end
@doc """ @doc """
@ -440,6 +451,7 @@ defmodule Mobilizon.Events do
actor_id actor_id
|> do_list_public_events_for_actor() |> do_list_public_events_for_actor()
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@doc """ @doc """
@ -467,6 +479,7 @@ defmodule Mobilizon.Events do
|> do_list_public_events_for_actor() |> do_list_public_events_for_actor()
|> event_filter_begins_on(DateTime.utc_now(), nil) |> event_filter_begins_on(DateTime.utc_now(), nil)
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@spec do_list_public_events_for_actor(integer()) :: Ecto.Query.t() @spec do_list_public_events_for_actor(integer()) :: Ecto.Query.t()
@ -485,6 +498,7 @@ defmodule Mobilizon.Events do
|> event_for_actor_query(desc: :begins_on) |> event_for_actor_query(desc: :begins_on)
|> preload_for_event() |> preload_for_event()
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@spec list_simple_organized_events_for_group(Actor.t(), integer | nil, integer | nil) :: @spec list_simple_organized_events_for_group(Actor.t(), integer | nil, integer | nil) ::
@ -520,6 +534,7 @@ defmodule Mobilizon.Events do
|> event_order_by(order_by, order_direction) |> event_order_by(order_by, order_direction)
|> preload_for_event() |> preload_for_event()
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@spec list_drafts_for_user(integer, integer | nil, integer | nil) :: Page.t(Event.t()) @spec list_drafts_for_user(integer, integer | nil, integer | nil) :: Page.t(Event.t())
@ -529,6 +544,7 @@ defmodule Mobilizon.Events do
|> filter_draft(true) |> filter_draft(true)
|> order_by(desc: :updated_at) |> order_by(desc: :updated_at)
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@spec user_moderator_for_event?(integer | String.t(), integer | String.t()) :: boolean @spec user_moderator_for_event?(integer | String.t(), integer | String.t()) :: boolean
@ -552,6 +568,7 @@ defmodule Mobilizon.Events do
|> close_events_query(radius) |> close_events_query(radius)
|> filter_draft() |> filter_draft()
|> Repo.all() |> Repo.all()
|> with_virtual_fields()
end end
@doc """ @doc """
@ -603,6 +620,7 @@ defmodule Mobilizon.Events do
|> filter_public_visibility() |> filter_public_visibility()
|> event_order(Map.get(args, :sort_by, :match_desc), search_string) |> event_order(Map.get(args, :sort_by, :match_desc), search_string)
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@doc """ @doc """
@ -978,6 +996,7 @@ defmodule Mobilizon.Events do
actor_id actor_id
|> event_participations_for_actor_query() |> event_participations_for_actor_query()
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@doc """ @doc """
@ -993,6 +1012,7 @@ defmodule Mobilizon.Events do
actor_id actor_id
|> event_participations_for_actor_query(DateTime.utc_now()) |> event_participations_for_actor_query(DateTime.utc_now())
|> Page.build_page(page, limit) |> Page.build_page(page, limit)
|> with_virtual_fields()
end end
@doc """ @doc """
@ -1871,6 +1891,7 @@ defmodule Mobilizon.Events do
) )
) )
|> Repo.all() |> Repo.all()
|> with_virtual_fields()
end end
@spec list_participations_for_user_query(integer()) :: Ecto.Query.t() @spec list_participations_for_user_query(integer()) :: Ecto.Query.t()
@ -2098,4 +2119,40 @@ defmodule Mobilizon.Events do
|> preload_for_event() |> preload_for_event()
|> Page.chunk(chunk_size) |> Page.chunk(chunk_size)
end end
# Handling the case where Repo.XXXX() return nil
def with_virtual_fields(nil), do: nil
# Handling the case where there is an event
# Using Repo.one(), for example
def with_virtual_fields(%Event{} = event) do
duration = Config.get([:instance, :duration_of_long_event], 0)
event_duration = DateTime.diff(event.ends_on, event.begins_on, :day)
# duration need to be > 0 for long event to be activated
long_event = duration > 0 && event_duration > duration
%{event | long_event: long_event}
end
# Handling the case where there is a list of events
# Using Repo.all(), for example
def with_virtual_fields(events) when is_list(events) do
Enum.map(events, &with_virtual_fields/1)
end
# Handling the case of a paginated list of events
def with_virtual_fields(%Page{total: _total, elements: elements} = page) do
elements_with_virtual_fields = Enum.map(elements, &with_virtual_fields/1)
%{page | elements: elements_with_virtual_fields}
end
# In case the function is called on an element without virtual_fields
def with_virtual_fields(invalid) do
Logger.warning("with_virtual_fields called on invalid element : #{inspect(invalid)}")
# Return the element without modification
invalid
end
end end