Dashboard calculations are done with an async job now
This commit is contained in:
parent
1552c4b69e
commit
98b58150bb
|
@ -42,5 +42,9 @@ Discourse.AdminDashboardController = Ember.Controller.extend({
|
||||||
|
|
||||||
problemsTimestamp: function() {
|
problemsTimestamp: function() {
|
||||||
return moment(this.get('problemsFetchedAt')).format('LLL');
|
return moment(this.get('problemsFetchedAt')).format('LLL');
|
||||||
}.property('problemsFetchedAt')
|
}.property('problemsFetchedAt'),
|
||||||
|
|
||||||
|
updatedTimestamp: function() {
|
||||||
|
return moment(this.get('updated_at')).format('LLL');
|
||||||
|
}.property('updated_at')
|
||||||
});
|
});
|
||||||
|
|
|
@ -14,7 +14,7 @@ Discourse.AdminDashboardRoute = Discourse.Route.extend({
|
||||||
},
|
},
|
||||||
|
|
||||||
fetchDashboardData: function(c) {
|
fetchDashboardData: function(c) {
|
||||||
if( !c.get('dashboardFetchedAt') || moment().subtract('hour', 1).toDate() > c.get('dashboardFetchedAt') ) {
|
if( !c.get('dashboardFetchedAt') || moment().subtract('minutes', 30).toDate() > c.get('dashboardFetchedAt') ) {
|
||||||
c.set('dashboardFetchedAt', new Date());
|
c.set('dashboardFetchedAt', new Date());
|
||||||
Discourse.AdminDashboard.find().then(function(d) {
|
Discourse.AdminDashboard.find().then(function(d) {
|
||||||
if( Discourse.SiteSettings.version_checks ){
|
if( Discourse.SiteSettings.version_checks ){
|
||||||
|
@ -32,7 +32,7 @@ Discourse.AdminDashboardRoute = Discourse.Route.extend({
|
||||||
c.set('top_referrers', topReferrers);
|
c.set('top_referrers', topReferrers);
|
||||||
}
|
}
|
||||||
|
|
||||||
['admins', 'moderators', 'blocked', 'banned', 'top_traffic_sources', 'top_referred_topics'].forEach(function(x) {
|
['admins', 'moderators', 'blocked', 'banned', 'top_traffic_sources', 'top_referred_topics', 'updated_at'].forEach(function(x) {
|
||||||
c.set(x, d[x]);
|
c.set(x, d[x]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -276,3 +276,9 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class='clearfix'></div>
|
<div class='clearfix'></div>
|
||||||
|
|
||||||
|
<div class="dashboard-stats pull-right">
|
||||||
|
<div class="pull-right">{{i18n admin.dashboard.last_updated}} {{updatedTimestamp}}</div>
|
||||||
|
<div class='clearfix'></div>
|
||||||
|
</div>
|
||||||
|
<div class='clearfix'></div>
|
|
@ -1,9 +1,6 @@
|
||||||
class Admin::DashboardController < Admin::AdminController
|
class Admin::DashboardController < Admin::AdminController
|
||||||
|
|
||||||
def index
|
def index
|
||||||
dashboard_data = Rails.cache.fetch("admin-dashboard-data-#{Discourse::VERSION::STRING}", expires_in: 1.hour) do
|
dashboard_data = AdminDashboardData.fetch_cached_stats || Jobs::DashboardStats.new.execute({})
|
||||||
AdminDashboardData.fetch_all.as_json
|
|
||||||
end
|
|
||||||
dashboard_data.merge!({version_check: DiscourseUpdates.check_version.as_json}) if SiteSetting.version_checks?
|
dashboard_data.merge!({version_check: DiscourseUpdates.check_version.as_json}) if SiteSetting.version_checks?
|
||||||
render json: dashboard_data
|
render json: dashboard_data
|
||||||
end
|
end
|
||||||
|
|
|
@ -41,9 +41,17 @@ class AdminDashboardData
|
||||||
notification_email_check ].compact
|
notification_email_check ].compact
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.fetch_all
|
def self.fetch_stats
|
||||||
AdminDashboardData.new
|
AdminDashboardData.new
|
||||||
end
|
end
|
||||||
|
def self.fetch_cached_stats
|
||||||
|
# The DashboardStats job is responsible for generating and caching this.
|
||||||
|
stats = $redis.get(stats_cache_key)
|
||||||
|
stats ? JSON.parse(stats) : nil
|
||||||
|
end
|
||||||
|
def self.stats_cache_key
|
||||||
|
'dash-stats'
|
||||||
|
end
|
||||||
|
|
||||||
def self.fetch_problems
|
def self.fetch_problems
|
||||||
AdminDashboardData.new.problems
|
AdminDashboardData.new.problems
|
||||||
|
@ -58,10 +66,16 @@ class AdminDashboardData
|
||||||
blocked: User.blocked.count,
|
blocked: User.blocked.count,
|
||||||
top_referrers: IncomingLinksReport.find('top_referrers').as_json,
|
top_referrers: IncomingLinksReport.find('top_referrers').as_json,
|
||||||
top_traffic_sources: IncomingLinksReport.find('top_traffic_sources').as_json,
|
top_traffic_sources: IncomingLinksReport.find('top_traffic_sources').as_json,
|
||||||
top_referred_topics: IncomingLinksReport.find('top_referred_topics').as_json
|
top_referred_topics: IncomingLinksReport.find('top_referred_topics').as_json,
|
||||||
|
updated_at: Time.zone.now.as_json
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def self.recalculate_interval
|
||||||
|
# Could be configurable, but clockwork + multisite need to support it.
|
||||||
|
30 # minutes
|
||||||
|
end
|
||||||
|
|
||||||
def rails_env_check
|
def rails_env_check
|
||||||
I18n.t("dashboard.rails_env_warning", env: Rails.env) unless Rails.env == 'production'
|
I18n.t("dashboard.rails_env_warning", env: Rails.env) unless Rails.env == 'production'
|
||||||
end
|
end
|
||||||
|
|
|
@ -239,6 +239,7 @@ class SiteSetting < ActiveRecord::Base
|
||||||
setting(:delete_user_max_age, 7)
|
setting(:delete_user_max_age, 7)
|
||||||
setting(:delete_all_posts_max, 10)
|
setting(:delete_all_posts_max, 10)
|
||||||
|
|
||||||
|
|
||||||
def self.generate_api_key!
|
def self.generate_api_key!
|
||||||
self.api_key = SecureRandom.hex(32)
|
self.api_key = SecureRandom.hex(32)
|
||||||
end
|
end
|
||||||
|
|
|
@ -35,5 +35,5 @@ module Clockwork
|
||||||
every(1.minute, 'clockwork_heartbeat')
|
every(1.minute, 'clockwork_heartbeat')
|
||||||
every(1.minute, 'poll_mailbox')
|
every(1.minute, 'poll_mailbox')
|
||||||
every(30.minutes, 'destroy_old_deletion_stubs')
|
every(30.minutes, 'destroy_old_deletion_stubs')
|
||||||
|
every(AdminDashboardData.recalculate_interval.minutes, 'dashboard_stats')
|
||||||
end
|
end
|
||||||
|
|
|
@ -1027,6 +1027,7 @@ en:
|
||||||
|
|
||||||
dashboard:
|
dashboard:
|
||||||
title: "Dashboard"
|
title: "Dashboard"
|
||||||
|
last_updated: "Dashboard last updated:"
|
||||||
version: "Version"
|
version: "Version"
|
||||||
up_to_date: "You're up to date!"
|
up_to_date: "You're up to date!"
|
||||||
critical_available: "A critical update is available."
|
critical_available: "A critical update is available."
|
||||||
|
|
|
@ -663,7 +663,6 @@ en:
|
||||||
delete_user_max_age: "The maximum age of a user, in days, which can be deleted by an admin."
|
delete_user_max_age: "The maximum age of a user, in days, which can be deleted by an admin."
|
||||||
delete_all_posts_max: "The maximum number of posts that can be deleted at once with the Delete All Posts button. If a user has more than this many posts, the posts cannot all be deleted at once and the user can't be deleted."
|
delete_all_posts_max: "The maximum number of posts that can be deleted at once with the Delete All Posts button. If a user has more than this many posts, the posts cannot all be deleted at once and the user can't be deleted."
|
||||||
|
|
||||||
|
|
||||||
notification_types:
|
notification_types:
|
||||||
mentioned: "%{display_username} mentioned you in %{link}"
|
mentioned: "%{display_username} mentioned you in %{link}"
|
||||||
liked: "%{display_username} liked your post in %{link}"
|
liked: "%{display_username} liked your post in %{link}"
|
||||||
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
module Jobs
|
||||||
|
class DashboardStats < Jobs::Base
|
||||||
|
|
||||||
|
def execute(args)
|
||||||
|
stats_json = AdminDashboardData.fetch_stats.as_json
|
||||||
|
|
||||||
|
# Add some extra time to the expiry so that the next job run has plenty of time to
|
||||||
|
# finish before previous cached value expires.
|
||||||
|
$redis.setex AdminDashboardData.stats_cache_key, (AdminDashboardData.recalculate_interval + 5).minutes, stats_json.to_json
|
||||||
|
|
||||||
|
stats_json
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
end
|
|
@ -3,9 +3,7 @@ require_dependency 'discourse_version_check'
|
||||||
|
|
||||||
describe Admin::DashboardController do
|
describe Admin::DashboardController do
|
||||||
before do
|
before do
|
||||||
#NOTE: Rails.cache should be blanked between tests, at the moment we can share state with it
|
AdminDashboardData.stubs(:fetch_cached_stats).returns({reports:[]})
|
||||||
# that is seriously bust on quite a few levels
|
|
||||||
Rails.cache.delete("admin-dashboard-data-#{Discourse::VERSION::STRING}")
|
|
||||||
Jobs::VersionCheck.any_instance.stubs(:execute).returns(true)
|
Jobs::VersionCheck.any_instance.stubs(:execute).returns(true)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -45,13 +43,6 @@ describe Admin::DashboardController do
|
||||||
json['version_check'].should_not be_present
|
json['version_check'].should_not be_present
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'returns report data' do
|
|
||||||
xhr :get, :index
|
|
||||||
json = JSON.parse(response.body)
|
|
||||||
json.should have_key('reports')
|
|
||||||
json['reports'].should be_a(Array)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
context '.problems' do
|
context '.problems' do
|
||||||
|
|
Loading…
Reference in New Issue