mirror of
https://github.com/philomena-dev/philomena.git
synced 2025-01-20 14:47:58 +01:00
341 lines
7.5 KiB
Elixir
341 lines
7.5 KiB
Elixir
defmodule Philomena.Posts do
|
|
@moduledoc """
|
|
The Posts context.
|
|
"""
|
|
|
|
import Ecto.Query, warn: false
|
|
alias Ecto.Multi
|
|
alias Philomena.Repo
|
|
|
|
alias Philomena.Elasticsearch
|
|
alias Philomena.Topics.Topic
|
|
alias Philomena.Topics
|
|
alias Philomena.UserStatistics
|
|
alias Philomena.Posts.Post
|
|
alias Philomena.Posts.ElasticsearchIndex, as: PostIndex
|
|
alias Philomena.IndexWorker
|
|
alias Philomena.Forums.Forum
|
|
alias Philomena.Notifications
|
|
alias Philomena.NotificationWorker
|
|
alias Philomena.Versions
|
|
alias Philomena.Reports
|
|
alias Philomena.Reports.Report
|
|
alias Philomena.Users.User
|
|
|
|
@doc """
|
|
Gets a single post.
|
|
|
|
Raises `Ecto.NoResultsError` if the Post does not exist.
|
|
|
|
## Examples
|
|
|
|
iex> get_post!(123)
|
|
%Post{}
|
|
|
|
iex> get_post!(456)
|
|
** (Ecto.NoResultsError)
|
|
|
|
"""
|
|
def get_post!(id), do: Repo.get!(Post, id)
|
|
|
|
@doc """
|
|
Creates a post.
|
|
|
|
## Examples
|
|
|
|
iex> create_post(%{field: value})
|
|
{:ok, %Post{}}
|
|
|
|
iex> create_post(%{field: bad_value})
|
|
{:error, %Ecto.Changeset{}}
|
|
|
|
"""
|
|
def create_post(topic, attributes, params \\ %{}) do
|
|
now = DateTime.utc_now()
|
|
|
|
topic_query =
|
|
Topic
|
|
|> where(id: ^topic.id)
|
|
|
|
topic_lock_query =
|
|
topic_query
|
|
|> lock("FOR UPDATE")
|
|
|
|
forum_query =
|
|
Forum
|
|
|> where(id: ^topic.forum_id)
|
|
|
|
Multi.new()
|
|
|> Multi.all(:topic_lock, topic_lock_query)
|
|
|> Multi.run(:post, fn repo, _ ->
|
|
last_position =
|
|
Post
|
|
|> where(topic_id: ^topic.id)
|
|
|> order_by(desc: :topic_position)
|
|
|> select([p], p.topic_position)
|
|
|> limit(1)
|
|
|> repo.one()
|
|
|
|
Ecto.build_assoc(topic, :posts, [topic_position: (last_position || -1) + 1] ++ attributes)
|
|
|> Post.creation_changeset(params, attributes)
|
|
|> repo.insert()
|
|
end)
|
|
|> Multi.run(:update_topic, fn repo, %{post: %{id: post_id}} ->
|
|
{count, nil} =
|
|
repo.update_all(topic_query,
|
|
inc: [post_count: 1],
|
|
set: [last_post_id: post_id, last_replied_to_at: now]
|
|
)
|
|
|
|
{:ok, count}
|
|
end)
|
|
|> Multi.run(:update_forum, fn repo, %{post: %{id: post_id}} ->
|
|
{count, nil} =
|
|
repo.update_all(forum_query, inc: [post_count: 1], set: [last_post_id: post_id])
|
|
|
|
{:ok, count}
|
|
end)
|
|
|> maybe_create_subscription_on_reply(topic, attributes[:user])
|
|
|> Repo.transaction()
|
|
|> case do
|
|
{:ok, %{post: post}} = result ->
|
|
reindex_post(post)
|
|
|
|
result
|
|
|
|
error ->
|
|
error
|
|
end
|
|
end
|
|
|
|
defp maybe_create_subscription_on_reply(multi, topic, %User{watch_on_reply: true} = user) do
|
|
multi
|
|
|> Multi.run(:subscribe, fn _repo, _changes ->
|
|
Topics.create_subscription(topic, user)
|
|
end)
|
|
end
|
|
|
|
defp maybe_create_subscription_on_reply(multi, _topic, _user) do
|
|
multi
|
|
end
|
|
|
|
def notify_post(post) do
|
|
Exq.enqueue(Exq, "notifications", NotificationWorker, ["Posts", post.id])
|
|
end
|
|
|
|
def report_non_approved(%Post{approved: true}), do: false
|
|
|
|
def report_non_approved(post) do
|
|
Reports.create_system_report(
|
|
post.id,
|
|
"Post",
|
|
"Approval",
|
|
"Post contains externally-embedded images and has been flagged for review."
|
|
)
|
|
end
|
|
|
|
def perform_notify(post_id) do
|
|
post = get_post!(post_id)
|
|
|
|
topic =
|
|
post
|
|
|> Repo.preload(:topic)
|
|
|> Map.fetch!(:topic)
|
|
|
|
subscriptions =
|
|
topic
|
|
|> 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 reply in"
|
|
}
|
|
)
|
|
end
|
|
|
|
@doc """
|
|
Updates a post.
|
|
|
|
## Examples
|
|
|
|
iex> update_post(post, %{field: new_value})
|
|
{:ok, %Post{}}
|
|
|
|
iex> update_post(post, %{field: bad_value})
|
|
{:error, %Ecto.Changeset{}}
|
|
|
|
"""
|
|
def update_post(%Post{} = post, editor, attrs) do
|
|
now = DateTime.utc_now() |> DateTime.truncate(:second)
|
|
current_body = post.body
|
|
current_reason = post.edit_reason
|
|
|
|
post_changes = Post.changeset(post, attrs, now)
|
|
|
|
Multi.new()
|
|
|> Multi.update(:post, post_changes)
|
|
|> Multi.run(:version, fn _repo, _changes ->
|
|
Versions.create_version("Post", post.id, editor.id, %{
|
|
"body" => current_body,
|
|
"edit_reason" => current_reason
|
|
})
|
|
end)
|
|
|> Repo.transaction()
|
|
|> case do
|
|
{:ok, %{post: post}} = result ->
|
|
reindex_post(post)
|
|
|
|
result
|
|
|
|
error ->
|
|
error
|
|
end
|
|
end
|
|
|
|
@doc """
|
|
Deletes a Post.
|
|
|
|
## Examples
|
|
|
|
iex> delete_post(post)
|
|
{:ok, %Post{}}
|
|
|
|
iex> delete_post(post)
|
|
{:error, %Ecto.Changeset{}}
|
|
|
|
"""
|
|
def delete_post(%Post{} = post) do
|
|
Repo.delete(post)
|
|
end
|
|
|
|
def hide_post(%Post{} = post, attrs, user) do
|
|
reports =
|
|
Report
|
|
|> where(reportable_type: "Post", reportable_id: ^post.id)
|
|
|> select([r], r.id)
|
|
|> update(set: [open: false, state: "closed", admin_id: ^user.id])
|
|
|
|
topics =
|
|
Topic
|
|
|> where(last_post_id: ^post.id)
|
|
|> update(set: [last_post_id: nil])
|
|
|
|
forums =
|
|
Forum
|
|
|> where(last_post_id: ^post.id)
|
|
|> update(set: [last_post_id: nil])
|
|
|
|
post = Post.hide_changeset(post, attrs, user)
|
|
|
|
Multi.new()
|
|
|> Multi.update(:post, post)
|
|
|> Multi.update_all(:reports, reports, [])
|
|
|> Multi.update_all(:topics, topics, [])
|
|
|> Multi.update_all(:forums, forums, [])
|
|
|> Repo.transaction()
|
|
|> case do
|
|
{:ok, %{post: post, reports: {_count, reports}}} ->
|
|
Reports.reindex_reports(reports)
|
|
reindex_post(post)
|
|
|
|
{:ok, post}
|
|
|
|
error ->
|
|
error
|
|
end
|
|
end
|
|
|
|
def unhide_post(%Post{} = post) do
|
|
post
|
|
|> Post.unhide_changeset()
|
|
|> Repo.update()
|
|
|> reindex_after_update()
|
|
end
|
|
|
|
def destroy_post(%Post{} = post) do
|
|
post
|
|
|> Post.destroy_changeset()
|
|
|> Repo.update()
|
|
|> reindex_after_update()
|
|
end
|
|
|
|
def approve_post(%Post{} = post, user) do
|
|
reports =
|
|
Report
|
|
|> where(reportable_type: "Post", reportable_id: ^post.id)
|
|
|> select([r], r.id)
|
|
|> update(set: [open: false, state: "closed", admin_id: ^user.id])
|
|
|
|
post = Post.approve_changeset(post)
|
|
|
|
Multi.new()
|
|
|> Multi.update(:post, post)
|
|
|> Multi.update_all(:reports, reports, [])
|
|
|> Repo.transaction()
|
|
|> case do
|
|
{:ok, %{post: post, reports: {_count, reports}}} ->
|
|
notify_post(post)
|
|
UserStatistics.inc_stat(post.user, :forum_posts)
|
|
Reports.reindex_reports(reports)
|
|
reindex_post(post)
|
|
|
|
{:ok, post}
|
|
|
|
error ->
|
|
error
|
|
end
|
|
end
|
|
|
|
@doc """
|
|
Returns an `%Ecto.Changeset{}` for tracking post changes.
|
|
|
|
## Examples
|
|
|
|
iex> change_post(post)
|
|
%Ecto.Changeset{source: %Post{}}
|
|
|
|
"""
|
|
def change_post(%Post{} = post) do
|
|
Post.changeset(post, %{})
|
|
end
|
|
|
|
def user_name_reindex(old_name, new_name) do
|
|
data = PostIndex.user_name_update_by_query(old_name, new_name)
|
|
|
|
Elasticsearch.update_by_query(Post, data.query, data.set_replacements, data.replacements)
|
|
end
|
|
|
|
defp reindex_after_update({:ok, post}) do
|
|
reindex_post(post)
|
|
|
|
{:ok, post}
|
|
end
|
|
|
|
defp reindex_after_update(result) do
|
|
result
|
|
end
|
|
|
|
def reindex_post(%Post{} = post) do
|
|
Exq.enqueue(Exq, "indexing", IndexWorker, ["Posts", "id", [post.id]])
|
|
|
|
post
|
|
end
|
|
|
|
def indexing_preloads do
|
|
[:user, topic: :forum]
|
|
end
|
|
|
|
def perform_reindex(column, condition) do
|
|
Post
|
|
|> preload(^indexing_preloads())
|
|
|> where([p], field(p, ^column) in ^condition)
|
|
|> Elasticsearch.reindex(Post)
|
|
end
|
|
end
|