2019-04-10 00:34:53 -07:00
|
|
|
# Pleroma: A lightweight social networking server
|
2020-03-03 14:44:49 -08:00
|
|
|
# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/>
|
2019-04-10 00:34:53 -07:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
defmodule Pleroma.Conversation do
|
|
|
|
alias Pleroma.Conversation.Participation
|
2019-08-02 02:55:41 -07:00
|
|
|
alias Pleroma.Conversation.Participation.RecipientShip
|
2019-04-10 09:17:22 -07:00
|
|
|
alias Pleroma.Repo
|
2019-04-10 07:33:45 -07:00
|
|
|
alias Pleroma.User
|
2019-04-10 00:34:53 -07:00
|
|
|
use Ecto.Schema
|
|
|
|
import Ecto.Changeset
|
|
|
|
|
|
|
|
schema "conversations" do
|
2019-04-10 07:33:45 -07:00
|
|
|
# This is the context ap id.
|
2019-04-10 00:34:53 -07:00
|
|
|
field(:ap_id, :string)
|
|
|
|
has_many(:participations, Participation)
|
2019-04-15 13:28:42 -07:00
|
|
|
has_many(:users, through: [:participations, :user])
|
2019-04-10 00:34:53 -07:00
|
|
|
|
|
|
|
timestamps()
|
|
|
|
end
|
|
|
|
|
|
|
|
def creation_cng(struct, params) do
|
|
|
|
struct
|
|
|
|
|> cast(params, [:ap_id])
|
|
|
|
|> validate_required([:ap_id])
|
|
|
|
|> unique_constraint(:ap_id)
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_for_ap_id(ap_id) do
|
|
|
|
%__MODULE__{}
|
|
|
|
|> creation_cng(%{ap_id: ap_id})
|
2019-04-10 07:33:45 -07:00
|
|
|
|> Repo.insert(
|
|
|
|
on_conflict: [set: [updated_at: NaiveDateTime.utc_now()]],
|
|
|
|
returning: true,
|
|
|
|
conflict_target: :ap_id
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_for_ap_id(ap_id) do
|
|
|
|
Repo.get_by(__MODULE__, ap_id: ap_id)
|
|
|
|
end
|
|
|
|
|
2019-08-06 06:06:19 -07:00
|
|
|
def maybe_create_recipientships(participation, activity) do
|
2019-08-02 02:55:41 -07:00
|
|
|
participation = Repo.preload(participation, :recipients)
|
|
|
|
|
2020-08-31 23:37:08 -07:00
|
|
|
if Enum.empty?(participation.recipients) do
|
2019-08-02 02:55:41 -07:00
|
|
|
recipients = User.get_all_by_ap_id(activity.recipients)
|
|
|
|
RecipientShip.create(recipients, participation)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-04-10 07:33:45 -07:00
|
|
|
@doc """
|
|
|
|
This will
|
|
|
|
1. Create a conversation if there isn't one already
|
|
|
|
2. Create a participation for all the people involved who don't have one already
|
|
|
|
3. Bump all relevant participations to 'unread'
|
|
|
|
"""
|
2019-05-09 07:39:28 -07:00
|
|
|
def create_or_bump_for(activity, opts \\ []) do
|
2019-04-10 07:33:45 -07:00
|
|
|
with true <- Pleroma.Web.ActivityPub.Visibility.is_direct?(activity),
|
|
|
|
"Create" <- activity.data["type"],
|
2019-05-11 17:01:42 -07:00
|
|
|
object <- Pleroma.Object.normalize(activity),
|
2019-05-21 07:33:54 -07:00
|
|
|
true <- object.data["type"] in ["Note", "Question"],
|
2019-04-28 13:44:04 -07:00
|
|
|
ap_id when is_binary(ap_id) and byte_size(ap_id) > 0 <- object.data["context"] do
|
2019-04-10 07:33:45 -07:00
|
|
|
{:ok, conversation} = create_for_ap_id(ap_id)
|
|
|
|
|
2020-05-25 06:06:35 -07:00
|
|
|
users = User.get_users_from_set(activity.recipients, local_only: false)
|
2019-04-10 07:33:45 -07:00
|
|
|
|
|
|
|
participations =
|
2019-04-10 08:05:33 -07:00
|
|
|
Enum.map(users, fn user ->
|
2019-10-25 11:29:23 -07:00
|
|
|
invisible_conversation = Enum.any?(users, &User.blocks?(user, &1))
|
|
|
|
|
|
|
|
opts = Keyword.put(opts, :invisible_conversation, invisible_conversation)
|
2019-10-01 14:37:08 -07:00
|
|
|
|
2019-04-10 07:33:45 -07:00
|
|
|
{:ok, participation} =
|
2019-05-09 07:39:28 -07:00
|
|
|
Participation.create_for_user_and_conversation(user, conversation, opts)
|
2019-04-10 07:33:45 -07:00
|
|
|
|
2019-08-06 06:06:19 -07:00
|
|
|
maybe_create_recipientships(participation, activity)
|
2019-04-10 07:33:45 -07:00
|
|
|
participation
|
|
|
|
end)
|
|
|
|
|
2019-05-03 04:39:14 -07:00
|
|
|
{:ok,
|
|
|
|
%{
|
|
|
|
conversation
|
|
|
|
| participations: participations
|
|
|
|
}}
|
|
|
|
else
|
|
|
|
e -> {:error, e}
|
2019-04-10 07:33:45 -07:00
|
|
|
end
|
2019-04-10 00:34:53 -07:00
|
|
|
end
|
2019-05-08 08:37:00 -07:00
|
|
|
|
|
|
|
@doc """
|
|
|
|
This is only meant to be run by a mix task. It creates conversations/participations for all direct messages in the database.
|
|
|
|
"""
|
2019-05-08 09:19:20 -07:00
|
|
|
def bump_for_all_activities do
|
2019-05-08 08:37:00 -07:00
|
|
|
stream =
|
|
|
|
Pleroma.Web.ActivityPub.ActivityPub.fetch_direct_messages_query()
|
|
|
|
|> Repo.stream()
|
|
|
|
|
2019-05-08 09:09:07 -07:00
|
|
|
Repo.transaction(
|
|
|
|
fn ->
|
|
|
|
stream
|
2019-05-09 07:39:28 -07:00
|
|
|
|> Enum.each(fn a -> create_or_bump_for(a, read: true) end)
|
2019-05-08 09:09:07 -07:00
|
|
|
end,
|
|
|
|
timeout: :infinity
|
|
|
|
)
|
2019-05-08 08:37:00 -07:00
|
|
|
end
|
2019-04-10 00:34:53 -07:00
|
|
|
end
|