329 lines
11 KiB
Ruby
329 lines
11 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
#
|
|
# Copyright (C) 2016 - 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/>.
|
|
#
|
|
|
|
describe Score do
|
|
before(:once) do
|
|
@grading_periods = grading_periods
|
|
@assignment_group = test_course.assignment_groups.create!(name: "Assignments")
|
|
end
|
|
|
|
let(:test_course) { Course.create! }
|
|
let(:student) { student_in_course(course: test_course) }
|
|
let(:params) do
|
|
{
|
|
course: test_course,
|
|
current_score: 80.2,
|
|
final_score: 74.0,
|
|
updated_at: 1.week.ago
|
|
}
|
|
end
|
|
|
|
let(:grading_period_score_params) do
|
|
params.merge(grading_period_id: @grading_periods.first.id)
|
|
end
|
|
let(:assignment_group_score_params) do
|
|
params.merge(assignment_group_id: @assignment_group.id)
|
|
end
|
|
let(:grading_period_score) { student.scores.create!(grading_period_score_params) }
|
|
let(:assignment_group_score) { student.scores.create!(assignment_group_score_params) }
|
|
|
|
subject_once(:score) { student.scores.create!(params) }
|
|
|
|
it { is_expected.to belong_to(:enrollment).required }
|
|
# shoulda-matchers will have an `optional` method in version 4. As a workaround,
|
|
# I've used the validates_presence_of matcher on the line following the belong_to matcher
|
|
it { is_expected.to belong_to(:grading_period) }
|
|
it { is_expected.not_to validate_presence_of(:grading_period) }
|
|
it { is_expected.to belong_to(:assignment_group) }
|
|
it { is_expected.not_to validate_presence_of(:assignment_group) }
|
|
it { is_expected.to have_one(:score_metadata) }
|
|
it { is_expected.to have_one(:course).through(:enrollment) }
|
|
|
|
it_behaves_like "soft deletion" do
|
|
subject { student.scores }
|
|
|
|
let(:creation_arguments) do
|
|
[
|
|
params.merge(grading_period: @grading_periods.first),
|
|
params.merge(grading_period: @grading_periods.last)
|
|
]
|
|
end
|
|
end
|
|
|
|
describe "validations" do
|
|
it { is_expected.to be_valid }
|
|
it { is_expected.to validate_numericality_of(:current_score).allow_nil }
|
|
it { is_expected.to validate_numericality_of(:unposted_current_score).allow_nil }
|
|
it { is_expected.to validate_numericality_of(:final_score).allow_nil }
|
|
it { is_expected.to validate_numericality_of(:unposted_final_score).allow_nil }
|
|
|
|
it "is invalid without an enrollment" do
|
|
score.enrollment = nil
|
|
expect(score).to be_invalid
|
|
end
|
|
|
|
it { is_expected.to validate_presence_of(:enrollment) }
|
|
|
|
it "is invalid without unique enrollment for course" do
|
|
student.scores.create!(params)
|
|
expect { student.scores.create!(params) }.to raise_error(ActiveRecord::RecordNotUnique)
|
|
end
|
|
|
|
it "is invalid without unique enrollment for grading period" do
|
|
student.scores.create!(grading_period_score_params)
|
|
expect { student.scores.create!(grading_period_score_params) }.to raise_error(ActiveRecord::RecordNotUnique)
|
|
end
|
|
|
|
it("is invalid without unique enrollment for assignment group") do
|
|
student.scores.create!(assignment_group_score_params)
|
|
expect { student.scores.create!(assignment_group_score_params) }.to raise_error(ActiveRecord::RecordNotUnique)
|
|
end
|
|
|
|
context("scorable associations") do
|
|
it "is valid with course_score true and no scorable associations" do
|
|
expect(student.scores.create!(course_score: true, **params)).to be_valid
|
|
end
|
|
|
|
it "is valid with course_score false and a grading period association" do
|
|
expect(student.scores.create!(course_score: false, **grading_period_score_params)).to be_valid
|
|
end
|
|
|
|
it "is valid with course_score false and an assignment group association" do
|
|
expect(student.scores.create!(course_score: false, **assignment_group_score_params)).to be_valid
|
|
end
|
|
|
|
it "is invalid with course_score false and no scorable associations" do
|
|
expect do
|
|
score = student.scores.create!(params)
|
|
score.update!(course_score: false)
|
|
end.to raise_error(ActiveRecord::RecordInvalid)
|
|
end
|
|
|
|
it "is invalid with course_score true and a scorable association" do
|
|
expect do
|
|
student.scores.create!(course_score: true, **grading_period_score_params)
|
|
end.to raise_error(ActiveRecord::RecordInvalid)
|
|
end
|
|
|
|
it "is invalid with multiple scorable associations" do
|
|
expect do
|
|
student.scores.create!(grading_period_id: @grading_periods.first.id, **assignment_group_score_params)
|
|
end.to raise_error(ActiveRecord::RecordInvalid)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "root_account_id" do
|
|
context "on create" do
|
|
it "sets root_account_id to the enrollment's root_account_id if root_account_id is nil" do
|
|
score = student.scores.create!(params)
|
|
expect(score.root_account_id).to eq student.root_account_id
|
|
end
|
|
|
|
it "does not modify root_account_id if it is already set" do
|
|
second_account = account_model
|
|
score = student.scores.create!(params.merge(root_account_id: second_account.id))
|
|
expect(score.root_account_id).to eq second_account.id
|
|
end
|
|
end
|
|
|
|
context "on update" do
|
|
it "sets root_account_id to the enrollment's root_account_id if root_account_id is nil" do
|
|
score.update_column(:root_account_id, nil)
|
|
score.update!(current_score: 0)
|
|
expect(score.root_account_id).to eq student.root_account_id
|
|
end
|
|
|
|
it "does not modify root_account_id if it is already set" do
|
|
second_account = account_model
|
|
score.update!(root_account_id: second_account.id)
|
|
expect(score.root_account_id).to eq second_account.id
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#destroy" do
|
|
context "with score metadata" do
|
|
let(:metadata) { score.create_score_metadata!(calculation_details: { foo: :bar }) }
|
|
|
|
describe "score_metadata association" do
|
|
it "also destroys score metadata" do
|
|
metadata.score.destroy
|
|
expect(metadata).to be_deleted
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#destroy_permanently" do
|
|
context "with score metadata" do
|
|
let(:metadata) { score.create_score_metadata!(calculation_details: { foo: :bar }) }
|
|
|
|
describe "score_metadata association" do
|
|
it "also permanently destroys score metadata" do
|
|
metadata.score.destroy_permanently!
|
|
expect { metadata.reload }.to raise_error(ActiveRecord::RecordNotFound)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#undestroy" do
|
|
context "without score metadata" do
|
|
it "is active" do
|
|
score.destroy
|
|
score.undestroy
|
|
expect(score).to be_active
|
|
end
|
|
end
|
|
|
|
context "with score metadata" do
|
|
let(:metadata) { score.create_score_metadata!(calculation_details: { foo: :bar }) }
|
|
|
|
describe "score_metadata association" do
|
|
it "is active" do
|
|
metadata.score.destroy
|
|
metadata.score.undestroy
|
|
expect(metadata).to be_active
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#current_grade" do
|
|
it "delegates the grade conversion to the course" do
|
|
expect(score.course).to receive(:score_to_grade).once.with(score.current_score)
|
|
score.current_grade
|
|
end
|
|
|
|
it "returns nil if grading schemes are not used in the course" do
|
|
expect(score.course).to receive(:grading_standard_enabled?).and_return(false)
|
|
expect(score.current_grade).to be_nil
|
|
end
|
|
|
|
it "returns the grade according to the course grading scheme" do
|
|
expect(score.course).to receive(:grading_standard_enabled?).and_return(true)
|
|
expect(score.current_grade).to eq "B-"
|
|
end
|
|
end
|
|
|
|
describe "#final_grade" do
|
|
it "delegates the grade conversion to the course" do
|
|
expect(score.course).to receive(:score_to_grade).once.with(score.final_score)
|
|
score.final_grade
|
|
end
|
|
|
|
it "returns nil if grading schemes are not used in the course" do
|
|
expect(score.course).to receive(:grading_standard_enabled?).and_return(false)
|
|
expect(score.final_grade).to be_nil
|
|
end
|
|
|
|
it "returns the grade according to the course grading scheme" do
|
|
expect(score.course).to receive(:grading_standard_enabled?).and_return(true)
|
|
expect(score.final_grade).to eq "C"
|
|
end
|
|
end
|
|
|
|
describe("#scorable") do
|
|
it "returns course for course score" do
|
|
expect(score.scorable).to be score.enrollment.course
|
|
end
|
|
|
|
it "returns grading period for grading period score" do
|
|
expect(grading_period_score.scorable).to be grading_period_score.grading_period
|
|
end
|
|
|
|
it "returns assignment group for assignment group score" do
|
|
expect(assignment_group_score.scorable).to be assignment_group_score.assignment_group
|
|
end
|
|
end
|
|
|
|
describe("#course_score") do
|
|
it "sets course_score to true when there are no scorable associations" do
|
|
expect(score.course_score).to be true
|
|
end
|
|
|
|
it "sets course_score to false for grading period scores" do
|
|
expect(grading_period_score.course_score).to be false
|
|
end
|
|
|
|
it "sets course_score to false for assignment group scores" do
|
|
expect(assignment_group_score.course_score).to be false
|
|
end
|
|
end
|
|
|
|
describe("#params_for_course") do
|
|
it("uses course_score") do
|
|
expect(Score.params_for_course).to eq(course_score: true)
|
|
end
|
|
end
|
|
|
|
context "permissions" do
|
|
it "allows the proper people" do
|
|
expect(score.grants_right?(@enrollment.user, :read)).to be true
|
|
|
|
teacher_in_course(active_all: true)
|
|
expect(score.grants_right?(@teacher, :read)).to be true
|
|
end
|
|
|
|
it "doesn't work for nobody" do
|
|
expect(score.grants_right?(nil, :read)).to be false
|
|
end
|
|
|
|
it "doesn't allow random classmates to read" do
|
|
score
|
|
student_in_course(active_all: true)
|
|
expect(score.grants_right?(@student, :read)).to be false
|
|
end
|
|
|
|
it "doesn't work for yourself if the course is configured badly" do
|
|
@enrollment.course.hide_final_grade = true
|
|
@enrollment.course.save!
|
|
expect(score.grants_right?(@enrollment.user, :read)).to be false
|
|
end
|
|
end
|
|
|
|
describe "final grade override" do
|
|
describe "#effective_final_score" do
|
|
it "returns the override score when one is present" do
|
|
score.update!(override_score: 88)
|
|
expect(score.effective_final_score).to eq 88
|
|
end
|
|
|
|
it "returns the calculated final score when no override is present" do
|
|
expect(score.effective_final_score).to eq 74
|
|
end
|
|
end
|
|
|
|
describe "#effective_final_grade" do
|
|
it "returns a grade commensurate with the override score when one is present" do
|
|
score.update!(override_score: 88)
|
|
allow(score.course).to receive(:grading_standard_enabled?).and_return(true)
|
|
expect(score.effective_final_grade).to eq "B+"
|
|
end
|
|
|
|
it "returns the calculated final grade when no override score is present" do
|
|
allow(score.course).to receive(:grading_standard_enabled?).and_return(true)
|
|
expect(score.effective_final_grade).to eq "C"
|
|
end
|
|
end
|
|
end
|
|
end
|