discourse/lib/post_merger.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

68 lines
1.6 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
class PostMerger
class CannotMergeError < StandardError; end
def initialize(user, posts)
@user = user
@posts = posts
end
def merge
return unless ensure_at_least_two_posts
ensure_same_topic!
ensure_same_user!
guardian = Guardian.new(@user)
ensure_staff_user!(guardian)
posts = @posts.sort_by do |post|
guardian.ensure_can_delete!(post)
post.post_number
end
post_content = posts.map(&:raw)
post = posts.pop
merged_post_raw = post_content.join("\n\n")
changes = {
raw: merged_post_raw,
edit_reason: I18n.t("merge_posts.edit_reason", count: posts.length, username: @user.username)
}
ensure_max_post_length!(merged_post_raw)
PostRevisor.new(post, post.topic).revise!(@user, changes) do
posts.each { |p| PostDestroyer.new(@user, p).destroy }
end
end
private
def ensure_at_least_two_posts
@posts.count >= 2
end
def ensure_same_topic!
unless @posts.map(&:topic_id).uniq.length == 1
raise CannotMergeError.new(I18n.t("merge_posts.errors.different_topics"))
end
end
def ensure_same_user!
unless @posts.map(&:user_id).uniq.length == 1
raise CannotMergeError.new(I18n.t("merge_posts.errors.different_users"))
end
end
def ensure_staff_user!(guardian)
raise Discourse::InvalidAccess unless guardian.is_staff?
end
def ensure_max_post_length!(raw)
value = StrippedLengthValidator.get_sanitized_value(raw)
if value.length > SiteSetting.max_post_length
raise CannotMergeError.new(I18n.t("merge_posts.errors.max_post_length"))
end
end
end