2013-03-23 03:47:25 +08:00
|
|
|
require_dependency 'mem_info'
|
|
|
|
|
2013-03-20 11:18:00 +08:00
|
|
|
class AdminDashboardData
|
2015-07-07 12:52:19 +08:00
|
|
|
include StatsCacheable
|
2013-03-20 11:18:00 +08:00
|
|
|
|
2015-06-25 08:42:08 +08:00
|
|
|
GLOBAL_REPORTS ||= [
|
2013-04-17 04:56:18 +08:00
|
|
|
'visits',
|
|
|
|
'signups',
|
2015-09-15 01:30:06 +08:00
|
|
|
'profile_views',
|
2013-04-17 04:56:18 +08:00
|
|
|
'topics',
|
|
|
|
'posts',
|
2015-06-23 01:46:51 +08:00
|
|
|
'time_to_first_response',
|
|
|
|
'topics_with_no_response',
|
2013-04-17 04:56:18 +08:00
|
|
|
'likes',
|
2015-06-25 08:42:08 +08:00
|
|
|
'flags',
|
2013-04-19 02:27:22 +08:00
|
|
|
'bookmarks',
|
2013-04-17 04:56:18 +08:00
|
|
|
'emails',
|
2015-06-25 08:42:08 +08:00
|
|
|
]
|
|
|
|
|
2015-07-08 00:31:07 +08:00
|
|
|
PAGE_VIEW_REPORTS ||= ['page_view_total_reqs'] + ApplicationRequest.req_types.keys.select { |r| r =~ /^page_view_/ && r !~ /mobile/ }.map { |r| r + "_reqs" }
|
2015-06-25 08:42:08 +08:00
|
|
|
|
|
|
|
PRIVATE_MESSAGE_REPORTS ||= [
|
2013-04-17 04:56:18 +08:00
|
|
|
'user_to_user_private_messages',
|
|
|
|
'system_private_messages',
|
|
|
|
'notify_moderators_private_messages',
|
2015-02-06 11:39:04 +08:00
|
|
|
'notify_user_private_messages',
|
2015-06-25 08:42:08 +08:00
|
|
|
'moderator_warning_private_messages',
|
|
|
|
]
|
|
|
|
|
|
|
|
HTTP_REPORTS ||= ApplicationRequest.req_types.keys.select { |r| r =~ /^http_/ }.map { |r| r + "_reqs" }.sort
|
|
|
|
|
|
|
|
USER_REPORTS ||= ['users_by_trust_level']
|
|
|
|
|
2015-07-08 00:31:07 +08:00
|
|
|
MOBILE_REPORTS ||= ['mobile_visits'] + ApplicationRequest.req_types.keys.select {|r| r =~ /mobile/}.map { |r| r + "_reqs" }
|
2013-03-20 11:18:00 +08:00
|
|
|
|
2015-08-26 08:07:40 +08:00
|
|
|
def self.add_problem_check(*syms, &blk)
|
|
|
|
@problem_syms.push(*syms) if syms
|
|
|
|
@problem_blocks << blk if blk
|
|
|
|
end
|
2016-04-06 02:42:24 +08:00
|
|
|
class << self; attr_reader :problem_syms, :problem_blocks, :problem_messages; end
|
2015-08-26 08:07:40 +08:00
|
|
|
|
2013-04-26 05:53:31 +08:00
|
|
|
def problems
|
2015-08-26 08:07:40 +08:00
|
|
|
problems = []
|
|
|
|
AdminDashboardData.problem_syms.each do |sym|
|
|
|
|
problems << send(sym)
|
|
|
|
end
|
|
|
|
AdminDashboardData.problem_blocks.each do |blk|
|
|
|
|
problems << instance_exec(&blk)
|
|
|
|
end
|
2016-04-06 02:42:24 +08:00
|
|
|
AdminDashboardData.problem_messages.each do |i18n_key|
|
|
|
|
problems << AdminDashboardData.problem_message_check(i18n_key)
|
|
|
|
end
|
2016-04-09 04:44:04 +08:00
|
|
|
problems.compact!
|
|
|
|
|
|
|
|
if problems.empty?
|
|
|
|
self.class.clear_problems_started
|
|
|
|
else
|
|
|
|
self.class.set_problems_started
|
|
|
|
end
|
|
|
|
|
|
|
|
problems
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.problems_started_key
|
|
|
|
"dash-problems-started-at"
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.set_problems_started
|
|
|
|
existing_time = $redis.get(problems_started_key)
|
|
|
|
$redis.setex(problems_started_key, 14.days.to_i, existing_time || Time.zone.now.to_s)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.clear_problems_started
|
|
|
|
$redis.del problems_started_key
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.problems_started_at
|
|
|
|
s = $redis.get(problems_started_key)
|
|
|
|
s ? Time.zone.parse(s) : nil
|
2015-08-26 08:07:40 +08:00
|
|
|
end
|
|
|
|
|
2015-09-02 04:32:35 +08:00
|
|
|
# used for testing
|
|
|
|
def self.reset_problem_checks
|
|
|
|
@problem_syms = []
|
|
|
|
@problem_blocks = []
|
2016-04-09 05:33:47 +08:00
|
|
|
|
|
|
|
@problem_messages = [
|
|
|
|
'dashboard.bad_favicon_url',
|
|
|
|
'dashboard.poll_pop3_timeout',
|
|
|
|
'dashboard.poll_pop3_auth_error'
|
|
|
|
]
|
2015-09-02 04:32:35 +08:00
|
|
|
|
2016-06-10 11:00:15 +08:00
|
|
|
add_problem_check :rails_env_check, :host_names_check,
|
2016-05-25 19:08:48 +08:00
|
|
|
:ram_check, :google_oauth2_config_check,
|
2015-09-02 04:32:35 +08:00
|
|
|
:facebook_config_check, :twitter_config_check,
|
|
|
|
:github_config_check, :s3_config_check, :image_magick_check,
|
2016-10-21 05:49:06 +08:00
|
|
|
:failing_emails_check,
|
2016-10-06 00:14:56 +08:00
|
|
|
:subfolder_ends_in_slash_check,
|
2016-03-17 04:17:48 +08:00
|
|
|
:pop3_polling_configuration, :email_polling_errored_recently
|
2015-09-02 04:32:35 +08:00
|
|
|
|
|
|
|
add_problem_check do
|
|
|
|
sidekiq_check || queue_size_check
|
|
|
|
end
|
2013-04-26 05:53:31 +08:00
|
|
|
end
|
2015-09-02 04:32:35 +08:00
|
|
|
reset_problem_checks
|
2014-05-22 06:19:40 +08:00
|
|
|
|
2013-08-03 06:31:25 +08:00
|
|
|
def self.fetch_stats
|
2015-07-07 12:52:19 +08:00
|
|
|
AdminDashboardData.new.as_json
|
2013-08-03 06:31:25 +08:00
|
|
|
end
|
2015-06-25 08:42:08 +08:00
|
|
|
|
2013-08-03 06:31:25 +08:00
|
|
|
def self.stats_cache_key
|
|
|
|
'dash-stats'
|
|
|
|
end
|
2013-03-20 11:18:00 +08:00
|
|
|
|
2013-03-30 03:48:26 +08:00
|
|
|
def self.fetch_problems
|
|
|
|
AdminDashboardData.new.problems
|
|
|
|
end
|
|
|
|
|
2016-04-06 02:42:24 +08:00
|
|
|
def self.problem_message_check(i18n_key)
|
|
|
|
$redis.get(problem_message_key(i18n_key)) ? I18n.t(i18n_key) : nil
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.add_problem_message(i18n_key, expire_seconds=nil)
|
|
|
|
if expire_seconds.to_i > 0
|
|
|
|
$redis.setex problem_message_key(i18n_key), expire_seconds.to_i, 1
|
|
|
|
else
|
|
|
|
$redis.set problem_message_key(i18n_key), 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.clear_problem_message(i18n_key)
|
|
|
|
$redis.del problem_message_key(i18n_key)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.problem_message_key(i18n_key)
|
|
|
|
"admin-problem:#{i18n_key}"
|
|
|
|
end
|
|
|
|
|
2014-08-21 02:14:19 +08:00
|
|
|
def as_json(_options = nil)
|
2013-03-20 11:18:00 +08:00
|
|
|
@json ||= {
|
2015-06-25 08:42:08 +08:00
|
|
|
global_reports: AdminDashboardData.reports(GLOBAL_REPORTS),
|
|
|
|
page_view_reports: AdminDashboardData.reports(PAGE_VIEW_REPORTS),
|
|
|
|
private_message_reports: AdminDashboardData.reports(PRIVATE_MESSAGE_REPORTS),
|
|
|
|
http_reports: AdminDashboardData.reports(HTTP_REPORTS),
|
|
|
|
user_reports: AdminDashboardData.reports(USER_REPORTS),
|
2015-07-08 00:31:07 +08:00
|
|
|
mobile_reports: AdminDashboardData.reports(MOBILE_REPORTS),
|
2013-03-29 14:29:58 +08:00
|
|
|
admins: User.admins.count,
|
2013-05-02 06:12:02 +08:00
|
|
|
moderators: User.moderators.count,
|
2013-11-08 02:53:32 +08:00
|
|
|
suspended: User.suspended.count,
|
2013-06-04 23:53:19 +08:00
|
|
|
blocked: User.blocked.count,
|
2013-05-15 04:17:17 +08:00
|
|
|
top_referrers: IncomingLinksReport.find('top_referrers').as_json,
|
|
|
|
top_traffic_sources: IncomingLinksReport.find('top_traffic_sources').as_json,
|
2013-08-03 06:31:25 +08:00
|
|
|
top_referred_topics: IncomingLinksReport.find('top_referred_topics').as_json,
|
|
|
|
updated_at: Time.zone.now.as_json
|
2013-07-31 00:11:51 +08:00
|
|
|
}
|
2013-03-20 11:18:00 +08:00
|
|
|
end
|
|
|
|
|
2015-06-25 08:42:08 +08:00
|
|
|
def self.reports(source)
|
|
|
|
source.map { |type| Report.find(type).as_json }
|
|
|
|
end
|
|
|
|
|
2013-03-20 11:18:00 +08:00
|
|
|
def rails_env_check
|
2014-08-18 14:42:48 +08:00
|
|
|
I18n.t("dashboard.rails_env_warning", env: Rails.env) unless Rails.env.production?
|
2013-03-20 11:18:00 +08:00
|
|
|
end
|
2013-03-21 03:38:28 +08:00
|
|
|
|
|
|
|
def host_names_check
|
|
|
|
I18n.t("dashboard.host_names_warning") if ['localhost', 'production.localhost'].include?(Discourse.current_hostname)
|
|
|
|
end
|
2013-03-21 04:16:23 +08:00
|
|
|
|
2013-03-22 23:35:32 +08:00
|
|
|
def sidekiq_check
|
|
|
|
last_job_performed_at = Jobs.last_job_performed_at
|
2017-02-15 16:21:17 +08:00
|
|
|
I18n.t('dashboard.sidekiq_warning') if Jobs.queued > 0 && (last_job_performed_at.nil? || last_job_performed_at < 2.minutes.ago)
|
2013-03-22 23:35:32 +08:00
|
|
|
end
|
|
|
|
|
2015-08-07 04:46:49 +08:00
|
|
|
def queue_size_check
|
|
|
|
queue_size = Jobs.queued
|
|
|
|
I18n.t('dashboard.queue_size_warning', queue_size: queue_size) unless queue_size < 100_000
|
|
|
|
end
|
|
|
|
|
2013-03-23 03:47:25 +08:00
|
|
|
def ram_check
|
2017-02-15 16:21:17 +08:00
|
|
|
I18n.t('dashboard.memory_warning') if MemInfo.new.mem_total && MemInfo.new.mem_total < 1_000_000
|
2013-03-23 03:47:25 +08:00
|
|
|
end
|
2013-03-30 01:31:00 +08:00
|
|
|
|
2014-05-22 06:19:40 +08:00
|
|
|
def google_oauth2_config_check
|
2014-05-22 07:11:02 +08:00
|
|
|
I18n.t('dashboard.google_oauth2_config_warning') if SiteSetting.enable_google_oauth2_logins && (SiteSetting.google_oauth2_client_id.blank? || SiteSetting.google_oauth2_client_secret.blank?)
|
2014-05-22 06:19:40 +08:00
|
|
|
end
|
|
|
|
|
2013-03-30 01:31:00 +08:00
|
|
|
def facebook_config_check
|
2014-05-22 06:19:40 +08:00
|
|
|
I18n.t('dashboard.facebook_config_warning') if SiteSetting.enable_facebook_logins && (SiteSetting.facebook_app_id.blank? || SiteSetting.facebook_app_secret.blank?)
|
2013-03-30 01:31:00 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def twitter_config_check
|
2017-02-15 16:21:17 +08:00
|
|
|
I18n.t('dashboard.twitter_config_warning') if SiteSetting.enable_twitter_logins && (SiteSetting.twitter_consumer_key.blank? || SiteSetting.twitter_consumer_secret.blank?)
|
2013-03-30 01:31:00 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def github_config_check
|
2017-02-15 16:21:17 +08:00
|
|
|
I18n.t('dashboard.github_config_warning') if SiteSetting.enable_github_logins && (SiteSetting.github_client_id.blank? || SiteSetting.github_client_secret.blank?)
|
2013-06-20 04:11:11 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def s3_config_check
|
2017-02-15 16:08:32 +08:00
|
|
|
bad_keys = (SiteSetting.s3_access_key_id.blank? || SiteSetting.s3_secret_access_key.blank?) && !SiteSetting.s3_use_iam_profile
|
2014-07-06 06:16:13 +08:00
|
|
|
|
2017-02-15 16:21:17 +08:00
|
|
|
return I18n.t('dashboard.s3_config_warning') if SiteSetting.enable_s3_uploads && (bad_keys || SiteSetting.s3_upload_bucket.blank?)
|
|
|
|
return I18n.t('dashboard.s3_backup_config_warning') if SiteSetting.enable_s3_backups && (bad_keys || SiteSetting.s3_backup_bucket.blank?)
|
2014-03-18 03:56:59 +08:00
|
|
|
nil
|
2013-03-30 01:31:00 +08:00
|
|
|
end
|
2013-04-23 01:37:16 +08:00
|
|
|
|
2013-06-20 04:36:56 +08:00
|
|
|
def image_magick_check
|
2017-02-15 16:21:17 +08:00
|
|
|
I18n.t('dashboard.image_magick_warning') if SiteSetting.create_thumbnails && !system("command -v convert >/dev/null;")
|
2013-06-20 04:36:56 +08:00
|
|
|
end
|
|
|
|
|
2013-04-23 01:37:16 +08:00
|
|
|
def failing_emails_check
|
|
|
|
num_failed_jobs = Jobs.num_email_retry_jobs
|
|
|
|
I18n.t('dashboard.failing_emails_warning', num_failed_jobs: num_failed_jobs) if num_failed_jobs > 0
|
|
|
|
end
|
2013-04-23 03:38:48 +08:00
|
|
|
|
2015-09-07 11:20:59 +08:00
|
|
|
def subfolder_ends_in_slash_check
|
|
|
|
I18n.t('dashboard.subfolder_ends_in_slash') if Discourse.base_uri =~ /\/$/
|
|
|
|
end
|
|
|
|
|
2016-02-17 18:25:49 +08:00
|
|
|
def pop3_polling_configuration
|
|
|
|
POP3PollingEnabledSettingValidator.new.error_message if SiteSetting.pop3_polling_enabled
|
|
|
|
end
|
|
|
|
|
2016-03-17 04:17:48 +08:00
|
|
|
def email_polling_errored_recently
|
|
|
|
errors = Jobs::PollMailbox.errors_in_past_24_hours
|
|
|
|
I18n.t('dashboard.email_polling_errored_recently', count: errors) if errors > 0
|
|
|
|
end
|
|
|
|
|
2016-05-30 23:11:17 +08:00
|
|
|
def missing_mailgun_api_key
|
|
|
|
return unless SiteSetting.reply_by_email_enabled
|
|
|
|
return unless ActionMailer::Base.smtp_settings[:address]["smtp.mailgun.org"]
|
|
|
|
return unless SiteSetting.mailgun_api_key.blank?
|
|
|
|
I18n.t('dashboard.missing_mailgun_api_key')
|
|
|
|
end
|
|
|
|
|
2013-06-20 04:11:11 +08:00
|
|
|
end
|