discourse/app/controllers/posts_controller.rb

253 lines
7.0 KiB
Ruby
Raw Normal View History

2013-02-06 03:16:51 +08:00
require_dependency 'post_creator'
require_dependency 'post_destroyer'
require_dependency 'distributed_memoizer'
2013-02-06 03:16:51 +08:00
class PostsController < ApplicationController
# Need to be logged in for all actions here
before_filter :ensure_logged_in, except: [:show, :replies, :by_number, :short_link, :versions, :reply_history]
2013-02-06 03:16:51 +08:00
skip_before_filter :store_incoming_links, only: [:short_link]
skip_before_filter :check_xhr, only: [:markdown,:short_link]
def markdown
post = Post.where(topic_id: params[:topic_id].to_i, post_number: (params[:post_number] || 1).to_i).first
if post && guardian.can_see?(post)
render text: post.raw, content_type: 'text/plain'
else
raise Discourse::NotFound
end
end
def short_link
post = Post.find(params[:post_id].to_i)
IncomingLink.add(request,current_user)
redirect_to post.url
end
2013-02-06 03:16:51 +08:00
def create
params = create_params
key = params_key(params)
result = DistributedMemoizer.memoize(key, 120) do
post_creator = PostCreator.new(current_user, params)
post = post_creator.create
if post_creator.errors.present?
# If the post was spam, flag all the user's posts as spam
current_user.flag_linked_posts_as_spam if post_creator.spam?
"e" << MultiJson.dump(errors: post_creator.errors.full_messages)
else
post_serializer = PostSerializer.new(post, scope: guardian, root: false)
post_serializer.topic_slug = post.topic.slug if post.topic.present?
post_serializer.draft_sequence = DraftSequence.current(current_user, post.topic.draft_key)
"s" << MultiJson.dump(post_serializer)
end
2013-02-06 03:16:51 +08:00
end
payload = result[1..-1]
if result[0] == "e"
# don't memoize errors
$redis.del(DistributedMemoizer.redis_key(key))
render json: payload, status: 422
else
render json: payload
end
2013-02-06 03:16:51 +08:00
end
def update
params.require(:post)
2013-02-07 23:45:24 +08:00
post = Post.where(id: params[:id]).first
post.image_sizes = params[:image_sizes] if params[:image_sizes].present?
guardian.ensure_can_edit!(post)
2013-04-10 17:00:50 +08:00
# to stay consistent with the create api,
# we should allow for title changes and category changes here
# we should also move all of this to a post updater.
2013-04-10 17:00:50 +08:00
if post.post_number == 1 && (params[:title] || params[:post][:category])
post.topic.title = params[:title] if params[:title]
2013-04-10 17:00:50 +08:00
Topic.transaction do
post.topic.change_category(params[:post][:category])
post.topic.save
end
if post.topic.errors.present?
render_json_error(post.topic)
return
end
end
revisor = PostRevisor.new(post)
if revisor.revise!(current_user, params[:post][:raw])
TopicLink.extract_from(post)
2013-02-06 03:16:51 +08:00
end
if post.errors.present?
render_json_error(post)
2013-02-06 03:16:51 +08:00
return
end
post_serializer = PostSerializer.new(post, scope: guardian, root: false)
post_serializer.draft_sequence = DraftSequence.current(current_user, post.topic.draft_key)
link_counts = TopicLink.counts_for(guardian,post.topic, [post])
post_serializer.single_post_link_counts = link_counts[post.id] if link_counts.present?
post_serializer.topic_slug = post.topic.slug if post.topic.present?
result = {post: post_serializer.as_json}
if revisor.category_changed.present?
result[:category] = BasicCategorySerializer.new(revisor.category_changed, scope: guardian, root: false).as_json
end
render_json_dump(result)
2013-02-06 03:16:51 +08:00
end
def by_number
@post = Post.where(topic_id: params[:topic_id], post_number: params[:post_number]).first
2013-02-07 23:45:24 +08:00
guardian.ensure_can_see!(@post)
2013-02-06 03:16:51 +08:00
@post.revert_to(params[:version].to_i) if params[:version].present?
render_post_json(@post)
2013-02-06 03:16:51 +08:00
end
def reply_history
@post = Post.where(id: params[:id]).first
guardian.ensure_can_see!(@post)
render_serialized(@post.reply_history, PostSerializer)
end
2013-02-06 03:16:51 +08:00
def show
2013-02-09 06:49:15 +08:00
@post = find_post_from_params
2013-02-06 03:16:51 +08:00
@post.revert_to(params[:version].to_i) if params[:version].present?
render_post_json(@post)
2013-02-06 03:16:51 +08:00
end
def destroy
2013-02-09 06:49:15 +08:00
post = find_post_from_params
guardian.ensure_can_delete!(post)
destroyer = PostDestroyer.new(current_user, post)
destroyer.destroy
render nothing: true
end
def recover
2013-02-09 06:49:15 +08:00
post = find_post_from_params
guardian.ensure_can_recover_post!(post)
destroyer = PostDestroyer.new(current_user, post)
destroyer.recover
post.reload
render_post_json(post)
2013-02-06 03:16:51 +08:00
end
def destroy_many
params.require(:post_ids)
2013-02-06 03:16:51 +08:00
2013-09-04 23:53:00 +08:00
posts = Post.where(id: post_ids_including_replies)
2013-02-06 03:16:51 +08:00
raise Discourse::InvalidParameters.new(:post_ids) if posts.blank?
# Make sure we can delete the posts
2013-09-04 23:53:00 +08:00
2013-02-06 03:16:51 +08:00
posts.each {|p| guardian.ensure_can_delete!(p) }
Post.transaction do
topic_id = posts.first.topic_id
posts.each {|p| PostDestroyer.new(current_user, p).destroy }
2013-02-06 03:16:51 +08:00
end
render nothing: true
end
# Retrieves a list of versions and who made them for a post
def versions
2013-02-09 06:49:15 +08:00
post = find_post_from_params
2013-02-06 03:16:51 +08:00
render_serialized(post.all_versions, VersionSerializer)
end
# Direct replies to this post
def replies
2013-02-09 06:49:15 +08:00
post = find_post_from_params
2013-02-06 03:16:51 +08:00
render_serialized(post.replies, PostSerializer)
end
def bookmark
2013-02-09 06:49:15 +08:00
post = find_post_from_params
2013-02-07 23:45:24 +08:00
if current_user
2013-02-06 03:16:51 +08:00
if params[:bookmarked] == "true"
2013-03-01 20:07:44 +08:00
PostAction.act(current_user, post, PostActionType.types[:bookmark])
2013-02-06 03:16:51 +08:00
else
2013-03-01 20:07:44 +08:00
PostAction.remove_act(current_user, post, PostActionType.types[:bookmark])
2013-02-06 03:16:51 +08:00
end
end
render nothing: true
2013-02-06 03:16:51 +08:00
end
2013-02-09 06:49:15 +08:00
protected
def find_post_from_params
finder = Post.where(id: params[:id] || params[:post_id])
2013-05-07 15:56:56 +08:00
# Include deleted posts if the user is staff
finder = finder.with_deleted if current_user.try(:staff?)
post = finder.first
guardian.ensure_can_see!(post)
post
2013-02-26 00:42:20 +08:00
end
def render_post_json(post)
post_serializer = PostSerializer.new(post, scope: guardian, root: false)
post_serializer.add_raw = true
render_json_dump(post_serializer)
end
private
def params_key(params)
"post##" << Digest::SHA1.hexdigest(params
.to_a
.concat([["user", current_user.id]])
.sort{|x,y| x[0] <=> y[0]}.join do |x,y|
"#{x}:#{y}"
end)
end
def create_params
permitted = [
:raw,
:topic_id,
:title,
:archetype,
:category,
:target_usernames,
:reply_to_post_number,
2013-07-22 09:48:29 +08:00
:auto_close_days,
:auto_track
]
# param munging for WordPress
params[:auto_track] = !(params[:auto_track].to_s == "false") if params[:auto_track]
if api_key_valid?
# php seems to be sending this incorrectly, don't fight with it
params[:skip_validations] = params[:skip_validations].to_s == "true"
permitted << :skip_validations
end
params.require(:raw)
params.permit(*permitted).tap do |whitelisted|
2013-09-27 16:55:50 +08:00
whitelisted[:image_sizes] = params[:image_sizes]
# TODO this does not feel right, we should name what meta_data is allowed
whitelisted[:meta_data] = params[:meta_data]
end
end
2013-02-06 03:16:51 +08:00
end