Chats: Remove `unread` from the db, calculate from unseen messages.
This commit is contained in:
parent
8edead7c1d
commit
7f5c5b11a5
|
@ -19,14 +19,13 @@ defmodule Pleroma.Chat do
|
||||||
schema "chats" do
|
schema "chats" do
|
||||||
belongs_to(:user, User, type: FlakeId.Ecto.CompatType)
|
belongs_to(:user, User, type: FlakeId.Ecto.CompatType)
|
||||||
field(:recipient, :string)
|
field(:recipient, :string)
|
||||||
field(:unread, :integer, default: 0, read_after_writes: true)
|
|
||||||
|
|
||||||
timestamps()
|
timestamps()
|
||||||
end
|
end
|
||||||
|
|
||||||
def creation_cng(struct, params) do
|
def creation_cng(struct, params) do
|
||||||
struct
|
struct
|
||||||
|> cast(params, [:user_id, :recipient, :unread])
|
|> cast(params, [:user_id, :recipient])
|
||||||
|> validate_change(:recipient, fn
|
|> validate_change(:recipient, fn
|
||||||
:recipient, recipient ->
|
:recipient, recipient ->
|
||||||
case User.get_cached_by_ap_id(recipient) do
|
case User.get_cached_by_ap_id(recipient) do
|
||||||
|
@ -61,16 +60,10 @@ def get_or_create(user_id, recipient) do
|
||||||
|
|
||||||
def bump_or_create(user_id, recipient) do
|
def bump_or_create(user_id, recipient) do
|
||||||
%__MODULE__{}
|
%__MODULE__{}
|
||||||
|> creation_cng(%{user_id: user_id, recipient: recipient, unread: 1})
|
|> creation_cng(%{user_id: user_id, recipient: recipient})
|
||||||
|> Repo.insert(
|
|> Repo.insert(
|
||||||
on_conflict: [set: [updated_at: NaiveDateTime.utc_now()], inc: [unread: 1]],
|
on_conflict: [set: [updated_at: NaiveDateTime.utc_now()]],
|
||||||
conflict_target: [:user_id, :recipient]
|
conflict_target: [:user_id, :recipient]
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def mark_as_read(chat) do
|
|
||||||
chat
|
|
||||||
|> change(%{unread: 0})
|
|
||||||
|> Repo.update()
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -84,4 +84,20 @@ def create(chat, object, seen) do
|
||||||
|> changeset(params)
|
|> changeset(params)
|
||||||
|> Repo.insert()
|
|> Repo.insert()
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def unread_count_for_chat(chat) do
|
||||||
|
chat
|
||||||
|
|> for_chat_query()
|
||||||
|
|> where([cmr], cmr.seen == false)
|
||||||
|
|> Repo.aggregate(:count)
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_all_seen_for_chat(chat) do
|
||||||
|
chat
|
||||||
|
|> for_chat_query()
|
||||||
|
|> exclude(:order_by)
|
||||||
|
|> exclude(:preload)
|
||||||
|
|> where([cmr], cmr.seen == false)
|
||||||
|
|> Repo.update_all(set: [seen: true])
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -139,13 +139,8 @@ def handle_object_creation(%{"type" => "ChatMessage"} = object, meta) do
|
||||||
[[actor, recipient], [recipient, actor]]
|
[[actor, recipient], [recipient, actor]]
|
||||||
|> Enum.each(fn [user, other_user] ->
|
|> Enum.each(fn [user, other_user] ->
|
||||||
if user.local do
|
if user.local do
|
||||||
if user.ap_id == actor.ap_id do
|
|
||||||
{:ok, chat} = Chat.get_or_create(user.id, other_user.ap_id)
|
|
||||||
ChatMessageReference.create(chat, object, true)
|
|
||||||
else
|
|
||||||
{:ok, chat} = Chat.bump_or_create(user.id, other_user.ap_id)
|
{:ok, chat} = Chat.bump_or_create(user.id, other_user.ap_id)
|
||||||
ChatMessageReference.create(chat, object, false)
|
ChatMessageReference.create(chat, object, user.ap_id == actor.ap_id)
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end)
|
end)
|
||||||
|
|
||||||
|
|
|
@ -90,7 +90,7 @@ def post_chat_message(
|
||||||
|
|
||||||
def mark_as_read(%{assigns: %{user: %{id: user_id}}} = conn, %{id: id}) do
|
def mark_as_read(%{assigns: %{user: %{id: user_id}}} = conn, %{id: id}) do
|
||||||
with %Chat{} = chat <- Repo.get_by(Chat, id: id, user_id: user_id),
|
with %Chat{} = chat <- Repo.get_by(Chat, id: id, user_id: user_id),
|
||||||
{:ok, chat} <- Chat.mark_as_read(chat) do
|
{_n, _} <- ChatMessageReference.set_all_seen_for_chat(chat) do
|
||||||
conn
|
conn
|
||||||
|> put_view(ChatView)
|
|> put_view(ChatView)
|
||||||
|> render("show.json", chat: chat)
|
|> render("show.json", chat: chat)
|
||||||
|
|
|
@ -20,7 +20,7 @@ def render("show.json", %{chat: %Chat{} = chat} = opts) do
|
||||||
%{
|
%{
|
||||||
id: chat.id |> to_string(),
|
id: chat.id |> to_string(),
|
||||||
account: AccountView.render("show.json", Map.put(opts, :user, recipient)),
|
account: AccountView.render("show.json", Map.put(opts, :user, recipient)),
|
||||||
unread: chat.unread,
|
unread: ChatMessageReference.unread_count_for_chat(chat),
|
||||||
last_message:
|
last_message:
|
||||||
last_message &&
|
last_message &&
|
||||||
ChatMessageReferenceView.render("show.json", chat_message_reference: last_message),
|
ChatMessageReferenceView.render("show.json", chat_message_reference: last_message),
|
||||||
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
defmodule Pleroma.Repo.Migrations.RemoveUnreadFromChats do
|
||||||
|
use Ecto.Migration
|
||||||
|
|
||||||
|
def change do
|
||||||
|
alter table(:chats) do
|
||||||
|
remove(:unread, :integer, default: 0)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -6,7 +6,6 @@ defmodule Pleroma.ChatTest do
|
||||||
use Pleroma.DataCase, async: true
|
use Pleroma.DataCase, async: true
|
||||||
|
|
||||||
alias Pleroma.Chat
|
alias Pleroma.Chat
|
||||||
alias Pleroma.Web.CommonAPI
|
|
||||||
|
|
||||||
import Pleroma.Factory
|
import Pleroma.Factory
|
||||||
|
|
||||||
|
@ -35,7 +34,6 @@ test "it returns and bumps a chat for a user and recipient if it already exists"
|
||||||
{:ok, chat_two} = Chat.bump_or_create(user.id, other_user.ap_id)
|
{:ok, chat_two} = Chat.bump_or_create(user.id, other_user.ap_id)
|
||||||
|
|
||||||
assert chat.id == chat_two.id
|
assert chat.id == chat_two.id
|
||||||
assert chat_two.unread == 2
|
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it returns a chat for a user and recipient if it already exists" do
|
test "it returns a chat for a user and recipient if it already exists" do
|
||||||
|
@ -48,15 +46,13 @@ test "it returns a chat for a user and recipient if it already exists" do
|
||||||
assert chat.id == chat_two.id
|
assert chat.id == chat_two.id
|
||||||
end
|
end
|
||||||
|
|
||||||
test "a returning chat will have an updated `update_at` field and an incremented unread count" do
|
test "a returning chat will have an updated `update_at` field" do
|
||||||
user = insert(:user)
|
user = insert(:user)
|
||||||
other_user = insert(:user)
|
other_user = insert(:user)
|
||||||
|
|
||||||
{:ok, chat} = Chat.bump_or_create(user.id, other_user.ap_id)
|
{:ok, chat} = Chat.bump_or_create(user.id, other_user.ap_id)
|
||||||
assert chat.unread == 1
|
|
||||||
:timer.sleep(1500)
|
:timer.sleep(1500)
|
||||||
{:ok, chat_two} = Chat.bump_or_create(user.id, other_user.ap_id)
|
{:ok, chat_two} = Chat.bump_or_create(user.id, other_user.ap_id)
|
||||||
assert chat_two.unread == 2
|
|
||||||
|
|
||||||
assert chat.id == chat_two.id
|
assert chat.id == chat_two.id
|
||||||
assert chat.updated_at != chat_two.updated_at
|
assert chat.updated_at != chat_two.updated_at
|
||||||
|
|
|
@ -346,7 +346,6 @@ test "it creates a Chat and ChatMessageReferences for the local users and bumps
|
||||||
SideEffects.handle(create_activity, local: false, object_data: chat_message_data)
|
SideEffects.handle(create_activity, local: false, object_data: chat_message_data)
|
||||||
|
|
||||||
chat = Chat.get(author.id, recipient.ap_id)
|
chat = Chat.get(author.id, recipient.ap_id)
|
||||||
assert chat.unread == 0
|
|
||||||
|
|
||||||
[cm_ref] = ChatMessageReference.for_chat_query(chat) |> Repo.all()
|
[cm_ref] = ChatMessageReference.for_chat_query(chat) |> Repo.all()
|
||||||
|
|
||||||
|
@ -354,7 +353,6 @@ test "it creates a Chat and ChatMessageReferences for the local users and bumps
|
||||||
assert cm_ref.seen == true
|
assert cm_ref.seen == true
|
||||||
|
|
||||||
chat = Chat.get(recipient.id, author.ap_id)
|
chat = Chat.get(recipient.id, author.ap_id)
|
||||||
assert chat.unread == 1
|
|
||||||
|
|
||||||
[cm_ref] = ChatMessageReference.for_chat_query(chat) |> Repo.all()
|
[cm_ref] = ChatMessageReference.for_chat_query(chat) |> Repo.all()
|
||||||
|
|
||||||
|
|
|
@ -19,9 +19,12 @@ defmodule Pleroma.Web.PleromaAPI.ChatControllerTest do
|
||||||
test "it marks all messages in a chat as read", %{conn: conn, user: user} do
|
test "it marks all messages in a chat as read", %{conn: conn, user: user} do
|
||||||
other_user = insert(:user)
|
other_user = insert(:user)
|
||||||
|
|
||||||
{:ok, chat} = Chat.bump_or_create(user.id, other_user.ap_id)
|
{:ok, create} = CommonAPI.post_chat_message(other_user, user, "sup")
|
||||||
|
{:ok, chat} = Chat.get_or_create(user.id, other_user.ap_id)
|
||||||
|
object = Object.normalize(create, false)
|
||||||
|
cm_ref = ChatMessageReference.for_chat_and_object(chat, object)
|
||||||
|
|
||||||
assert chat.unread == 1
|
assert cm_ref.seen == false
|
||||||
|
|
||||||
result =
|
result =
|
||||||
conn
|
conn
|
||||||
|
@ -30,9 +33,9 @@ test "it marks all messages in a chat as read", %{conn: conn, user: user} do
|
||||||
|
|
||||||
assert result["unread"] == 0
|
assert result["unread"] == 0
|
||||||
|
|
||||||
{:ok, chat} = Chat.get_or_create(user.id, other_user.ap_id)
|
cm_ref = ChatMessageReference.for_chat_and_object(chat, object)
|
||||||
|
|
||||||
assert chat.unread == 0
|
assert cm_ref.seen == true
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue