philomena/lib/philomena/topics.ex
2022-03-24 18:44:39 +01:00

291 lines
6.3 KiB
Elixir

defmodule Philomena.Topics do
@moduledoc """
The Topics context.
"""
import Ecto.Query, warn: false
alias Ecto.Multi
alias Philomena.Repo
alias Philomena.Topics.Topic
alias Philomena.Forums.Forum
alias Philomena.Posts
alias Philomena.Notifications
alias Philomena.NotificationWorker
alias Philomena.Users.User
@doc """
Gets a single topic.
Raises `Ecto.NoResultsError` if the Topic does not exist.
## Examples
iex> get_topic!(123)
%Topic{}
iex> get_topic!(456)
** (Ecto.NoResultsError)
"""
def get_topic!(id), do: Repo.get!(Topic, id)
@doc """
Creates a topic.
## Examples
iex> create_topic(%{field: value})
{:ok, %Topic{}}
iex> create_topic(%{field: bad_value})
{:error, %Ecto.Changeset{}}
"""
def create_topic(forum, attribution, attrs \\ %{}) do
now = DateTime.utc_now() |> DateTime.truncate(:second)
topic =
%Topic{}
|> Topic.creation_changeset(attrs, forum, attribution)
Multi.new()
|> Multi.insert(:topic, topic)
|> Multi.run(:update_topic, fn repo, %{topic: topic} ->
{count, nil} =
Topic
|> where(id: ^topic.id)
|> repo.update_all(set: [last_post_id: hd(topic.posts).id, last_replied_to_at: now])
{:ok, count}
end)
|> Multi.run(:update_forum, fn repo, %{topic: topic} ->
{count, nil} =
Forum
|> where(id: ^topic.forum_id)
|> repo.update_all(
inc: [post_count: 1, topic_count: 1],
set: [last_post_id: hd(topic.posts).id]
)
{:ok, count}
end)
|> maybe_create_subscription_on_new_topic(attribution[:user])
|> Repo.transaction()
|> case do
{:ok, %{topic: topic}} = result ->
Posts.reindex_post(hd(topic.posts))
Posts.report_non_approved(hd(topic.posts))
result
error ->
error
end
end
defp maybe_create_subscription_on_new_topic(multi, %User{watch_on_new_topic: true} = user) do
multi
|> Multi.run(:subscribe, fn _repo, %{topic: topic} ->
create_subscription(topic, user)
end)
end
defp maybe_create_subscription_on_new_topic(multi, _user) do
multi
end
def notify_topic(topic, post) do
Exq.enqueue(Exq, "notifications", NotificationWorker, ["Topics", [topic.id, post.id]])
end
def perform_notify([topic_id, post_id]) do
topic = get_topic!(topic_id)
post = Posts.get_post!(post_id)
forum =
topic
|> Repo.preload(:forum)
|> Map.fetch!(:forum)
subscriptions =
forum
|> Repo.preload(:subscriptions)
|> Map.fetch!(:subscriptions)
Notifications.notify(
post,
subscriptions,
%{
actor_id: topic.id,
actor_type: "Topic",
actor_child_id: post.id,
actor_child_type: "Post",
action: "posted a new topic in #{forum.name}"
}
)
end
@doc """
Updates a topic.
## Examples
iex> update_topic(topic, %{field: new_value})
{:ok, %Topic{}}
iex> update_topic(topic, %{field: bad_value})
{:error, %Ecto.Changeset{}}
"""
def update_topic(%Topic{} = topic, attrs) do
topic
|> Topic.changeset(attrs)
|> Repo.update()
end
@doc """
Deletes a Topic.
## Examples
iex> delete_topic(topic)
{:ok, %Topic{}}
iex> delete_topic(topic)
{:error, %Ecto.Changeset{}}
"""
def delete_topic(%Topic{} = topic) do
Repo.delete(topic)
end
@doc """
Returns an `%Ecto.Changeset{}` for tracking topic changes.
## Examples
iex> change_topic(topic)
%Ecto.Changeset{source: %Topic{}}
"""
def change_topic(%Topic{} = topic) do
Topic.changeset(topic, %{})
end
alias Philomena.Topics.Subscription
def subscribed?(_topic, nil), do: false
def subscribed?(topic, user) do
Subscription
|> where(topic_id: ^topic.id, user_id: ^user.id)
|> Repo.exists?()
end
@doc """
Creates a subscription.
## Examples
iex> create_subscription(%{field: value})
{:ok, %Subscription{}}
iex> create_subscription(%{field: bad_value})
{:error, %Ecto.Changeset{}}
"""
def create_subscription(_topic, nil), do: {:ok, nil}
def create_subscription(topic, user) do
%Subscription{topic_id: topic.id, user_id: user.id}
|> Subscription.changeset(%{})
|> Repo.insert(on_conflict: :nothing)
end
@doc """
Deletes a Subscription.
## Examples
iex> delete_subscription(subscription)
{:ok, %Subscription{}}
iex> delete_subscription(subscription)
{:error, %Ecto.Changeset{}}
"""
def delete_subscription(topic, user) do
clear_notification(topic, user)
%Subscription{topic_id: topic.id, user_id: user.id}
|> Repo.delete()
end
def stick_topic(topic) do
Topic.stick_changeset(topic)
|> Repo.update()
end
def unstick_topic(topic) do
Topic.unstick_changeset(topic)
|> Repo.update()
end
def lock_topic(%Topic{} = topic, attrs, user) do
Topic.lock_changeset(topic, attrs, user)
|> Repo.update()
end
def unlock_topic(%Topic{} = topic) do
Topic.unlock_changeset(topic)
|> Repo.update()
end
def move_topic(topic, new_forum_id) do
old_forum_id = topic.forum_id
topic_changes = Topic.move_changeset(topic, new_forum_id)
Multi.new()
|> Multi.update(:topic, topic_changes)
|> Multi.run(:update_old_forum, fn repo, %{topic: topic} ->
{count, nil} =
Forum
|> where(id: ^old_forum_id)
|> repo.update_all(inc: [post_count: -topic.post_count, topic_count: -1])
{:ok, count}
end)
|> Multi.run(:update_new_forum, fn repo, %{topic: topic} ->
{count, nil} =
Forum
|> where(id: ^topic.forum_id)
|> repo.update_all(inc: [post_count: topic.post_count, topic_count: 1])
{:ok, count}
end)
|> Repo.transaction()
end
def hide_topic(topic, deletion_reason, user) do
Topic.hide_changeset(topic, deletion_reason, user)
|> Repo.update()
end
def unhide_topic(topic) do
Topic.unhide_changeset(topic)
|> Repo.update()
end
def update_topic_title(topic, attrs) do
topic
|> Topic.title_changeset(attrs)
|> Repo.update()
end
def clear_notification(_topic, nil), do: nil
def clear_notification(topic, user) do
Notifications.delete_unread_notification("Topic", topic.id, user)
end
end