408 lines
14 KiB
Ruby
408 lines
14 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/>.
|
|
#
|
|
|
|
require 'hashery/dictionary'
|
|
|
|
# Contains a dictionary of arrays with hashes in them. This is so that
|
|
# we can get all the submissions for a course grouped by date and
|
|
# ordered by date, person, then assignment. Since working with this is
|
|
# a loop in a loop in a loop, it gets a little awkward for controllers
|
|
# and views, so I contain it in a class with some helper methods. The
|
|
# dictionary comes from facets, a stable and mature library that's been
|
|
# around for years. A dictionary is just an ordered hash.
|
|
#
|
|
# To use this:
|
|
#
|
|
# s = SubmissionList.new(course)
|
|
# s.each {|e| # lists each submission hash in the right order }
|
|
# s.each_day {|e| # lists each day with an array of submission hashes }
|
|
#
|
|
# The submission hash has some very useful meta data in there:
|
|
#
|
|
# :grader => printable name of the grader, or Graded on submission if unknown
|
|
# :grader_id => user_id of the grader
|
|
# :previous_grade => the grade previous to this one, or nil
|
|
# :current_grade => the most current grade, the last submission for this assignment and student
|
|
# :new_grade => the new grade this submission received
|
|
# :assignment_name => a printable name of the assignment
|
|
# :student_user_id => the user_id of the student
|
|
# :course_id => the course id
|
|
# :assignment_id => the assignment id
|
|
# :student_name => a printable name of the student
|
|
# :graded_on => the day (not the time) the submission was made
|
|
# :score_before_regrade => the score prior to regrading
|
|
#
|
|
# The version data is actually pulled from some yaml storage through
|
|
# simply_versioned.
|
|
class SubmissionList
|
|
VALID_KEYS = [
|
|
:assignment_id, :assignment_name, :attachment_id, :attachment_ids,
|
|
:body, :course_id, :created_at, :current_grade, :current_graded_at,
|
|
:current_grader, :grade_matches_current_submission, :graded_at,
|
|
:graded_on, :grader, :grader_id, :group_id, :id, :new_grade,
|
|
:new_graded_at, :new_grader, :previous_grade, :previous_graded_at,
|
|
:previous_grader, :processed, :published_grade,
|
|
:published_score, :safe_grader_id, :score, :student_entered_score,
|
|
:student_user_id, :submission_id, :student_name, :submission_type,
|
|
:updated_at, :url, :user_id, :workflow_state, :score_before_regrade
|
|
].freeze
|
|
|
|
class << self
|
|
# Shortcut for SubmissionList.each(course) { ... }
|
|
def each(course, &block)
|
|
sl = new(course)
|
|
sl.each(&block)
|
|
end
|
|
|
|
def each_day(course, &block)
|
|
sl = new(course)
|
|
sl.each_day(&block)
|
|
end
|
|
|
|
def days(course)
|
|
new(course).days
|
|
end
|
|
|
|
def submission_entries(course)
|
|
new(course).submission_entries
|
|
end
|
|
|
|
def list(course)
|
|
new(course).list
|
|
end
|
|
end
|
|
|
|
# The course
|
|
attr_reader :course
|
|
|
|
# The dictionary of submissions
|
|
attr_reader :list
|
|
|
|
def initialize(course)
|
|
raise ArgumentError, "Must provide a course." unless course && course.is_a?(Course)
|
|
|
|
@course = course
|
|
process
|
|
end
|
|
|
|
# An iterator on a sorted and filtered list of submission versions.
|
|
def each(&block)
|
|
self.submission_entries.each(&block)
|
|
end
|
|
|
|
# An iterator on the day only, not each submission
|
|
def each_day(&block)
|
|
self.list.each(&block)
|
|
end
|
|
|
|
# An array of days with an array of grader open structs for that day and course.
|
|
# TODO - This needes to be refactored, it is way slow and I cant figure out why.
|
|
def days
|
|
# start = Time.now
|
|
# current = Time.now
|
|
# puts "----------------------------------------------"
|
|
# puts "starting"
|
|
# puts "---------------------------------------------------------------------------------"
|
|
self.list.map do |day, _value|
|
|
# puts "-----------------------------------------------item #{Time.now - current}----------------------------"
|
|
# current = Time.now
|
|
OpenObject.new(:date => day, :graders => graders_for_day(day))
|
|
end
|
|
# puts "----------------------------------------------"
|
|
# puts Time.now - start
|
|
# puts "---------------------------------------------------------------------------------"
|
|
# foo
|
|
end
|
|
|
|
# A filtered list of hashes of all submission versions that change the
|
|
# grade with all the meta data finally included. This list can be sorted
|
|
# and displayed.
|
|
def submission_entries
|
|
return @submission_entries if @submission_entries
|
|
|
|
@submission_entries = filtered_submissions.map do |s|
|
|
entry = current_grade_map[s[:id]]
|
|
s[:current_grade] = entry.grade
|
|
s[:current_graded_at] = entry.graded_at
|
|
s[:current_grader] = entry.grader
|
|
s
|
|
end
|
|
trim_keys(@submission_entries)
|
|
end
|
|
|
|
# A cleaner look at a SubmissionList
|
|
def inspect
|
|
"SubmissionList: course: #{self.course.name} submissions used: #{self.submission_entries.size} days used: #{self.list.keys.inspect} graders: #{self.graders.map(&:name).inspect}"
|
|
end
|
|
|
|
protected
|
|
|
|
# Returns an array of graders with an array of assignment open structs
|
|
def graders_for_day(day)
|
|
hsh = self.list[day].inject({}) do |h, submission|
|
|
grader = submission[:grader]
|
|
h[grader] ||= OpenObject.new(
|
|
:assignments => assignments_for_grader_and_day(grader, day),
|
|
:name => grader,
|
|
:grader_id => submission[:grader_id]
|
|
)
|
|
h
|
|
end
|
|
hsh.values
|
|
end
|
|
|
|
# Returns an array of assignments with an array of submission open structs.
|
|
def assignments_for_grader_and_day(grader, day)
|
|
hsh = submission_entries.find_all { |e| e[:grader] == grader and e[:graded_on] == day }.inject({}) do |h, submission|
|
|
assignment = submission[:assignment_name]
|
|
h[assignment] ||= OpenObject.new(
|
|
:name => assignment,
|
|
:assignment_id => submission[:assignment_id],
|
|
:submissions => []
|
|
)
|
|
|
|
h[assignment].submissions << OpenObject.new(submission)
|
|
|
|
h
|
|
end
|
|
|
|
hsh.each_value do |v|
|
|
v['submissions'] = Canvas::ICU.collate_by(v.submissions, &:student_name)
|
|
v.submission_count = v.submissions.size
|
|
end
|
|
|
|
hsh.values
|
|
end
|
|
|
|
# Produce @list, wich is a sorted, filtered, list of submissions with
|
|
# all the meta data we need and no banned keys included.
|
|
def process
|
|
@list = self.submission_entries.sort_by { |a| [a[:graded_at] ? -a[:graded_at].to_f : CanvasSort::Last, a[:safe_grader_id], a[:assignment_id]] }
|
|
.inject(Hashery::Dictionary.new) do |d, se|
|
|
d[se[:graded_on]] ||= []
|
|
d[se[:graded_on]] << se
|
|
d
|
|
end
|
|
end
|
|
|
|
# A hash of the current grades of each submission, keyed by submission.id
|
|
def current_grade_map
|
|
@current_grade_map ||= self.course.submissions.not_placeholder.inject({}) do |hash, submission|
|
|
grader = if submission.grader_id.present?
|
|
self.grader_map[submission.grader_id].try(:name)
|
|
end
|
|
grader ||= I18n.t('gradebooks.history.graded_on_submission', 'Graded on submission')
|
|
|
|
hash[submission.id] = OpenObject.new(:grade => translate_grade(submission),
|
|
:graded_at => submission.graded_at,
|
|
:grader => grader)
|
|
hash
|
|
end
|
|
end
|
|
|
|
# Ensures that the final product only has approved keys in it. This
|
|
# makes our final product much more yummy.
|
|
def trim_keys(list)
|
|
list.each do |hsh|
|
|
hsh.slice!(*VALID_KEYS)
|
|
end
|
|
end
|
|
|
|
# Creates a list of any submissions that change the grade. Adds:
|
|
# * previous_grade
|
|
# * previous_graded_at
|
|
# * previous_grader
|
|
# * new_grade
|
|
# * new_graded_at
|
|
# * new_grader
|
|
# * current_grade
|
|
# * current_graded_at
|
|
# * current_grader
|
|
def filtered_submissions
|
|
return @filtered_submissions if @filtered_submissions
|
|
|
|
# Sorts by submission then updated at in ascending order. So:
|
|
# submission 1 1/1/2009, submission 1 1/15/2009, submission 2 1/1/2009
|
|
full_hash_list.sort_by! { |a| [a[:id], a[:updated_at]] }
|
|
prior_submission_id, prior_grade, prior_score, prior_graded_at, prior_grader = nil
|
|
|
|
@filtered_submissions = full_hash_list.inject([]) do |l, h|
|
|
# If the submission is different (not null for the first one, or just
|
|
# different than the last one), set the previous_grade to nil (this is
|
|
# the first version that changes a grade), set the new_grade to this
|
|
# version's grade, and add this to the list.
|
|
if prior_submission_id != h[:submission_id]
|
|
h[:previous_grade] = nil
|
|
h[:previous_graded_at] = nil
|
|
h[:previous_grader] = nil
|
|
h[:new_grade] = translate_grade(h)
|
|
h[:new_score] = translate_score(h)
|
|
h[:new_graded_at] = h[:graded_at]
|
|
h[:new_grader] = h[:grader]
|
|
l << h
|
|
# If the prior_grade is different than the grade for this version, the
|
|
# grade for this submission has been changed. That's because we know
|
|
# that this submission must be the same as the prior submission.
|
|
# Set the prevous grade and the new grade and add this to the list.
|
|
# Remove the old submission so that it doesn't show up twice in the
|
|
# grade history.
|
|
elsif prior_score != h[:score]
|
|
l.pop if prior_graded_at.try(:to_date) == h[:graded_at].try(:to_date) && prior_grader == h[:grader]
|
|
h[:previous_grade] = prior_grade
|
|
h[:previous_graded_at] = prior_graded_at
|
|
h[:previous_grader] = prior_grader
|
|
h[:new_grade] = translate_grade(h)
|
|
h[:new_score] = translate_score(h)
|
|
h[:new_graded_at] = h[:graded_at]
|
|
h[:new_grader] = h[:grader]
|
|
l << h
|
|
end
|
|
|
|
# At this point, we are only working with versions that have changed a
|
|
# grade. Go ahead and save that grade and save this version as the
|
|
# prior version and iterate.
|
|
prior_grade = translate_grade(h)
|
|
prior_score = translate_score(h)
|
|
prior_graded_at = h[:graded_at]
|
|
prior_grader = h[:grader]
|
|
prior_submission_id = h[:submission_id]
|
|
l
|
|
end
|
|
end
|
|
|
|
def translate_grade(submission)
|
|
submission[:excused] ? "EX" : submission[:grade]
|
|
end
|
|
|
|
def translate_score(submission)
|
|
submission[:excused] ? "EX" : submission[:score]
|
|
end
|
|
|
|
# A list of all versions in YAML format
|
|
def yaml_list
|
|
@yaml_list ||= self.course.submissions.not_placeholder.preload(:versions).map do |s|
|
|
s.versions.map { |v| v.yaml }
|
|
end.flatten
|
|
end
|
|
|
|
# A list of hashes. All the versions of all the submissions for a
|
|
# course, unfiltered and unsorted.
|
|
def raw_hash_list
|
|
@hash_list ||= begin
|
|
hash_list = yaml_list.map { |y| YAML.load(y).symbolize_keys }
|
|
add_regrade_info(hash_list)
|
|
end
|
|
end
|
|
|
|
# This method will add regrade details to the existing raw_hash_list
|
|
def add_regrade_info(hash_list)
|
|
quiz_submission_ids = hash_list.map { |y| y[:quiz_submission_id] }.compact
|
|
return hash_list if quiz_submission_ids.blank?
|
|
|
|
quiz_submissions = Quizzes::QuizSubmission.where("id IN (?) AND score_before_regrade IS NOT NULL", quiz_submission_ids)
|
|
quiz_submissions.each do |qs|
|
|
matches = hash_list.select { |a| a[:id] == qs.submission_id }
|
|
matches.each do |h|
|
|
h[:score_before_regrade] = qs.score_before_regrade
|
|
end
|
|
end
|
|
|
|
hash_list
|
|
end
|
|
|
|
# Still a list of unsorted, unfiltered hashes, but the meta data is inserted at this point
|
|
def full_hash_list
|
|
@full_hash_list ||= self.raw_hash_list.map do |h|
|
|
h[:grader] = if h.has_key? :score_before_regrade
|
|
I18n.t('gradebooks.history.regraded', "Regraded")
|
|
elsif h[:grader_id] && grader_map[h[:grader_id]]
|
|
grader_map[h[:grader_id]].name
|
|
else
|
|
I18n.t('gradebooks.history.graded_on_submission', 'Graded on submission')
|
|
end
|
|
h[:safe_grader_id] = h[:grader_id] ? h[:grader_id] : 0
|
|
h[:assignment_name] = self.assignment_map[h[:assignment_id]].title
|
|
h[:student_user_id] = h[:user_id]
|
|
h[:student_name] = self.student_map[h[:user_id]].name
|
|
h[:course_id] = self.course.id
|
|
h[:submission_id] = h[:id]
|
|
h[:graded_on] = h[:graded_at].in_time_zone.to_date if h[:graded_at]
|
|
|
|
h
|
|
end
|
|
end
|
|
|
|
# A unique list of all grader ids
|
|
def all_grader_ids
|
|
@all_grader_ids ||= raw_hash_list.map { |e| e[:grader_id] }.uniq.compact
|
|
end
|
|
|
|
# A complete list of all graders that have graded submissions for this
|
|
# course as User models
|
|
def graders
|
|
@graders ||= User.where(:id => all_grader_ids).to_a
|
|
end
|
|
|
|
# A hash of graders by their ids, for easy lookup in full_hash_list
|
|
def grader_map
|
|
@grader_map ||= graders.inject({}) do |h, g|
|
|
h[g.id] = g
|
|
h
|
|
end
|
|
end
|
|
|
|
# A unique list of all student ids
|
|
def all_student_ids
|
|
@all_student_ids ||= raw_hash_list.map { |e| e[:user_id] }.uniq.compact
|
|
end
|
|
|
|
# A complete list of all students that have submissions for this course
|
|
# as User models
|
|
def students
|
|
@students ||= User.where(:id => all_student_ids).to_a
|
|
end
|
|
|
|
# A hash of students by their ids, for easy lookup in full_hash_list
|
|
def student_map
|
|
@student_map ||= students.inject({}) do |h, s|
|
|
h[s.id] = s
|
|
h
|
|
end
|
|
end
|
|
|
|
# A unique list of all assignment ids
|
|
def all_assignment_ids
|
|
@all_assignment_ids ||= raw_hash_list.map { |e| e[:assignment_id] }.uniq.compact
|
|
end
|
|
|
|
# A complete list of assignments that have submissions for this course
|
|
def assignments
|
|
@assignments ||= Assignment.where(:id => all_assignment_ids).to_a
|
|
end
|
|
|
|
# A hash of assignments by their ids, for easy lookup in full_hash_list
|
|
def assignment_map
|
|
@assignment_map ||= assignments.inject({}) do |h, a|
|
|
h[a.id] = a
|
|
h
|
|
end
|
|
end
|
|
end
|