189 lines
6.6 KiB
Ruby
189 lines
6.6 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
#
|
|
# Copyright (C) 2011 - present Instructure, Inc.
|
|
#
|
|
# This file is part of Canvas.
|
|
#
|
|
# Canvas is free software: you can redistribute it and/or modify it under
|
|
# the terms of the GNU Affero General Public License as published by the Free
|
|
# Software Foundation, version 3 of the License.
|
|
#
|
|
# Canvas is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
# A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
|
|
# details.
|
|
#
|
|
# You should have received a copy of the GNU Affero General Public License along
|
|
# with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
|
|
module Reporting
|
|
class CountsReport
|
|
MONTHS_TO_KEEP = 24
|
|
WEEKS_TO_KEEP = 52
|
|
|
|
def self.process_shard
|
|
reporter = new
|
|
|
|
Account.root_accounts.active.non_shadow.each do |account|
|
|
next if account.external_status == "test"
|
|
|
|
reporter.process_account(account)
|
|
end
|
|
end
|
|
|
|
def initialize
|
|
date = Date.yesterday
|
|
@yesterday = Time.parse("#{date} 23:59:00 UTC")
|
|
@week = date.cweek
|
|
@timestamp = @yesterday
|
|
end
|
|
|
|
def process_account(account)
|
|
GuardRail.activate(:secondary) do
|
|
data = {}.with_indifferent_access
|
|
data[:generated_at] = @timestamp
|
|
data[:id] = account.id
|
|
data[:name] = account.name
|
|
data[:external_status] = account.external_status
|
|
|
|
course_ids = get_course_ids(account)
|
|
data[:courses] = course_ids.length
|
|
|
|
if data[:courses] == 0
|
|
data[:teachers] = 0
|
|
data[:students] = 0
|
|
data[:users] = 0
|
|
data[:files] = 0
|
|
data[:files_size] = 0
|
|
data[:media_files] = 0
|
|
data[:media_files_size] = 0
|
|
else
|
|
timespan = Setting.get("recently_logged_in_timespan", 30.days.to_s).to_i.seconds
|
|
enrollment_scope = Enrollment.active.not_fake
|
|
.joins("INNER JOIN #{Pseudonym.quoted_table_name} ON enrollments.user_id=pseudonyms.user_id")
|
|
.where(pseudonyms: { workflow_state: "active" })
|
|
.where("course_id IN (?) AND pseudonyms.last_request_at>?", course_ids, timespan.seconds.ago)
|
|
|
|
data[:teachers] = enrollment_scope.where(type: "TeacherEnrollment").distinct.count(:user_id)
|
|
data[:students] = enrollment_scope.where(type: "StudentEnrollment").distinct.count(:user_id)
|
|
data[:users] = enrollment_scope.distinct.count(:user_id)
|
|
|
|
# ActiveRecord::Base.calculate doesn't support multiple calculations in account single pass
|
|
data[:files], data[:files_size] = Attachment.connection.select_rows("SELECT COUNT(id), SUM(size) FROM #{Attachment.quoted_table_name} WHERE namespace IN ('account_%s','account_%s') AND root_attachment_id IS NULL AND file_state != 'deleted'" % [account.local_id, account.global_id]).first.map(&:to_i)
|
|
data[:media_files], data[:media_files_size] = MediaObject.connection.select_rows("SELECT COUNT(id), SUM(total_size) FROM #{MediaObject.quoted_table_name} WHERE root_account_id='%s' AND attachment_id IS NULL AND workflow_state != 'deleted'" % [account.id]).first.map(&:to_i)
|
|
data[:media_files_size] *= 1000
|
|
end
|
|
|
|
GuardRail.activate(:primary) do
|
|
detailed = account.report_snapshots.detailed.build
|
|
detailed.created_at = @yesterday
|
|
detailed.data = data
|
|
detailed.save!
|
|
|
|
save_detailed_progressive(account, data)
|
|
end
|
|
end
|
|
|
|
nil
|
|
end
|
|
|
|
private
|
|
|
|
def save_detailed_progressive(account, data)
|
|
progressive = if (snapshot = account.report_snapshots.progressive.last)
|
|
snapshot.data.with_indifferent_access
|
|
else
|
|
new_progressive_hash.with_indifferent_access
|
|
end
|
|
progressive[:generated_at] = @timestamp
|
|
create_progressive_hashes(progressive, data)
|
|
|
|
snapshot = account.report_snapshots.progressive.build
|
|
snapshot.created_at = @yesterday
|
|
snapshot.data = progressive
|
|
snapshot.save!
|
|
end
|
|
|
|
def create_progressive_hashes(cumulative, totals)
|
|
year = { year: @yesterday.year }
|
|
copy_counts(year, totals)
|
|
cumulative[:yearly].pop if cumulative[:yearly].last && cumulative[:yearly].last[:year] == year[:year]
|
|
cumulative[:yearly] << year
|
|
|
|
month = { year: @yesterday.year, month: @yesterday.month }
|
|
copy_counts(month, totals)
|
|
if cumulative[:monthly].last && (cumulative[:monthly].last[:year] == month[:year]) && cumulative[:monthly].last[:month] == month[:month]
|
|
cumulative[:monthly].pop
|
|
end
|
|
cumulative[:monthly] << month
|
|
while cumulative[:monthly].length > MONTHS_TO_KEEP
|
|
cumulative[:monthly].shift
|
|
end
|
|
|
|
week = { year: @yesterday.year, month: @yesterday.month, week: @week }
|
|
copy_counts(week, totals)
|
|
if cumulative[:weekly].last && (cumulative[:weekly].last[:year] == week[:year]) && cumulative[:weekly].last[:week] == week[:week]
|
|
cumulative[:weekly].pop
|
|
end
|
|
cumulative[:weekly] << week
|
|
while cumulative[:weekly].length > WEEKS_TO_KEEP
|
|
cumulative[:weekly].shift
|
|
end
|
|
end
|
|
|
|
def copy_counts(to, from)
|
|
to[:institutions] = from[:institutions]
|
|
to[:courses] = from[:courses]
|
|
to[:teachers] = from[:teachers]
|
|
to[:students] = from[:students]
|
|
to[:users] = from[:users]
|
|
to[:files] = from[:files]
|
|
to[:files_size] = from[:files_size]
|
|
to[:media_files] = from[:media_files]
|
|
to[:media_files_size] = from[:media_files_size]
|
|
end
|
|
|
|
def get_course_ids(account)
|
|
is_default_account = account.external_status == ExternalStatuses.default_external_status.to_s
|
|
course_ids = []
|
|
account.all_courses.where(workflow_state: "available").select([:id, :updated_at]).find_in_batches do |batch|
|
|
course_ids.concat batch.select { |course| !is_default_account || should_use_default_account_course(course) }.map(&:id)
|
|
end
|
|
course_ids
|
|
end
|
|
|
|
def should_use_default_account_course(course)
|
|
@one_month_ago ||= @yesterday - 1.month
|
|
course.updated_at > @one_month_ago
|
|
end
|
|
|
|
def new_counts_hash
|
|
{
|
|
institutions: 0,
|
|
courses: 0,
|
|
teachers: 0,
|
|
students: 0,
|
|
users: 0,
|
|
files: 0,
|
|
files_size: 0,
|
|
media_files: 0,
|
|
media_files_size: 0,
|
|
}
|
|
end
|
|
|
|
def new_progressive_hash
|
|
{ yearly: [], monthly: [], weekly: [] }
|
|
end
|
|
|
|
def start_progressive_hash
|
|
{
|
|
generated_at: @timestamp,
|
|
totals: new_progressive_hash,
|
|
detailed: {}
|
|
}
|
|
end
|
|
end
|
|
end
|