2013-02-05 13:16:51 -06:00
|
|
|
# Responsible for creating posts and topics
|
|
|
|
#
|
|
|
|
require_dependency 'rate_limiter'
|
2013-06-04 13:13:01 -05:00
|
|
|
require_dependency 'topic_creator'
|
2014-02-17 00:57:37 -06:00
|
|
|
require_dependency 'post_jobs_enqueuer'
|
2013-02-05 13:16:51 -06:00
|
|
|
|
|
|
|
class PostCreator
|
|
|
|
|
2013-03-18 12:55:34 -05:00
|
|
|
attr_reader :errors, :opts
|
2013-02-05 13:16:51 -06:00
|
|
|
|
|
|
|
# Acceptable options:
|
|
|
|
#
|
|
|
|
# raw - raw text of post
|
2013-02-25 10:42:20 -06:00
|
|
|
# image_sizes - We can pass a list of the sizes of images in the post as a shortcut.
|
2013-03-18 12:55:34 -05:00
|
|
|
# invalidate_oneboxes - Whether to force invalidation of oneboxes in this post
|
2013-05-13 13:06:16 -05:00
|
|
|
# acting_user - The user performing the action might be different than the user
|
|
|
|
# who is the post "author." For example when copying posts to a new
|
|
|
|
# topic.
|
2013-05-20 01:44:06 -05:00
|
|
|
# created_at - Post creation time (optional)
|
2013-07-21 20:40:39 -05:00
|
|
|
# auto_track - Automatically track this topic if needed (default true)
|
2014-05-17 20:33:34 -05:00
|
|
|
# custom_fields - Custom fields to be added to the post, Hash (default nil)
|
2013-02-05 13:16:51 -06:00
|
|
|
#
|
|
|
|
# When replying to a topic:
|
|
|
|
# topic_id - topic we're replying to
|
|
|
|
# reply_to_post_number - post number we're replying to
|
|
|
|
#
|
|
|
|
# When creating a topic:
|
|
|
|
# title - New topic title
|
|
|
|
# archetype - Topic archetype
|
|
|
|
# category - Category to assign to topic
|
|
|
|
# target_usernames - comma delimited list of usernames for membership (private message)
|
2013-05-09 02:37:34 -05:00
|
|
|
# target_group_names - comma delimited list of groups for membership (private message)
|
2013-02-05 13:16:51 -06:00
|
|
|
# meta_data - Topic meta data hash
|
2013-06-21 10:36:33 -05:00
|
|
|
# cooking_options - Options for rendering the text
|
|
|
|
#
|
2013-02-05 13:16:51 -06:00
|
|
|
def initialize(user, opts)
|
2013-04-04 23:29:46 -05:00
|
|
|
# TODO: we should reload user in case it is tainted, should take in a user_id as opposed to user
|
|
|
|
# If we don't do this we introduce a rather risky dependency
|
2013-02-25 10:42:20 -06:00
|
|
|
@user = user
|
2013-06-21 10:36:33 -05:00
|
|
|
@opts = opts || {}
|
2013-05-10 15:58:23 -05:00
|
|
|
@spam = false
|
2013-02-05 13:16:51 -06:00
|
|
|
end
|
|
|
|
|
2013-05-10 15:58:23 -05:00
|
|
|
# True if the post was considered spam
|
|
|
|
def spam?
|
|
|
|
@spam
|
|
|
|
end
|
|
|
|
|
2013-02-05 13:16:51 -06:00
|
|
|
def guardian
|
|
|
|
@guardian ||= Guardian.new(@user)
|
|
|
|
end
|
|
|
|
|
|
|
|
def create
|
2013-06-04 13:13:01 -05:00
|
|
|
@topic = nil
|
|
|
|
@post = nil
|
2013-02-05 13:16:51 -06:00
|
|
|
|
|
|
|
Post.transaction do
|
2013-06-04 13:13:01 -05:00
|
|
|
setup_topic
|
|
|
|
setup_post
|
|
|
|
rollback_if_host_spam_detected
|
|
|
|
save_post
|
|
|
|
extract_links
|
|
|
|
store_unique_post_key
|
|
|
|
track_topic
|
2013-07-22 00:06:53 -05:00
|
|
|
update_topic_stats
|
2013-06-04 13:13:01 -05:00
|
|
|
update_user_counts
|
2014-04-03 13:42:26 -05:00
|
|
|
create_embedded_topic
|
2014-03-17 21:12:07 -05:00
|
|
|
|
2013-06-04 13:13:01 -05:00
|
|
|
publish
|
2013-09-05 23:07:23 -05:00
|
|
|
ensure_in_allowed_users if guardian.is_staff?
|
2013-06-04 13:13:01 -05:00
|
|
|
@post.advance_draft_sequence
|
|
|
|
@post.save_reply_relationships
|
2013-02-05 13:16:51 -06:00
|
|
|
end
|
|
|
|
|
2014-03-17 23:22:39 -05:00
|
|
|
if @post
|
2014-07-03 13:43:24 -05:00
|
|
|
PostAlerter.post_created(@post) unless @opts[:import_mode]
|
2014-03-17 23:22:39 -05:00
|
|
|
|
2014-07-03 13:43:24 -05:00
|
|
|
handle_spam unless @opts[:import_mode]
|
2014-03-17 23:22:39 -05:00
|
|
|
track_latest_on_category
|
|
|
|
enqueue_jobs
|
2014-07-22 20:42:24 -05:00
|
|
|
BadgeGranter.queue_badge_grant(Badge::Trigger::PostRevision, post: @post)
|
2014-03-17 23:22:39 -05:00
|
|
|
end
|
2014-02-10 13:29:31 -06:00
|
|
|
|
2013-06-04 13:13:01 -05:00
|
|
|
@post
|
2013-02-05 13:16:51 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.create(user, opts)
|
2013-02-25 10:42:20 -06:00
|
|
|
PostCreator.new(user, opts).create
|
2013-02-05 13:16:51 -06:00
|
|
|
end
|
|
|
|
|
2013-06-09 11:48:44 -05:00
|
|
|
def self.before_create_tasks(post)
|
2014-02-10 13:29:31 -06:00
|
|
|
set_reply_user_id(post)
|
2013-06-09 11:48:44 -05:00
|
|
|
|
2013-12-10 12:47:07 -06:00
|
|
|
post.word_count = post.raw.scan(/\w+/).size
|
2013-06-09 11:48:44 -05:00
|
|
|
post.post_number ||= Topic.next_post_number(post.topic_id, post.reply_to_post_number.present?)
|
2013-06-21 10:36:33 -05:00
|
|
|
|
|
|
|
cooking_options = post.cooking_options || {}
|
|
|
|
cooking_options[:topic_id] = post.topic_id
|
|
|
|
|
|
|
|
post.cooked ||= post.cook(post.raw, cooking_options)
|
2013-06-09 11:48:44 -05:00
|
|
|
post.sort_order = post.post_number
|
|
|
|
DiscourseEvent.trigger(:before_create_post, post)
|
|
|
|
post.last_version_at ||= Time.now
|
|
|
|
end
|
|
|
|
|
2014-02-10 13:29:31 -06:00
|
|
|
def self.set_reply_user_id(post)
|
|
|
|
return unless post.reply_to_post_number.present?
|
|
|
|
|
2014-05-06 08:41:59 -05:00
|
|
|
post.reply_to_user_id ||= Post.select(:user_id).find_by(topic_id: post.topic_id, post_number: post.reply_to_post_number).try(:user_id)
|
2014-02-10 13:29:31 -06:00
|
|
|
end
|
2013-06-09 11:48:44 -05:00
|
|
|
|
2013-05-02 00:15:17 -05:00
|
|
|
protected
|
|
|
|
|
2014-04-03 13:42:26 -05:00
|
|
|
# You can supply an `embed_url` for a post to set up the embedded relationship.
|
|
|
|
# This is used by the wp-discourse plugin to associate a remote post with a
|
|
|
|
# discourse post.
|
|
|
|
def create_embedded_topic
|
|
|
|
return unless @opts[:embed_url].present?
|
|
|
|
TopicEmbed.create!(topic_id: @post.topic_id, post_id: @post.id, embed_url: @opts[:embed_url])
|
|
|
|
end
|
|
|
|
|
2014-02-10 13:29:31 -06:00
|
|
|
def handle_spam
|
|
|
|
if @spam
|
|
|
|
GroupMessage.create( Group[:moderators].name,
|
|
|
|
:spam_post_blocked,
|
|
|
|
{ user: @user,
|
|
|
|
limit_once_per: 24.hours,
|
|
|
|
message_params: {domains: @post.linked_hosts.keys.join(', ')} } )
|
|
|
|
elsif @post && !@post.errors.present? && !@opts[:skip_validations]
|
|
|
|
SpamRulesEnforcer.enforce!(@post)
|
2013-10-17 01:44:56 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-02-10 13:29:31 -06:00
|
|
|
def track_latest_on_category
|
|
|
|
return unless @post && @post.errors.count == 0 && @topic && @topic.category_id
|
|
|
|
|
|
|
|
Category.where(id: @topic.category_id).update_all(latest_post_id: @post.id)
|
|
|
|
Category.where(id: @topic.category_id).update_all(latest_topic_id: @topic.id) if @post.post_number == 1
|
|
|
|
end
|
|
|
|
|
2013-09-05 23:07:23 -05:00
|
|
|
def ensure_in_allowed_users
|
|
|
|
return unless @topic.private_message?
|
|
|
|
|
|
|
|
unless @topic.topic_allowed_users.where(user_id: @user.id).exists?
|
|
|
|
@topic.topic_allowed_users.create!(user_id: @user.id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-06-04 13:13:01 -05:00
|
|
|
private
|
|
|
|
|
|
|
|
def setup_topic
|
2014-02-17 00:57:37 -06:00
|
|
|
if new_topic?
|
2013-06-04 13:13:01 -05:00
|
|
|
topic_creator = TopicCreator.new(@user, guardian, @opts)
|
2013-06-07 11:36:37 -05:00
|
|
|
|
|
|
|
begin
|
|
|
|
topic = topic_creator.create
|
|
|
|
@errors = topic_creator.errors
|
|
|
|
rescue ActiveRecord::Rollback => ex
|
|
|
|
# In the event of a rollback, grab the errors from the topic
|
|
|
|
@errors = topic_creator.errors
|
|
|
|
raise ex
|
|
|
|
end
|
2013-06-04 13:13:01 -05:00
|
|
|
else
|
2014-05-06 08:41:59 -05:00
|
|
|
topic = Topic.find_by(id: @opts[:topic_id])
|
2013-06-04 13:13:01 -05:00
|
|
|
guardian.ensure_can_create!(Post, topic)
|
2013-05-02 00:15:17 -05:00
|
|
|
end
|
2013-06-04 13:13:01 -05:00
|
|
|
@topic = topic
|
2013-05-02 00:15:17 -05:00
|
|
|
end
|
|
|
|
|
2013-07-22 00:06:53 -05:00
|
|
|
def update_topic_stats
|
|
|
|
# Update attributes on the topic - featured users and last posted.
|
|
|
|
attrs = {last_posted_at: @post.created_at, last_post_user_id: @post.user_id}
|
|
|
|
attrs[:bumped_at] = @post.created_at unless @post.no_bump
|
2013-12-10 12:47:07 -06:00
|
|
|
attrs[:word_count] = (@topic.word_count || 0) + @post.word_count
|
2014-03-18 12:40:40 -05:00
|
|
|
attrs[:excerpt] = @post.excerpt(220, strip_links: true) if new_topic?
|
2013-07-22 00:06:53 -05:00
|
|
|
@topic.update_attributes(attrs)
|
|
|
|
end
|
|
|
|
|
2013-06-04 13:13:01 -05:00
|
|
|
def setup_post
|
|
|
|
post = @topic.posts.new(raw: @opts[:raw],
|
2013-06-29 16:57:10 -05:00
|
|
|
user: @user,
|
|
|
|
reply_to_post_number: @opts[:reply_to_post_number])
|
2013-06-04 13:13:01 -05:00
|
|
|
|
2013-06-21 10:36:33 -05:00
|
|
|
# Attributes we pass through to the post instance if present
|
2014-02-10 13:29:31 -06:00
|
|
|
[:post_type, :no_bump, :cooking_options, :image_sizes, :acting_user, :invalidate_oneboxes, :cook_method].each do |a|
|
2013-06-21 10:36:33 -05:00
|
|
|
post.send("#{a}=", @opts[a]) if @opts[a].present?
|
|
|
|
end
|
|
|
|
|
2013-06-04 13:13:01 -05:00
|
|
|
post.extract_quoted_post_numbers
|
|
|
|
post.created_at = Time.zone.parse(@opts[:created_at].to_s) if @opts[:created_at].present?
|
2014-02-10 13:29:31 -06:00
|
|
|
|
2014-05-17 20:33:34 -05:00
|
|
|
if fields = @opts[:custom_fields]
|
|
|
|
post.custom_fields = fields
|
|
|
|
end
|
|
|
|
|
2013-06-04 13:13:01 -05:00
|
|
|
@post = post
|
|
|
|
end
|
|
|
|
|
|
|
|
def rollback_if_host_spam_detected
|
2014-03-18 17:02:33 -05:00
|
|
|
return if @opts[:skip_validations]
|
2013-06-04 13:13:01 -05:00
|
|
|
if @post.has_host_spam?
|
|
|
|
@post.errors.add(:base, I18n.t(:spamming_host))
|
|
|
|
@errors = @post.errors
|
|
|
|
@spam = true
|
|
|
|
raise ActiveRecord::Rollback.new
|
2013-05-27 18:13:53 -05:00
|
|
|
end
|
|
|
|
end
|
2013-05-02 00:15:17 -05:00
|
|
|
|
2013-06-04 13:13:01 -05:00
|
|
|
def save_post
|
2013-07-01 21:22:56 -05:00
|
|
|
unless @post.save(validate: !@opts[:skip_validations])
|
2013-06-04 13:13:01 -05:00
|
|
|
@errors = @post.errors
|
2013-05-27 18:13:53 -05:00
|
|
|
raise ActiveRecord::Rollback.new
|
2013-05-02 00:15:17 -05:00
|
|
|
end
|
|
|
|
end
|
2013-06-04 13:13:01 -05:00
|
|
|
|
|
|
|
def store_unique_post_key
|
2013-09-09 15:17:31 -05:00
|
|
|
@post.store_unique_post_key
|
2013-06-04 13:13:01 -05:00
|
|
|
end
|
|
|
|
|
2014-07-28 12:17:37 -05:00
|
|
|
def update_user_counts
|
|
|
|
@user.create_user_stat if @user.user_stat.nil?
|
2014-02-10 13:29:31 -06:00
|
|
|
|
2014-07-28 12:17:37 -05:00
|
|
|
if @user.user_stat.first_post_created_at.nil?
|
|
|
|
@user.user_stat.first_post_created_at = @post.created_at
|
|
|
|
end
|
|
|
|
|
|
|
|
@user.user_stat.post_count += 1
|
|
|
|
@user.user_stat.topic_count += 1 if @post.post_number == 1
|
2013-06-04 13:13:01 -05:00
|
|
|
|
|
|
|
# We don't count replies to your own topics
|
2014-07-03 13:43:24 -05:00
|
|
|
if !@opts[:import_mode] && @user.id != @topic.user_id
|
2013-10-03 22:28:49 -05:00
|
|
|
@user.user_stat.update_topic_reply_count
|
2013-06-04 13:13:01 -05:00
|
|
|
end
|
|
|
|
|
2014-07-28 12:17:37 -05:00
|
|
|
@user.user_stat.save!
|
|
|
|
|
2013-06-04 13:13:01 -05:00
|
|
|
@user.last_posted_at = @post.created_at
|
|
|
|
@user.save!
|
|
|
|
end
|
|
|
|
|
|
|
|
def publish
|
2014-07-03 13:43:24 -05:00
|
|
|
return if @opts[:import_mode]
|
2014-02-10 13:29:31 -06:00
|
|
|
return unless @post.post_number > 1
|
|
|
|
|
|
|
|
MessageBus.publish("/topic/#{@post.topic_id}",{
|
|
|
|
id: @post.id,
|
|
|
|
created_at: @post.created_at,
|
|
|
|
user: BasicUserSerializer.new(@post.user).as_json(root: false),
|
|
|
|
post_number: @post.post_number
|
|
|
|
},
|
2014-03-23 20:19:08 -05:00
|
|
|
group_ids: @topic.secure_group_ids
|
2014-02-10 13:29:31 -06:00
|
|
|
)
|
2013-06-04 13:13:01 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def extract_links
|
|
|
|
TopicLink.extract_from(@post)
|
2014-07-15 02:47:24 -05:00
|
|
|
QuotedPost.extract_from(@post)
|
2013-06-04 13:13:01 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def track_topic
|
2014-02-10 13:29:31 -06:00
|
|
|
return if @opts[:auto_track] == false
|
|
|
|
|
|
|
|
TopicUser.change(@post.user.id,
|
|
|
|
@post.topic.id,
|
|
|
|
posted: true,
|
|
|
|
last_read_post_number: @post.post_number,
|
|
|
|
seen_post_count: @post.post_number)
|
2014-06-03 20:41:42 -05:00
|
|
|
|
|
|
|
|
|
|
|
# assume it took us 5 seconds of reading time to make a post
|
|
|
|
PostTiming.record_timing(topic_id: @post.topic_id,
|
|
|
|
user_id: @post.user_id,
|
|
|
|
post_number: @post.post_number,
|
|
|
|
msecs: 5000)
|
|
|
|
|
|
|
|
|
|
|
|
TopicUser.auto_track(@user.id, @topic.id, TopicUser.notification_reasons[:created_post])
|
2013-06-04 13:13:01 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def enqueue_jobs
|
2014-02-10 13:29:31 -06:00
|
|
|
return unless @post && !@post.errors.present?
|
2014-07-03 13:43:24 -05:00
|
|
|
PostJobsEnqueuer.new(@post, @topic, new_topic?, {import_mode: @opts[:import_mode]}).enqueue_jobs
|
2014-02-17 00:57:37 -06:00
|
|
|
end
|
2014-02-10 13:29:31 -06:00
|
|
|
|
2014-02-17 00:57:37 -06:00
|
|
|
def new_topic?
|
|
|
|
@opts[:topic_id].blank?
|
2013-06-04 13:13:01 -05:00
|
|
|
end
|
2014-02-17 00:57:37 -06:00
|
|
|
|
2013-02-05 13:16:51 -06:00
|
|
|
end
|