feat(reports): allow reports to hold multiple events

Signed-off-by: Thomas Citharel <tcit@tcit.fr>
This commit is contained in:
Thomas Citharel 2023-06-05 18:32:29 +02:00
parent 538139eefa
commit f2ac3e2e5d
No known key found for this signature in database
GPG key ID: A061B9DDE0CA0773
21 changed files with 185 additions and 104 deletions

View file

@ -352,7 +352,7 @@ const reportComment = async (
): Promise<void> => { ): Promise<void> => {
if (!props.comment.actor) return; if (!props.comment.actor) return;
createReportMutation({ createReportMutation({
eventId: props.event.id, eventsIds: [props.event.id ?? ""],
reportedId: props.comment.actor?.id ?? "", reportedId: props.comment.actor?.id ?? "",
commentsIds: [props.comment.id ?? ""], commentsIds: [props.comment.id ?? ""],
content, content,

View file

@ -648,7 +648,7 @@ const reportEvent = async (
if (!organizer.value) return; if (!organizer.value) return;
createReportMutation({ createReportMutation({
eventId: event.value?.id ?? "", eventsIds: [event.value?.id ?? ""],
reportedId: organizer.value?.id ?? "", reportedId: organizer.value?.id ?? "",
content, content,
forward, forward,

View file

@ -5,7 +5,7 @@ export function useCreateReport() {
return useMutation< return useMutation<
{ createReport: { id: string } }, { createReport: { id: string } },
{ {
eventId?: string; eventsIds?: string[];
reportedId: string; reportedId: string;
content?: string; content?: string;
commentsIds?: string[]; commentsIds?: string[];

View file

@ -20,7 +20,7 @@ export const REPORTS = gql`
...ActorFragment ...ActorFragment
suspended suspended
} }
event { events {
id id
uuid uuid
title title
@ -46,7 +46,7 @@ const REPORT_FRAGMENT = gql`
reporter { reporter {
...ActorFragment ...ActorFragment
} }
event { events {
id id
uuid uuid
title title
@ -97,14 +97,14 @@ export const REPORT = gql`
export const CREATE_REPORT = gql` export const CREATE_REPORT = gql`
mutation CreateReport( mutation CreateReport(
$eventId: ID $eventsIds: [ID]
$reportedId: ID! $reportedId: ID!
$content: String $content: String
$commentsIds: [ID] $commentsIds: [ID]
$forward: Boolean $forward: Boolean
) { ) {
createReport( createReport(
eventId: $eventId eventsIds: $eventsIds
reportedId: $reportedId reportedId: $reportedId
content: $content content: $content
commentsIds: $commentsIds commentsIds: $commentsIds

View file

@ -16,7 +16,7 @@ export interface IReport extends IActionLogObject {
id: string; id: string;
reported: IActor; reported: IActor;
reporter: IPerson; reporter: IPerson;
event?: IEvent; events?: IEvent[];
comments: IComment[]; comments: IComment[];
content: string; content: string;
notes: IReportNote[]; notes: IReportNote[];

View file

@ -150,14 +150,14 @@
<span v-else>{{ t("Unknown") }}</span> <span v-else>{{ t("Unknown") }}</span>
</td> </td>
</tr> </tr>
<tr v-if="report.event && report.comments.length > 0"> <!-- <tr v-if="report.events && report.comments.length > 0">
<td>{{ t("Event") }}</td> <td>{{ t("Events") }}</td>
<td class="flex gap-2 items-center"> <td class="flex gap-2 items-center">
<router-link <router-link
class="underline" class="underline"
:to="{ :to="{
name: RouteName.EVENT, name: RouteName.EVENT,
params: { uuid: report.event.uuid }, params: { uuid: report.events.uuid },
}" }"
> >
{{ report.event.title }} {{ report.event.title }}
@ -169,7 +169,7 @@
>{{ t("Delete") }}</o-button >{{ t("Delete") }}</o-button
> >
</td> </td>
</tr> </tr> -->
</tbody> </tbody>
</table> </table>
</section> </section>
@ -206,17 +206,23 @@
<section <section
class="bg-white dark:bg-zinc-700 rounded px-2 pt-1 pb-2 my-3" class="bg-white dark:bg-zinc-700 rounded px-2 pt-1 pb-2 my-3"
v-if="report.event && report.comments.length === 0" v-if="
report.events &&
report.events?.length > 0 &&
report.comments.length === 0
"
> >
<h2 class="mb-1">{{ t("Reported content") }}</h2> <h2 class="mb-1">{{ t("Reported content") }}</h2>
<EventCard :event="report.event" mode="row" class="my-2 max-w-4xl" /> <div v-for="event in report.events" :key="event.id">
<o-button <EventCard :event="event" mode="row" class="my-2 max-w-4xl" />
variant="danger" <o-button
@click="confirmEventDelete()" variant="danger"
icon-left="delete" @click="confirmEventDelete(event)"
size="small" icon-left="delete"
>{{ t("Delete") }}</o-button size="small"
> >{{ t("Delete") }}</o-button
>
</div>
</section> </section>
<section <section
@ -337,6 +343,7 @@ import { Dialog } from "@/plugins/dialog";
import { Notifier } from "@/plugins/notifier"; import { Notifier } from "@/plugins/notifier";
import EventCard from "@/components/Event/EventCard.vue"; import EventCard from "@/components/Event/EventCard.vue";
import { useFeatures } from "@/composition/apollo/config"; import { useFeatures } from "@/composition/apollo/config";
import { IEvent } from "@/types/event.model";
const router = useRouter(); const router = useRouter();
@ -419,7 +426,7 @@ createReportNoteMutationError((error) => {
const dialog = inject<Dialog>("dialog"); const dialog = inject<Dialog>("dialog");
const confirmEventDelete = (): void => { const confirmEventDelete = (event: IEvent): void => {
dialog?.confirm({ dialog?.confirm({
title: t("Deleting event"), title: t("Deleting event"),
message: t( message: t(
@ -428,7 +435,7 @@ const confirmEventDelete = (): void => {
confirmText: t("Delete Event"), confirmText: t("Delete Event"),
variant: "danger", variant: "danger",
hasIcon: true, hasIcon: true,
onConfirm: () => deleteEvent(), onConfirm: () => deleteEvent(event),
}); });
}; };
@ -451,8 +458,8 @@ const {
onError: deleteEventMutationError, onError: deleteEventMutationError,
} = useMutation<{ deleteEvent: { id: string } }>(DELETE_EVENT); } = useMutation<{ deleteEvent: { id: string } }>(DELETE_EVENT);
deleteEventMutationDone(() => { deleteEventMutationDone((result) => {
const eventTitle = report.value?.event?.title; const eventTitle = result?.context?.eventTitle;
notifier?.success( notifier?.success(
t("Event {eventTitle} deleted", { t("Event {eventTitle} deleted", {
eventTitle, eventTitle,
@ -464,10 +471,13 @@ deleteEventMutationError((error) => {
console.error(error); console.error(error);
}); });
const deleteEvent = async (): Promise<void> => { const deleteEvent = async (event: IEvent): Promise<void> => {
if (!report.value?.event?.id) return; if (!event?.id) return;
deleteEventMutation({ eventId: report.value.event.id }); deleteEventMutation(
{ eventId: event.id },
{ context: { eventTitle: event.title } }
);
}; };
const { const {

View file

@ -2,7 +2,6 @@ defmodule Mobilizon.Federation.ActivityPub.Types.Reports do
@moduledoc false @moduledoc false
alias Mobilizon.{Actors, Discussions, Events, Reports} alias Mobilizon.{Actors, Discussions, Events, Reports}
alias Mobilizon.Actors.Actor alias Mobilizon.Actors.Actor
alias Mobilizon.Events.Event
alias Mobilizon.Federation.ActivityStream alias Mobilizon.Federation.ActivityStream
alias Mobilizon.Federation.ActivityStream.Convertible alias Mobilizon.Federation.ActivityStream.Convertible
alias Mobilizon.Reports.Report alias Mobilizon.Reports.Report
@ -26,15 +25,16 @@ defmodule Mobilizon.Federation.ActivityPub.Types.Reports do
%Actor{} = reported_actor = Actors.get_actor!(args.reported_id) %Actor{} = reported_actor = Actors.get_actor!(args.reported_id)
content = HTML.strip_tags(args.content) content = HTML.strip_tags(args.content)
event_id = Map.get(args, :event_id) events =
args
event = |> Map.get(:events_ids, [])
if is_nil(event_id) do |> Enum.map(fn event_id ->
nil case Events.get_event(event_id) do
else {:ok, event} -> event
{:ok, %Event{} = event} = Events.get_event(event_id) {:error, :event_not_found} -> nil
event end
end end)
|> Enum.filter(& &1)
comments = comments =
Discussions.list_comments_by_actor_and_ids( Discussions.list_comments_by_actor_and_ids(
@ -46,7 +46,7 @@ defmodule Mobilizon.Federation.ActivityPub.Types.Reports do
reporter: reporter_actor, reporter: reporter_actor,
reported: reported_actor, reported: reported_actor,
content: content, content: content,
event: event, events: events,
comments: comments comments: comments
}) })
end end

View file

@ -9,11 +9,11 @@ defmodule Mobilizon.Federation.ActivityStream.Converter.Flag do
""" """
alias Mobilizon.Actors.Actor alias Mobilizon.Actors.Actor
alias Mobilizon.Discussions alias Mobilizon.Discussions.Comment
alias Mobilizon.Events
alias Mobilizon.Events.Event alias Mobilizon.Events.Event
alias Mobilizon.Reports.Report alias Mobilizon.Reports.Report
alias Mobilizon.Federation.ActivityPub
alias Mobilizon.Federation.ActivityPub.Actor, as: ActivityPubActor alias Mobilizon.Federation.ActivityPub.Actor, as: ActivityPubActor
alias Mobilizon.Federation.ActivityPub.Relay alias Mobilizon.Federation.ActivityPub.Relay
alias Mobilizon.Federation.ActivityStream.{Converter, Convertible} alias Mobilizon.Federation.ActivityStream.{Converter, Convertible}
@ -38,7 +38,7 @@ defmodule Mobilizon.Federation.ActivityStream.Converter.Flag do
"uri" => params["uri"], "uri" => params["uri"],
"content" => params["content"], "content" => params["content"],
"reported_id" => params["reported"].id, "reported_id" => params["reported"].id,
"event_id" => (!is_nil(params["event"]) && params["event"].id) || nil, "events" => params["events"],
"comments" => params["comments"] "comments" => params["comments"]
} }
end end
@ -50,9 +50,10 @@ defmodule Mobilizon.Federation.ActivityStream.Converter.Flag do
@impl Converter @impl Converter
@spec model_to_as(Report.t()) :: map @spec model_to_as(Report.t()) :: map
def model_to_as(%Report{} = report) do def model_to_as(%Report{} = report) do
object = [report.reported.url] ++ Enum.map(report.comments, fn comment -> comment.url end) object =
[report.reported.url] ++
object = if report.event, do: object ++ [report.event.url], else: object Enum.map(report.comments, fn comment -> comment.url end) ++
Enum.map(report.events, & &1.url)
%{ %{
"type" => "Flag", "type" => "Flag",
@ -68,14 +69,13 @@ defmodule Mobilizon.Federation.ActivityStream.Converter.Flag do
with {:ok, %Actor{} = reporter} <- with {:ok, %Actor{} = reporter} <-
ActivityPubActor.get_or_fetch_actor_by_url(object["actor"]), ActivityPubActor.get_or_fetch_actor_by_url(object["actor"]),
%Actor{} = reported <- find_reported(objects), %Actor{} = reported <- find_reported(objects),
event <- find_event(objects), %{events: events, comments: comments} <- find_events_and_comments(objects) do
comments <- find_comments(objects, reported, event) do
%{ %{
"reporter" => reporter, "reporter" => reporter,
"uri" => object["id"], "uri" => object["id"],
"content" => object["content"], "content" => object["content"],
"reported" => reported, "reported" => reported,
"event" => event, "events" => events,
"comments" => comments "comments" => comments
} }
end end
@ -94,26 +94,19 @@ defmodule Mobilizon.Federation.ActivityStream.Converter.Flag do
end) end)
end end
# Remove the reported actor and the event from the object list. defp find_events_and_comments(objects) do
@spec find_comments(list(String.t()), Actor.t() | nil, Event.t() | nil) :: list(Comment.t())
defp find_comments(objects, reported, event) do
objects objects
|> Enum.filter(fn url -> |> Enum.map(&ActivityPub.fetch_object_from_url/1)
!((!is_nil(reported) && url == reported.url) || (!is_nil(event) && event.url == url)) |> Enum.reduce(%{comments: [], events: []}, fn res, acc ->
end) case res do
|> Enum.map(&Discussions.get_comment_from_url/1) {:ok, %Event{} = event} ->
|> Enum.filter(& &1) Map.put(acc, :events, [event | acc.events])
end
@spec find_event(list(String.t())) :: Event.t() | nil {:ok, %Comment{} = comment} ->
defp find_event(objects) do Map.put(acc, :comments, [comment | acc.comments])
Enum.reduce_while(objects, nil, fn url, _ ->
case Events.get_event_by_url(url) do
%Event{} = event ->
{:halt, event}
_ -> _ ->
{:cont, nil} acc
end end
end) end)
end end

View file

@ -67,7 +67,7 @@ defmodule Mobilizon.GraphQL.Resolvers.Report do
{:ok, _, %Report{} = report} -> {:ok, _, %Report{} = report} ->
{:ok, report} {:ok, report}
_error -> error ->
{:error, dgettext("errors", "Error while saving report")} {:error, dgettext("errors", "Error while saving report")}
end end
end end

View file

@ -19,7 +19,7 @@ defmodule Mobilizon.GraphQL.Schema.ReportType do
field(:uri, :string, description: "The URI of the report", meta: [private: true]) field(:uri, :string, description: "The URI of the report", meta: [private: true])
field(:reported, :actor, description: "The actor that is being reported") field(:reported, :actor, description: "The actor that is being reported")
field(:reporter, :actor, description: "The actor that created the report") field(:reporter, :actor, description: "The actor that created the report")
field(:event, :event, description: "The event that is being reported") field(:events, list_of(:event), description: "The event that is being reported")
field(:comments, list_of(:comment), description: "The comments that are reported") field(:comments, list_of(:comment), description: "The comments that are reported")
field(:notes, list_of(:report_note), field(:notes, list_of(:report_note),
@ -100,11 +100,15 @@ defmodule Mobilizon.GraphQL.Schema.ReportType do
field :create_report, type: :report do field :create_report, type: :report do
arg(:content, :string, description: "The message sent with the report") arg(:content, :string, description: "The message sent with the report")
arg(:reported_id, non_null(:id), description: "The actor's ID that is being reported") arg(:reported_id, non_null(:id), description: "The actor's ID that is being reported")
arg(:event_id, :id, default_value: nil, description: "The event ID that is being reported")
arg(:events_ids, list_of(:id),
default_value: [],
description: "The list of event IDs that are being reported"
)
arg(:comments_ids, list_of(:id), arg(:comments_ids, list_of(:id),
default_value: [], default_value: [],
description: "The comment ID that is being reported" description: "The comment IDs that are being reported"
) )
arg(:forward, :boolean, arg(:forward, :boolean,

View file

@ -22,13 +22,13 @@ defmodule Mobilizon.Reports.Report do
reported: Actor.t(), reported: Actor.t(),
reporter: Actor.t(), reporter: Actor.t(),
manager: Actor.t(), manager: Actor.t(),
event: Event.t(), events: [Event.t()],
comments: [Comment.t()], comments: [Comment.t()],
notes: [Note.t()] notes: [Note.t()]
} }
@required_attrs [:url, :reported_id, :reporter_id] @required_attrs [:url, :reported_id, :reporter_id]
@optional_attrs [:content, :status, :manager_id, :event_id, :local] @optional_attrs [:content, :status, :manager_id, :local]
@attrs @required_attrs ++ @optional_attrs @attrs @required_attrs ++ @optional_attrs
@timestamps_opts [type: :utc_datetime] @timestamps_opts [type: :utc_datetime]
@ -46,8 +46,8 @@ defmodule Mobilizon.Reports.Report do
belongs_to(:reporter, Actor) belongs_to(:reporter, Actor)
# The actor who last acted on this report # The actor who last acted on this report
belongs_to(:manager, Actor) belongs_to(:manager, Actor)
# The eventual Event inside the report # The eventual Events inside the report
belongs_to(:event, Event) many_to_many(:events, Event, join_through: "reports_events", on_replace: :delete)
# The eventual Comments inside the report # The eventual Comments inside the report
many_to_many(:comments, Comment, join_through: "reports_comments", on_replace: :delete) many_to_many(:comments, Comment, join_through: "reports_comments", on_replace: :delete)
# The notes associated to the report # The notes associated to the report
@ -62,6 +62,7 @@ defmodule Mobilizon.Reports.Report do
report report
|> cast(attrs, @attrs) |> cast(attrs, @attrs)
|> maybe_generate_url() |> maybe_generate_url()
|> maybe_put_events(attrs)
|> maybe_put_comments(attrs) |> maybe_put_comments(attrs)
|> validate_required(@required_attrs) |> validate_required(@required_attrs)
end end
@ -72,6 +73,12 @@ defmodule Mobilizon.Reports.Report do
defp maybe_put_comments(%Ecto.Changeset{} = changeset, _), do: changeset defp maybe_put_comments(%Ecto.Changeset{} = changeset, _), do: changeset
defp maybe_put_events(%Ecto.Changeset{} = changeset, %{events: events}) do
put_assoc(changeset, :events, events)
end
defp maybe_put_events(%Ecto.Changeset{} = changeset, _), do: changeset
@spec maybe_generate_url(Ecto.Changeset.t()) :: Ecto.Changeset.t() @spec maybe_generate_url(Ecto.Changeset.t()) :: Ecto.Changeset.t()
defp maybe_generate_url(%Ecto.Changeset{} = changeset) do defp maybe_generate_url(%Ecto.Changeset{} = changeset) do
with res when res in [:error, {:data, nil}] <- fetch_field(changeset, :url), with res when res in [:error, {:data, nil}] <- fetch_field(changeset, :url),

View file

@ -21,7 +21,7 @@ defmodule Mobilizon.Reports do
def get_report(id) do def get_report(id) do
Report Report
|> Repo.get(id) |> Repo.get(id)
|> Repo.preload([:reported, :reporter, :manager, :event, :comments, :notes]) |> Repo.preload([:reported, :reporter, :manager, :events, :comments, :notes])
end end
@doc """ @doc """
@ -33,7 +33,7 @@ defmodule Mobilizon.Reports do
%Report{} %Report{}
|> Report.changeset(attrs) |> Report.changeset(attrs)
|> Repo.insert() do |> Repo.insert() do
{:ok, Repo.preload(report, [:event, :reported, :reporter, :comments])} {:ok, Repo.preload(report, [:events, :reported, :reporter, :comments])}
end end
end end
@ -102,7 +102,7 @@ defmodule Mobilizon.Reports do
@spec list_reports_query(atom()) :: Ecto.Query.t() @spec list_reports_query(atom()) :: Ecto.Query.t()
defp list_reports_query(status) do defp list_reports_query(status) do
Report Report
|> preload([:reported, :reporter, :manager, :event, :comments, :notes]) |> preload([:reported, :reporter, :manager, :events, :comments, :notes])
|> where([r], r.status == ^status) |> where([r], r.status == ^status)
end end

View file

@ -139,7 +139,7 @@ defmodule Mobilizon.Service.AntiSpam.Akismet do
end end
end end
defp report_to_akismet_comment(%Report{event: %Event{id: event_id}}) do defp report_to_akismet_comment(%Report{events: [%Event{id: event_id} | _]}) do
with %Event{description: body, organizer_actor: %Actor{} = actor} <- with %Event{description: body, organizer_actor: %Actor{} = actor} <-
Events.get_event_with_preload!(event_id), Events.get_event_with_preload!(event_id),
{email, preferred_username, ip} <- actor_details(actor) do {email, preferred_username, ip} <- actor_details(actor) do

View file

@ -104,7 +104,7 @@
</td> </td>
</tr> </tr>
<% end %> <% end %>
<%= if Map.has_key?(@report, :event) and @report.event do %> <%= if Map.has_key?(@report, :events) and length(@report.events) > 0 do %>
<tr> <tr>
<td <td
bgcolor="#ffffff" bgcolor="#ffffff"
@ -112,16 +112,19 @@
style="padding: 20px 30px 0px 30px; color: #474467; font-family: 'Roboto', Helvetica, Arial, sans-serif; font-size: 18px; font-weight: 400; line-height: 25px;" style="padding: 20px 30px 0px 30px; color: #474467; font-family: 'Roboto', Helvetica, Arial, sans-serif; font-size: 18px; font-weight: 400; line-height: 25px;"
> >
<p style="margin: 0;"> <p style="margin: 0;">
<h3><%= gettext("Event") %></h3> <h3><%= gettext("Flagged events") %></h3>
<a <%= for event <- @report.events do %>
href={"#{"#{Mobilizon.Web.Endpoint.url()}/events/#{@report.event.uuid}"}"} <a
target="_blank" href={"#{"#{Mobilizon.Web.Endpoint.url()}/events/#{event.uuid}"}"}
> target="_blank"
<%= gettext("%{title} by %{creator}", >
title: @report.event.title, <%= gettext("%{title} by %{creator}",
creator: Mobilizon.Actors.Actor.preferred_username_and_domain(@report.reported) title: event.title,
) %> creator:
</a> Mobilizon.Actors.Actor.preferred_username_and_domain(@report.reported)
) %>
</a>
<% end %>
</p> </p>
<table <table
cellspacing="0" cellspacing="0"

View file

@ -7,9 +7,11 @@
<%= gettext "Profile %{profile} was reported", profile: Mobilizon.Actors.Actor.display_name_and_username(@report.reported) %> <%= gettext "Profile %{profile} was reported", profile: Mobilizon.Actors.Actor.display_name_and_username(@report.reported) %>
<% end %> <% end %>
<% end %> <% end %>
<%= if Map.has_key?(@report, :event) and @report.event do %> <%= if Map.has_key?(@report, :event) && length(@report.events) > 0 do %>
<%= gettext "Event" %> <%= gettext "Events" %>
<%= @report.event.title %> <%= for event <- @report.events do %>
<%= event.title %>
<% end %>
<% end %> <% end %>
<%= if Map.has_key?(@report, :comments) && length(@report.comments) > 0 do %> <%= if Map.has_key?(@report, :comments) && length(@report.comments) > 0 do %>
<%= gettext "Comments" %> <%= gettext "Comments" %>

View file

@ -0,0 +1,14 @@
defmodule Mobilizon.Storage.Repo.Migrations.AllowMultipleEventsToBeReported do
use Ecto.Migration
def up do
create table(:reports_events, primary_key: false) do
add(:report_id, references(:reports, on_delete: :delete_all), null: false)
add(:event_id, references(:events, on_delete: :delete_all), null: false)
end
end
def down do
drop table(:reports_events)
end
end

View file

@ -0,0 +1,31 @@
defmodule Mobilizon.Storage.Repo.Migrations.BackfillReportEventsWithOldEvents do
use Ecto.Migration
def up do
process_reports_with_events()
end
def down do
IO.puts("Doing nothing, migration can't be reverted")
end
defp process_reports_with_events do
%Postgrex.Result{rows: rows} =
Ecto.Adapters.SQL.query!(
Mobilizon.Storage.Repo,
"SELECT id, event_id FROM reports WHERE event_id IS NOT NULL"
)
Enum.map(rows, &migrate_event_row/1)
end
defp migrate_event_row([report_id, event_id]) when not is_nil(event_id) do
Ecto.Adapters.SQL.query!(
Mobilizon.Storage.Repo,
"INSERT INTO reports_events VALUES ($1, $2)",
[report_id, event_id]
)
end
defp migrate_event_row(_), do: :ok
end

View file

@ -0,0 +1,15 @@
defmodule Mobilizon.Storage.Repo.Migrations.RemoveObsoleteEventIdOnReports do
use Ecto.Migration
def up do
alter table(:reports) do
remove_if_exists :event_id, :integer
end
end
def down do
alter table(:reports) do
add(:event_id, references(:events, on_delete: :delete_all), null: true)
end
end
end

View file

@ -30,7 +30,7 @@ defmodule Mobilizon.GraphQL.API.ReportTest do
reporter_id: reporter_id, reporter_id: reporter_id,
reported_id: reported_id, reported_id: reported_id,
content: comment, content: comment,
event_id: event_id, events_ids: [event_id],
comments_ids: [], comments_ids: [],
forward: false forward: false
}) })
@ -64,7 +64,7 @@ defmodule Mobilizon.GraphQL.API.ReportTest do
reporter_id: reporter_id, reporter_id: reporter_id,
reported_id: reported_id, reported_id: reported_id,
content: comment, content: comment,
event_id: nil, events_ids: [],
comments_ids: [comment_1_id, comment_2_id] comments_ids: [comment_1_id, comment_2_id]
}) })
@ -100,7 +100,7 @@ defmodule Mobilizon.GraphQL.API.ReportTest do
reporter_id: reporter_id, reporter_id: reporter_id,
reported_id: reported_id, reported_id: reported_id,
content: comment, content: comment,
event_id: nil, events_ids: [],
comments_ids: [comment_1_id, comment_2_id], comments_ids: [comment_1_id, comment_2_id],
forward: true forward: true
}) })
@ -131,7 +131,7 @@ defmodule Mobilizon.GraphQL.API.ReportTest do
reporter_id: reporter_id, reporter_id: reporter_id,
reported_id: reported_id, reported_id: reported_id,
content: "This is not a nice thing", content: "This is not a nice thing",
event_id: nil, events_ids: [],
comments_ids: [comment_1_id], comments_ids: [comment_1_id],
forward: true forward: true
}) })
@ -157,7 +157,7 @@ defmodule Mobilizon.GraphQL.API.ReportTest do
reporter_id: reporter_id, reporter_id: reporter_id,
reported_id: reported_id, reported_id: reported_id,
content: "This is not a nice thing", content: "This is not a nice thing",
event_id: nil, events_ids: [],
comments_ids: [comment_1_id], comments_ids: [comment_1_id],
forward: true forward: true
}) })

View file

@ -15,17 +15,17 @@ defmodule Mobilizon.GraphQL.Resolvers.ReportTest do
describe "Resolver: Report a content" do describe "Resolver: Report a content" do
@create_report_mutation """ @create_report_mutation """
mutation CreateReport($reportedId: ID!, $eventId: ID, $content: String) { mutation CreateReport($reportedId: ID!, $eventsIds: [ID], $content: String) {
createReport( createReport(
reportedId: $reportedId, reportedId: $reportedId,
eventId: $eventId, eventsIds: $eventsIds,
content: $content content: $content
) { ) {
content, content,
reporter { reporter {
id id
}, },
event { events {
id id
}, },
status status
@ -55,7 +55,7 @@ defmodule Mobilizon.GraphQL.Resolvers.ReportTest do
query: @create_report_mutation, query: @create_report_mutation,
variables: %{ variables: %{
reportedId: reported.id, reportedId: reported.id,
eventId: event.id, eventsIds: [event.id],
content: "This is an issue" content: "This is an issue"
} }
) )
@ -63,7 +63,7 @@ defmodule Mobilizon.GraphQL.Resolvers.ReportTest do
assert res["errors"] == nil assert res["errors"] == nil
assert res["data"]["createReport"]["content"] == "This is an issue" assert res["data"]["createReport"]["content"] == "This is an issue"
assert res["data"]["createReport"]["status"] == "OPEN" assert res["data"]["createReport"]["status"] == "OPEN"
assert res["data"]["createReport"]["event"]["id"] == to_string(event.id) assert res["data"]["createReport"]["events"] |> hd |> Map.get("id") == to_string(event.id)
assert res["data"]["createReport"]["reporter"]["id"] == assert res["data"]["createReport"]["reporter"]["id"] ==
to_string(reporter.id) to_string(reporter.id)
@ -122,7 +122,7 @@ defmodule Mobilizon.GraphQL.Resolvers.ReportTest do
reporter { reporter {
id id
}, },
event { events {
id id
}, },
status status
@ -280,7 +280,7 @@ defmodule Mobilizon.GraphQL.Resolvers.ReportTest do
reporter { reporter {
preferredUsername preferredUsername
}, },
event { events {
title title
}, },
comments { comments {
@ -312,7 +312,9 @@ defmodule Mobilizon.GraphQL.Resolvers.ReportTest do
reporter.preferred_username reporter.preferred_username
assert json_response(res, 200)["data"]["report"]["content"] == report.content assert json_response(res, 200)["data"]["report"]["content"] == report.content
assert json_response(res, 200)["data"]["report"]["event"]["title"] == report.event.title
assert json_response(res, 200)["data"]["report"]["events"] |> hd |> Map.get("title") ==
report.events |> hd |> Map.get(:title)
assert json_response(res, 200)["data"]["report"]["comments"] |> hd |> Map.get("text") == assert json_response(res, 200)["data"]["report"]["comments"] |> hd |> Map.get("text") ==
report.comments |> hd |> Map.get(:text) report.comments |> hd |> Map.get(:text)

View file

@ -322,7 +322,7 @@ defmodule Mobilizon.Factory do
url: "http://mobilizon.test/report/deae1020-54b8-47df-9eea-d8c0e943e57f/activity", url: "http://mobilizon.test/report/deae1020-54b8-47df-9eea-d8c0e943e57f/activity",
reported: build(:actor), reported: build(:actor),
reporter: build(:actor), reporter: build(:actor),
event: build(:event), events: build_list(1, :event),
comments: build_list(1, :comment) comments: build_list(1, :comment)
} }
end end