2013-10-09 14:22:41 +08:00
|
|
|
require 'current_user'
|
2013-10-09 12:10:37 +08:00
|
|
|
require_dependency 'canonical_url'
|
2013-02-06 03:16:51 +08:00
|
|
|
require_dependency 'discourse'
|
|
|
|
require_dependency 'custom_renderer'
|
2013-10-09 12:10:37 +08:00
|
|
|
require_dependency 'archetype'
|
2013-02-06 03:16:51 +08:00
|
|
|
require_dependency 'rate_limiter'
|
2014-02-21 05:07:02 +08:00
|
|
|
require_dependency 'crawler_detection'
|
2014-04-03 01:22:10 +08:00
|
|
|
require_dependency 'json_error'
|
2014-05-30 12:17:35 +08:00
|
|
|
require_dependency 'letter_avatar'
|
2014-11-14 12:39:17 +08:00
|
|
|
require_dependency 'distributed_cache'
|
2015-03-10 03:24:16 +08:00
|
|
|
require_dependency 'global_path'
|
2013-02-06 03:16:51 +08:00
|
|
|
|
|
|
|
class ApplicationController < ActionController::Base
|
|
|
|
include CurrentUser
|
2013-02-13 19:04:43 +08:00
|
|
|
include CanonicalURL::ControllerExtensions
|
2014-04-03 01:22:10 +08:00
|
|
|
include JsonError
|
2015-03-10 03:24:16 +08:00
|
|
|
include GlobalPath
|
2013-02-06 03:16:51 +08:00
|
|
|
|
|
|
|
serialization_scope :guardian
|
|
|
|
|
|
|
|
protect_from_forgery
|
|
|
|
|
2013-07-29 13:13:13 +08:00
|
|
|
# Default Rails 3.2 lets the request through with a blank session
|
|
|
|
# we are being more pedantic here and nulling session / current_user
|
|
|
|
# and then raising a CSRF exception
|
|
|
|
def handle_unverified_request
|
|
|
|
# NOTE: API key is secret, having it invalidates the need for a CSRF token
|
|
|
|
unless is_api?
|
|
|
|
super
|
|
|
|
clear_current_user
|
2013-08-27 13:56:12 +08:00
|
|
|
render text: "['BAD CSRF']", status: 403
|
2013-07-29 13:13:13 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-05-12 13:27:58 +08:00
|
|
|
before_filter :set_current_user_for_logs
|
2016-02-15 16:29:35 +08:00
|
|
|
before_filter :clear_notifications
|
2013-12-04 19:49:54 +08:00
|
|
|
before_filter :set_locale
|
2013-12-19 03:47:22 +08:00
|
|
|
before_filter :set_mobile_view
|
2013-02-06 03:16:51 +08:00
|
|
|
before_filter :inject_preview_style
|
2013-11-13 01:13:17 +08:00
|
|
|
before_filter :disable_customization
|
2014-02-13 12:37:28 +08:00
|
|
|
before_filter :block_if_readonly_mode
|
2013-02-06 03:16:51 +08:00
|
|
|
before_filter :authorize_mini_profiler
|
|
|
|
before_filter :preload_json
|
2013-06-05 06:32:36 +08:00
|
|
|
before_filter :redirect_to_login_if_required
|
2015-05-22 02:37:49 +08:00
|
|
|
before_filter :check_xhr
|
2015-04-25 02:32:18 +08:00
|
|
|
after_filter :add_readonly_header
|
2013-02-07 23:45:24 +08:00
|
|
|
|
2014-02-15 06:10:08 +08:00
|
|
|
layout :set_layout
|
|
|
|
|
2014-02-21 06:02:26 +08:00
|
|
|
def has_escaped_fragment?
|
|
|
|
SiteSetting.enable_escaped_fragments? && params.key?("_escaped_fragment_")
|
|
|
|
end
|
|
|
|
|
2014-10-31 02:26:35 +08:00
|
|
|
def use_crawler_layout?
|
|
|
|
@use_crawler_layout ||= (has_escaped_fragment? || CrawlerDetection.crawler?(request.user_agent))
|
|
|
|
end
|
|
|
|
|
2015-04-25 02:32:18 +08:00
|
|
|
def add_readonly_header
|
|
|
|
response.headers['Discourse-Readonly'] = 'true' if Discourse.readonly_mode?
|
|
|
|
end
|
|
|
|
|
2014-12-16 00:54:26 +08:00
|
|
|
def slow_platform?
|
|
|
|
request.user_agent =~ /Android/
|
|
|
|
end
|
|
|
|
|
2014-02-15 06:10:08 +08:00
|
|
|
def set_layout
|
2014-10-31 02:26:35 +08:00
|
|
|
use_crawler_layout? ? 'crawler' : 'application'
|
2014-02-15 06:10:08 +08:00
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
# Some exceptions
|
2015-03-23 09:20:50 +08:00
|
|
|
class RenderEmpty < StandardError; end
|
2013-02-06 03:16:51 +08:00
|
|
|
|
2015-02-09 04:25:03 +08:00
|
|
|
# Render nothing
|
2013-02-06 03:16:51 +08:00
|
|
|
rescue_from RenderEmpty do
|
|
|
|
render 'default/empty'
|
|
|
|
end
|
|
|
|
|
2016-03-16 04:06:50 +08:00
|
|
|
def render_rate_limit_error(e)
|
|
|
|
render_json_error e.description, type: :rate_limit, status: 429
|
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
# If they hit the rate limiter
|
|
|
|
rescue_from RateLimiter::LimitExceeded do |e|
|
2016-03-16 04:06:50 +08:00
|
|
|
render_rate_limit_error(e)
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
2015-04-29 23:49:58 +08:00
|
|
|
rescue_from PG::ReadOnlySqlTransaction do |e|
|
|
|
|
Discourse.received_readonly!
|
|
|
|
raise Discourse::ReadOnly
|
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
rescue_from Discourse::NotLoggedIn do |e|
|
|
|
|
raise e if Rails.env.test?
|
2015-02-09 05:35:09 +08:00
|
|
|
if (request.format && request.format.json?) || request.xhr? || !request.get?
|
2015-02-10 02:19:47 +08:00
|
|
|
rescue_discourse_actions(:not_logged_in, 403, true)
|
2014-03-06 00:48:06 +08:00
|
|
|
else
|
2015-10-30 19:51:16 +08:00
|
|
|
rescue_discourse_actions(:not_found, 404)
|
2014-03-06 00:48:06 +08:00
|
|
|
end
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
2016-02-24 17:09:30 +08:00
|
|
|
class PluginDisabled < StandardError; end
|
|
|
|
|
2016-03-24 00:13:29 +08:00
|
|
|
# Handles requests for giant IDs that throw pg exceptions
|
|
|
|
rescue_from RangeError do |e|
|
|
|
|
if e.message =~ /ActiveRecord::ConnectionAdapters::PostgreSQL::OID::Integer/
|
|
|
|
rescue_discourse_actions(:not_found, 404)
|
|
|
|
else
|
|
|
|
raise e
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
rescue_from Discourse::NotFound, PluginDisabled do
|
2015-02-10 02:19:47 +08:00
|
|
|
rescue_discourse_actions(:not_found, 404)
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
rescue_from Discourse::InvalidAccess do
|
2015-02-10 02:19:47 +08:00
|
|
|
rescue_discourse_actions(:invalid_access, 403, true)
|
2013-06-20 23:47:33 +08:00
|
|
|
end
|
|
|
|
|
2014-02-13 12:37:28 +08:00
|
|
|
rescue_from Discourse::ReadOnly do
|
2015-02-23 13:28:50 +08:00
|
|
|
render_json_error I18n.t('read_only_mode_enabled'), type: :read_only, status: 405
|
2014-02-13 12:37:28 +08:00
|
|
|
end
|
|
|
|
|
2015-02-09 04:25:03 +08:00
|
|
|
def rescue_discourse_actions(type, status_code, include_ember=false)
|
2015-02-09 05:35:09 +08:00
|
|
|
|
2015-02-09 04:25:03 +08:00
|
|
|
if (request.format && request.format.json?) || (request.xhr?)
|
2015-02-09 05:35:09 +08:00
|
|
|
# HACK: do not use render_json_error for topics#show
|
|
|
|
if request.params[:controller] == 'topics' && request.params[:action] == 'show'
|
2015-12-30 18:45:54 +08:00
|
|
|
return render status: status_code, layout: false, text: (status_code == 404 || status_code == 410) ? build_not_found_page(status_code) : I18n.t(type)
|
2015-02-09 05:35:09 +08:00
|
|
|
end
|
|
|
|
|
2015-02-23 13:28:50 +08:00
|
|
|
render_json_error I18n.t(type), type: type, status: status_code
|
2013-05-31 02:46:02 +08:00
|
|
|
else
|
2015-02-09 04:25:03 +08:00
|
|
|
render text: build_not_found_page(status_code, include_ember ? 'application' : 'no_ember')
|
2013-05-31 02:46:02 +08:00
|
|
|
end
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
2015-02-05 05:23:39 +08:00
|
|
|
# If a controller requires a plugin, it will raise an exception if that plugin is
|
|
|
|
# disabled. This allows plugins to be disabled programatically.
|
|
|
|
def self.requires_plugin(plugin_name)
|
|
|
|
before_filter do
|
|
|
|
raise PluginDisabled.new if Discourse.disabled_plugin_names.include?(plugin_name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-05-12 13:27:58 +08:00
|
|
|
def set_current_user_for_logs
|
|
|
|
if current_user
|
|
|
|
Logster.add_to_env(request.env,"username",current_user.username)
|
2015-06-16 15:43:36 +08:00
|
|
|
response.headers["X-Discourse-Username"] = current_user.username
|
2014-05-12 13:27:58 +08:00
|
|
|
end
|
2015-06-16 08:27:42 +08:00
|
|
|
response.headers["X-Discourse-Route"] = "#{controller_name}/#{action_name}"
|
2014-05-12 13:27:58 +08:00
|
|
|
end
|
|
|
|
|
2016-02-15 16:29:35 +08:00
|
|
|
def clear_notifications
|
|
|
|
if current_user && !Discourse.readonly_mode?
|
|
|
|
|
|
|
|
cookie_notifications = cookies['cn'.freeze]
|
|
|
|
notifications = request.headers['Discourse-Clear-Notifications'.freeze]
|
|
|
|
|
|
|
|
if cookie_notifications
|
|
|
|
if notifications.present?
|
|
|
|
notifications += "," << cookie_notifications
|
|
|
|
else
|
|
|
|
notifications = cookie_notifications
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if notifications.present?
|
|
|
|
notification_ids = notifications.split(",").map(&:to_i)
|
2016-02-22 09:24:51 +08:00
|
|
|
count = Notification.where(user_id: current_user.id, id: notification_ids, read: false).update_all(read: true)
|
|
|
|
if count > 0
|
|
|
|
current_user.publish_notifications_state
|
|
|
|
end
|
2016-02-15 16:29:35 +08:00
|
|
|
cookies.delete('cn')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-03-01 03:31:39 +08:00
|
|
|
def set_locale
|
2016-02-07 03:49:39 +08:00
|
|
|
if !current_user
|
2016-03-15 07:18:19 +08:00
|
|
|
if SiteSetting.set_locale_from_accept_language_header
|
2016-02-07 03:49:39 +08:00
|
|
|
I18n.locale = locale_from_header
|
|
|
|
else
|
|
|
|
I18n.locale = SiteSetting.default_locale
|
|
|
|
end
|
|
|
|
else
|
|
|
|
I18n.locale = current_user.effective_locale
|
|
|
|
end
|
2015-11-14 04:42:01 +08:00
|
|
|
I18n.ensure_all_loaded!
|
2013-03-01 03:31:39 +08:00
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
def store_preloaded(key, json)
|
|
|
|
@preloaded ||= {}
|
2013-02-26 00:42:20 +08:00
|
|
|
# I dislike that there is a gsub as opposed to a gsub!
|
|
|
|
# but we can not be mucking with user input, I wonder if there is a way
|
2013-02-11 14:28:21 +08:00
|
|
|
# to inject this safty deeper in the library or even in AM serializer
|
|
|
|
@preloaded[key] = json.gsub("</", "<\\/")
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
# If we are rendering HTML, preload the session data
|
|
|
|
def preload_json
|
2013-06-05 00:56:12 +08:00
|
|
|
# We don't preload JSON on xhr or JSON request
|
2015-01-24 13:03:44 +08:00
|
|
|
return if request.xhr? || request.format.json?
|
2013-02-06 03:16:51 +08:00
|
|
|
|
2015-05-20 15:12:16 +08:00
|
|
|
# if we are posting in makes no sense to preload
|
|
|
|
return if request.method != "GET"
|
|
|
|
|
|
|
|
# TODO should not be invoked on redirection so this should be further deferred
|
2013-08-06 04:25:44 +08:00
|
|
|
preload_anonymous_data
|
2013-06-05 00:56:12 +08:00
|
|
|
|
2013-08-06 04:25:44 +08:00
|
|
|
if current_user
|
|
|
|
preload_current_user_data
|
|
|
|
current_user.sync_notification_channel_position
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-12-19 03:47:22 +08:00
|
|
|
def set_mobile_view
|
|
|
|
session[:mobile_view] = params[:mobile_view] if params.has_key?(:mobile_view)
|
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
def inject_preview_style
|
|
|
|
style = request['preview-style']
|
2015-01-06 14:39:08 +08:00
|
|
|
|
|
|
|
if style.nil?
|
|
|
|
session[:preview_style] = cookies[:preview_style]
|
2014-06-21 00:06:36 +08:00
|
|
|
else
|
2015-01-06 14:39:08 +08:00
|
|
|
cookies.delete(:preview_style)
|
|
|
|
|
|
|
|
if style.blank? || style == 'default'
|
|
|
|
session[:preview_style] = nil
|
|
|
|
else
|
|
|
|
session[:preview_style] = style
|
|
|
|
if request['sticky']
|
|
|
|
cookies[:preview_style] = style
|
|
|
|
end
|
|
|
|
end
|
2014-06-21 00:06:36 +08:00
|
|
|
end
|
2015-01-06 14:39:08 +08:00
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
2013-11-13 01:13:17 +08:00
|
|
|
def disable_customization
|
|
|
|
session[:disable_customization] = params[:customization] == "0" if params.has_key?(:customization)
|
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
def guardian
|
|
|
|
@guardian ||= Guardian.new(current_user)
|
|
|
|
end
|
|
|
|
|
2015-06-09 00:07:35 +08:00
|
|
|
def current_homepage
|
|
|
|
current_user ? SiteSetting.homepage : SiteSetting.anonymous_homepage
|
|
|
|
end
|
|
|
|
|
2015-04-01 00:58:56 +08:00
|
|
|
def serialize_data(obj, serializer, opts=nil)
|
2013-02-06 03:16:51 +08:00
|
|
|
# If it's an array, apply the serializer as an each_serializer to the elements
|
2015-04-01 00:58:56 +08:00
|
|
|
serializer_opts = {scope: guardian}.merge!(opts || {})
|
2014-04-16 23:18:09 +08:00
|
|
|
if obj.respond_to?(:to_ary)
|
2013-02-06 03:16:51 +08:00
|
|
|
serializer_opts[:each_serializer] = serializer
|
2014-04-16 23:18:09 +08:00
|
|
|
ActiveModel::ArraySerializer.new(obj.to_ary, serializer_opts).as_json
|
2013-02-07 23:45:24 +08:00
|
|
|
else
|
2013-05-30 04:49:34 +08:00
|
|
|
serializer.new(obj, serializer_opts).as_json
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
2013-05-30 04:49:34 +08:00
|
|
|
end
|
2013-02-06 03:16:51 +08:00
|
|
|
|
2013-05-30 04:49:34 +08:00
|
|
|
# This is odd, but it seems that in Rails `render json: obj` is about
|
|
|
|
# 20% slower than calling MultiJSON.dump ourselves. I'm not sure why
|
|
|
|
# Rails doesn't call MultiJson.dump when you pass it json: obj but
|
|
|
|
# it seems we don't need whatever Rails is doing.
|
2015-04-01 00:58:56 +08:00
|
|
|
def render_serialized(obj, serializer, opts=nil)
|
|
|
|
render_json_dump(serialize_data(obj, serializer, opts), opts)
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
2015-04-01 00:58:56 +08:00
|
|
|
def render_json_dump(obj, opts=nil)
|
|
|
|
opts ||= {}
|
2015-04-28 01:52:37 +08:00
|
|
|
if opts[:rest_serializer]
|
|
|
|
obj['__rest_serializer'] = "1"
|
|
|
|
opts.each do |k, v|
|
|
|
|
obj[k] = v if k.to_s.start_with?("refresh_")
|
|
|
|
end
|
2015-12-01 04:22:58 +08:00
|
|
|
|
|
|
|
obj['extras'] = opts[:extras] if opts[:extras]
|
2015-04-28 01:52:37 +08:00
|
|
|
end
|
|
|
|
|
2015-04-01 00:58:56 +08:00
|
|
|
render json: MultiJson.dump(obj), status: opts[:status] || 200
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
2013-02-07 00:55:54 +08:00
|
|
|
def can_cache_content?
|
2015-10-29 03:11:36 +08:00
|
|
|
current_user.blank? && flash[:authentication_data].blank?
|
2013-02-07 00:55:54 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
# Our custom cache method
|
|
|
|
def discourse_expires_in(time_length)
|
|
|
|
return unless can_cache_content?
|
2014-01-04 15:53:27 +08:00
|
|
|
Middleware::AnonymousCache.anon_cache(request.env, time_length)
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
2014-08-29 00:07:13 +08:00
|
|
|
def fetch_user_from_params(opts=nil)
|
|
|
|
opts ||= {}
|
2014-06-19 02:40:15 +08:00
|
|
|
user = if params[:username]
|
2016-06-11 10:37:33 +08:00
|
|
|
username_lower = params[:username].downcase.chomp('.json')
|
2016-01-15 19:16:00 +08:00
|
|
|
find_opts = { username_lower: username_lower }
|
2016-01-15 19:34:28 +08:00
|
|
|
find_opts[:active] = true unless opts[:include_inactive] || current_user.try(:staff?)
|
2014-08-29 00:07:13 +08:00
|
|
|
User.find_by(find_opts)
|
2014-06-19 02:40:15 +08:00
|
|
|
elsif params[:external_id]
|
2016-06-11 10:37:33 +08:00
|
|
|
external_id = params[:external_id].chomp('.json')
|
2015-02-03 01:58:02 +08:00
|
|
|
SingleSignOnRecord.find_by(external_id: external_id).try(:user)
|
2014-06-19 02:40:15 +08:00
|
|
|
end
|
2015-05-07 09:00:51 +08:00
|
|
|
raise Discourse::NotFound if user.blank?
|
2013-05-22 23:20:16 +08:00
|
|
|
|
|
|
|
guardian.ensure_can_see!(user)
|
|
|
|
user
|
|
|
|
end
|
|
|
|
|
2013-09-04 23:53:00 +08:00
|
|
|
def post_ids_including_replies
|
|
|
|
post_ids = params[:post_ids].map {|p| p.to_i}
|
|
|
|
if params[:reply_post_ids]
|
|
|
|
post_ids << PostReply.where(post_id: params[:reply_post_ids].map {|p| p.to_i}).pluck(:reply_id)
|
|
|
|
post_ids.flatten!
|
|
|
|
post_ids.uniq!
|
|
|
|
end
|
|
|
|
post_ids
|
|
|
|
end
|
|
|
|
|
2015-05-22 14:15:46 +08:00
|
|
|
def no_cookies
|
|
|
|
# do your best to ensure response has no cookies
|
|
|
|
# longer term we may want to push this into middleware
|
|
|
|
headers.delete 'Set-Cookie'
|
|
|
|
request.session_options[:skip] = true
|
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
private
|
|
|
|
|
2016-02-07 03:49:39 +08:00
|
|
|
def locale_from_header
|
|
|
|
begin
|
|
|
|
# Rails I18n uses underscores between the locale and the region; the request
|
|
|
|
# headers use hyphens.
|
2016-03-04 20:17:14 +08:00
|
|
|
require 'http_accept_language' unless defined? HttpAcceptLanguage
|
2016-06-11 10:37:33 +08:00
|
|
|
available_locales = I18n.available_locales.map { |locale| locale.to_s.tr('_', '-') }
|
2016-03-04 20:17:14 +08:00
|
|
|
parser = HttpAcceptLanguage::Parser.new(request.env["HTTP_ACCEPT_LANGUAGE"])
|
2016-06-11 10:37:33 +08:00
|
|
|
parser.language_region_compatible_from(available_locales).tr('-', '_')
|
2016-02-07 03:49:39 +08:00
|
|
|
rescue
|
|
|
|
# If Accept-Language headers are not set.
|
|
|
|
I18n.default_locale
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-08-06 04:25:44 +08:00
|
|
|
def preload_anonymous_data
|
2014-02-25 03:24:18 +08:00
|
|
|
store_preloaded("site", Site.json_for(guardian))
|
2013-08-06 04:25:44 +08:00
|
|
|
store_preloaded("siteSettings", SiteSetting.client_settings_json)
|
2013-12-18 01:49:22 +08:00
|
|
|
store_preloaded("customHTML", custom_html_json)
|
2014-06-19 02:04:10 +08:00
|
|
|
store_preloaded("banner", banner_json)
|
2014-12-23 08:12:26 +08:00
|
|
|
store_preloaded("customEmoji", custom_emoji)
|
2015-12-24 01:09:18 +08:00
|
|
|
store_preloaded("translationOverrides", I18n.client_overrides_json(I18n.locale))
|
2013-08-06 04:25:44 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def preload_current_user_data
|
2013-11-07 01:56:26 +08:00
|
|
|
store_preloaded("currentUser", MultiJson.dump(CurrentUserSerializer.new(current_user, scope: guardian, root: false)))
|
2015-09-07 09:57:50 +08:00
|
|
|
report = TopicTrackingState.report(current_user.id)
|
|
|
|
serializer = ActiveModel::ArraySerializer.new(report, each_serializer: TopicTrackingStateSerializer)
|
2013-08-06 04:25:44 +08:00
|
|
|
store_preloaded("topicTrackingStates", MultiJson.dump(serializer))
|
|
|
|
end
|
|
|
|
|
2013-12-18 01:49:22 +08:00
|
|
|
def custom_html_json
|
2015-02-10 00:48:42 +08:00
|
|
|
target = view_context.mobile_view? ? :mobile : :desktop
|
2015-01-13 09:18:52 +08:00
|
|
|
data = {
|
2015-02-10 00:48:42 +08:00
|
|
|
top: SiteCustomization.custom_top(session[:preview_style], target),
|
|
|
|
footer: SiteCustomization.custom_footer(session[:preview_style], target)
|
2015-01-13 09:18:52 +08:00
|
|
|
}
|
2014-06-05 09:39:33 +08:00
|
|
|
|
|
|
|
if DiscoursePluginRegistry.custom_html
|
|
|
|
data.merge! DiscoursePluginRegistry.custom_html
|
|
|
|
end
|
|
|
|
|
|
|
|
MultiJson.dump(data)
|
2013-12-18 01:25:27 +08:00
|
|
|
end
|
|
|
|
|
2014-11-14 12:39:17 +08:00
|
|
|
def self.banner_json_cache
|
|
|
|
@banner_json_cache ||= DistributedCache.new("banner_json")
|
|
|
|
end
|
|
|
|
|
2014-06-19 02:04:10 +08:00
|
|
|
def banner_json
|
2014-11-14 12:39:17 +08:00
|
|
|
json = ApplicationController.banner_json_cache["json"]
|
|
|
|
|
|
|
|
unless json
|
|
|
|
topic = Topic.where(archetype: Archetype.banner).limit(1).first
|
|
|
|
banner = topic.present? ? topic.banner : {}
|
|
|
|
ApplicationController.banner_json_cache["json"] = json = MultiJson.dump(banner)
|
|
|
|
end
|
|
|
|
|
|
|
|
json
|
2014-06-19 02:04:10 +08:00
|
|
|
end
|
|
|
|
|
2014-12-23 08:12:26 +08:00
|
|
|
def custom_emoji
|
|
|
|
serializer = ActiveModel::ArraySerializer.new(Emoji.custom, each_serializer: EmojiSerializer)
|
|
|
|
MultiJson.dump(serializer)
|
|
|
|
end
|
|
|
|
|
2015-02-09 04:25:03 +08:00
|
|
|
# Render action for a JSON error.
|
|
|
|
#
|
2015-02-23 13:28:50 +08:00
|
|
|
# obj - a translated string, an ActiveRecord model, or an array of translated strings
|
|
|
|
# opts:
|
|
|
|
# type - a machine-readable description of the error
|
|
|
|
# status - HTTP status code to return
|
|
|
|
def render_json_error(obj, opts={})
|
2015-03-19 19:22:56 +08:00
|
|
|
opts = { status: opts } if opts.is_a?(Fixnum)
|
2015-02-23 13:28:50 +08:00
|
|
|
render json: MultiJson.dump(create_errors_json(obj, opts[:type])), status: opts[:status] || 422
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def success_json
|
2014-06-19 02:04:10 +08:00
|
|
|
{ success: 'OK' }
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def failed_json
|
2014-06-19 02:04:10 +08:00
|
|
|
{ failed: 'FAILED' }
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def json_result(obj, opts={})
|
|
|
|
if yield(obj)
|
|
|
|
json = success_json
|
|
|
|
|
|
|
|
# If we were given a serializer, add the class to the json that comes back
|
|
|
|
if opts[:serializer].present?
|
2013-04-12 08:07:46 +08:00
|
|
|
json[obj.class.name.underscore] = opts[:serializer].new(obj, scope: guardian).serializable_hash
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
render json: MultiJson.dump(json)
|
|
|
|
else
|
2014-09-09 03:17:31 +08:00
|
|
|
error_obj = nil
|
|
|
|
if opts[:additional_errors]
|
|
|
|
error_target = opts[:additional_errors].find do |o|
|
|
|
|
target = obj.send(o)
|
|
|
|
target && target.errors.present?
|
|
|
|
end
|
|
|
|
error_obj = obj.send(error_target) if error_target
|
|
|
|
end
|
|
|
|
render_json_error(error_obj || obj)
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
2013-02-07 23:45:24 +08:00
|
|
|
end
|
2013-02-06 03:16:51 +08:00
|
|
|
|
|
|
|
def mini_profiler_enabled?
|
2014-07-17 06:34:30 +08:00
|
|
|
defined?(Rack::MiniProfiler) && guardian.is_developer?
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def authorize_mini_profiler
|
|
|
|
return unless mini_profiler_enabled?
|
|
|
|
Rack::MiniProfiler.authorize_request
|
|
|
|
end
|
|
|
|
|
2013-02-07 23:45:24 +08:00
|
|
|
def check_xhr
|
2013-08-06 04:25:44 +08:00
|
|
|
# bypass xhr check on PUT / POST / DELETE provided api key is there, otherwise calling api is annoying
|
|
|
|
return if !request.get? && api_key_valid?
|
|
|
|
raise RenderEmpty.new unless ((request.format && request.format.json?) || request.xhr?)
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def ensure_logged_in
|
|
|
|
raise Discourse::NotLoggedIn.new unless current_user.present?
|
|
|
|
end
|
2013-02-07 23:45:24 +08:00
|
|
|
|
2015-04-11 05:00:50 +08:00
|
|
|
def ensure_staff
|
|
|
|
raise Discourse::InvalidAccess.new unless current_user && current_user.staff?
|
|
|
|
end
|
|
|
|
|
2015-08-11 23:27:56 +08:00
|
|
|
def destination_url
|
|
|
|
request.original_url unless request.original_url =~ /uploads/
|
|
|
|
end
|
|
|
|
|
2013-06-05 06:32:36 +08:00
|
|
|
def redirect_to_login_if_required
|
2014-01-24 20:47:35 +08:00
|
|
|
return if current_user || (request.format.json? && api_key_valid?)
|
|
|
|
|
2015-02-24 05:20:36 +08:00
|
|
|
# redirect user to the SSO page if we need to log in AND SSO is enabled
|
2015-02-24 06:10:44 +08:00
|
|
|
if SiteSetting.login_required?
|
|
|
|
if SiteSetting.enable_sso?
|
2015-08-11 23:27:56 +08:00
|
|
|
# save original URL in a session so we can redirect after login
|
|
|
|
session[:destination_url] = destination_url
|
2015-03-09 08:45:36 +08:00
|
|
|
redirect_to path('/session/sso')
|
2015-02-24 06:10:44 +08:00
|
|
|
else
|
2015-08-11 23:27:56 +08:00
|
|
|
# save original URL in a cookie (javascript redirects after login in this case)
|
|
|
|
cookies[:destination_url] = destination_url
|
2015-02-24 06:10:44 +08:00
|
|
|
redirect_to :login
|
|
|
|
end
|
2015-02-24 05:20:36 +08:00
|
|
|
end
|
2013-06-05 06:32:36 +08:00
|
|
|
end
|
|
|
|
|
2014-02-13 12:37:28 +08:00
|
|
|
def block_if_readonly_mode
|
2015-03-09 08:45:36 +08:00
|
|
|
return if request.fullpath.start_with?(path "/admin/backups")
|
2015-04-25 02:32:18 +08:00
|
|
|
raise Discourse::ReadOnly.new if !(request.get? || request.head?) && Discourse.readonly_mode?
|
2014-02-13 12:37:28 +08:00
|
|
|
end
|
|
|
|
|
2013-07-12 04:38:46 +08:00
|
|
|
def build_not_found_page(status=404, layout=false)
|
2014-07-05 04:18:09 +08:00
|
|
|
category_topic_ids = Category.pluck(:topic_id).compact
|
2015-07-28 16:02:39 +08:00
|
|
|
@container_class = "wrap not-found-container"
|
2016-02-24 15:56:48 +08:00
|
|
|
@top_viewed = TopicQuery.new(nil, {except_topic_ids: category_topic_ids}).list_top_for("monthly").topics.first(10)
|
2014-07-05 04:18:09 +08:00
|
|
|
@recent = Topic.where.not(id: category_topic_ids).recent(10)
|
2013-05-31 02:46:02 +08:00
|
|
|
@slug = params[:slug].class == String ? params[:slug] : ''
|
2013-06-07 02:41:27 +08:00
|
|
|
@slug = (params[:id].class == String ? params[:id] : '') if @slug.blank?
|
2016-06-11 10:37:33 +08:00
|
|
|
@slug.tr!('-',' ')
|
2013-07-12 04:38:46 +08:00
|
|
|
render_to_string status: status, layout: layout, formats: [:html], template: '/exceptions/not_found'
|
2013-05-31 02:46:02 +08:00
|
|
|
end
|
|
|
|
|
2013-06-20 10:11:14 +08:00
|
|
|
protected
|
2013-06-17 14:09:59 +08:00
|
|
|
|
2014-09-25 10:02:41 +08:00
|
|
|
def render_post_json(post, add_raw=true)
|
2014-09-03 05:37:19 +08:00
|
|
|
post_serializer = PostSerializer.new(post, scope: guardian, root: false)
|
2014-09-25 10:02:41 +08:00
|
|
|
post_serializer.add_raw = add_raw
|
2014-09-03 05:37:19 +08:00
|
|
|
|
|
|
|
counts = PostAction.counts_for([post], current_user)
|
|
|
|
if counts && counts = counts[post.id]
|
|
|
|
post_serializer.post_actions = counts
|
|
|
|
end
|
|
|
|
render_json_dump(post_serializer)
|
|
|
|
end
|
|
|
|
|
2013-06-17 14:09:59 +08:00
|
|
|
def api_key_valid?
|
2013-10-23 03:53:08 +08:00
|
|
|
request["api_key"] && ApiKey.where(key: request["api_key"]).exists?
|
2013-06-17 14:09:59 +08:00
|
|
|
end
|
|
|
|
|
2013-06-20 10:11:14 +08:00
|
|
|
# returns an array of integers given a param key
|
|
|
|
# returns nil if key is not found
|
|
|
|
def param_to_integer_list(key, delimiter = ',')
|
|
|
|
if params[key]
|
|
|
|
params[key].split(delimiter).map(&:to_i)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|