502 lines
19 KiB
Ruby
502 lines
19 KiB
Ruby
#
|
|
# Copyright (C) 2014 - 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_relative '../spec_helper'
|
|
require_relative '../sharding_spec_helper'
|
|
|
|
# need tests for:
|
|
# overrides that arent date related
|
|
|
|
describe "differentiated_assignments" do
|
|
specs_require_sharding
|
|
|
|
def course_with_differentiated_assignments_enabled
|
|
@course = Course.create!
|
|
@user = user_model
|
|
@course.enroll_user(@user)
|
|
@course.save!
|
|
end
|
|
|
|
def make_assignment(opts={})
|
|
@assignment = Assignment.create!({
|
|
context: @course,
|
|
description: 'descript foo',
|
|
only_visible_to_overrides: opts[:ovto],
|
|
points_possible: rand(1000),
|
|
submission_types: "online_text_entry",
|
|
title: "yes_due_date",
|
|
group_category: opts[:group_category]
|
|
})
|
|
@assignment.publish
|
|
@assignment.save!
|
|
end
|
|
|
|
def assignment_with_true_only_visible_to_overrides
|
|
make_assignment({date: nil, ovto: true})
|
|
end
|
|
|
|
def assignment_with_false_only_visible_to_overrides
|
|
make_assignment({date: Time.now, ovto: false})
|
|
end
|
|
|
|
def group_assignment_with_true_only_visible_to_overrides(opts={})
|
|
group_category = opts[:group_category] || @course.group_categories.first
|
|
make_assignment({date: nil, ovto: true, group_category: group_category})
|
|
end
|
|
|
|
def student_in_course_with_adhoc_override(assignment, opts={})
|
|
@user = opts[:user] || user_model
|
|
StudentEnrollment.create!(:user => @user, :course => @course)
|
|
ao = AssignmentOverride.new()
|
|
ao.assignment = assignment
|
|
ao.title = "ADHOC OVERRIDE"
|
|
ao.workflow_state = "active"
|
|
ao.set_type = "ADHOC"
|
|
ao.save!
|
|
assignment.reload
|
|
override_student = ao.assignment_override_students.build
|
|
override_student.user = @user
|
|
override_student.save!
|
|
@user
|
|
end
|
|
|
|
def enroller_user_in_section(section, opts={})
|
|
@user = opts[:user] || user_model
|
|
StudentEnrollment.create!(:user => @user, :course => @course, :course_section => section)
|
|
end
|
|
|
|
def enroller_user_in_both_sections
|
|
@user = user_model
|
|
StudentEnrollment.create!(:user => @user, :course => @course, :course_section => @section_foo)
|
|
StudentEnrollment.create!(:user => @user, :course => @course, :course_section => @section_bar)
|
|
end
|
|
|
|
def enroll_user_in_group(group, opts={})
|
|
@user = opts[:user] || user_model
|
|
group.add_user(@user, 'accepted', true)
|
|
end
|
|
|
|
def enroller_user_in_both_groups(opts={})
|
|
@user = opts[:user] || user_model
|
|
@group_foo.add_user(@user, 'accepted', true)
|
|
@group_bar.add_user(@user, 'accepted', true)
|
|
end
|
|
|
|
def add_multiple_sections
|
|
@default_section = @course.default_section
|
|
@section_foo = @course.course_sections.create!(:name => 'foo')
|
|
@section_bar = @course.course_sections.create!(:name => 'bar')
|
|
end
|
|
|
|
def add_multiple_groups
|
|
@group_foo = @course.groups.create!(:name => 'foo group')
|
|
@group_bar = @course.groups.create!(:name => 'bar group')
|
|
end
|
|
|
|
def create_override_for_assignment(assignment, &block)
|
|
ao = AssignmentOverride.new()
|
|
ao.assignment = assignment
|
|
ao.title = "Lorem"
|
|
ao.workflow_state = "active"
|
|
block.call(ao)
|
|
ao.save!
|
|
assignment.reload
|
|
end
|
|
|
|
def give_section_due_date(assignment, section)
|
|
create_override_for_assignment(assignment) do |ao|
|
|
ao.set = section
|
|
ao.due_at = 3.weeks.from_now
|
|
end
|
|
end
|
|
|
|
def give_group_due_date(assignment, group)
|
|
assignment.group_category = group.group_category
|
|
create_override_for_assignment(assignment) do |ao|
|
|
ao.set = group
|
|
ao.due_at = 3.weeks.from_now
|
|
end
|
|
end
|
|
|
|
def ensure_user_does_not_see_assignment
|
|
visible_assignment_ids = AssignmentStudentVisibility.where(user_id: @user.id, course_id: @course.id).pluck(:assignment_id)
|
|
expect(visible_assignment_ids.map(&:to_i).include?(@assignment.id)).to be_falsey
|
|
expect(AssignmentStudentVisibility.visible_assignment_ids_in_course_by_user(user_id: [@user.id], course_id: [@course.id])[@user.id]).not_to include(@assignment.id)
|
|
end
|
|
|
|
def ensure_user_sees_assignment
|
|
visible_assignment_ids = AssignmentStudentVisibility.where(user_id: @user.id, course_id: @course.id).pluck(:assignment_id)
|
|
expect(visible_assignment_ids.map(&:to_i).include?(@assignment.id)).to be_truthy
|
|
expect(AssignmentStudentVisibility.visible_assignment_ids_in_course_by_user(user_id: [@user.id], course_id: [@course.id])[@user.id]).to include(@assignment.id)
|
|
expect(AssignmentStudentVisibility.visible_assignment_ids_in_course_by_user(user_id: [@user.id], course_id: [@course.id], use_global_id: true)[Shard.global_id_for(@user.id)]).to include(@assignment.id)
|
|
end
|
|
|
|
context "table" do
|
|
before do
|
|
course_with_differentiated_assignments_enabled
|
|
add_multiple_sections
|
|
assignment_with_true_only_visible_to_overrides
|
|
give_section_due_date(@assignment, @section_foo)
|
|
enroller_user_in_section(@section_foo)
|
|
# at this point there should be an entry in the table
|
|
@visibility_object = AssignmentStudentVisibility.first
|
|
end
|
|
|
|
it "returns objects" do
|
|
expect(@visibility_object).not_to be_nil
|
|
end
|
|
|
|
it "doesnt allow updates" do
|
|
@visibility_object.user_id = @visibility_object.user_id + 1
|
|
expect {@visibility_object.save!}.to raise_error(ActiveRecord::ReadOnlyRecord)
|
|
end
|
|
|
|
it "doesnt allow new records" do
|
|
expect {
|
|
AssignmentStudentVisibility.create!(user_id: @user.id,
|
|
assignment_id: @assignment_id,
|
|
course_id: @course.id)
|
|
}.to raise_error(ActiveRecord::ReadOnlyRecord)
|
|
end
|
|
|
|
it "doesnt allow deletion" do
|
|
expect {@visibility_object.destroy}.to raise_error(ActiveRecord::ReadOnlyRecord)
|
|
end
|
|
end
|
|
|
|
context "course_with_differentiated_assignments_enabled" do
|
|
before do
|
|
course_with_differentiated_assignments_enabled
|
|
add_multiple_sections
|
|
end
|
|
context "assignment only visible to overrides" do
|
|
|
|
context "ADHOC overrides" do
|
|
before { assignment_with_true_only_visible_to_overrides }
|
|
|
|
it "should return a visibility for a student with an ADHOC override" do
|
|
student_in_course_with_adhoc_override(@assignment)
|
|
ensure_user_sees_assignment
|
|
end
|
|
|
|
it "should work with course section and return a single visibility" do
|
|
student_in_course_with_adhoc_override(@assignment)
|
|
give_section_due_date(@assignment, @section_foo)
|
|
enroller_user_in_section(@section_foo)
|
|
ensure_user_sees_assignment
|
|
expect(AssignmentStudentVisibility.where(user_id: @user.id, course_id: @course.id).count).to eq 1
|
|
end
|
|
|
|
it "should not return a visibility for a student without an ADHOC override" do
|
|
@user = user_model
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should not return a visibility if ADHOC override is deleted" do
|
|
student_in_course_with_adhoc_override(@assignment)
|
|
@assignment.assignment_overrides.each(&:destroy)
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
end
|
|
|
|
context "group overrides" do
|
|
before do
|
|
add_multiple_groups
|
|
group_assignment_with_true_only_visible_to_overrides(group_category: @group_foo.group_category)
|
|
give_group_due_date(@assignment, @group_foo)
|
|
end
|
|
|
|
context "user in group with override who then changes groups" do
|
|
before do
|
|
@student = @user
|
|
teacher_in_course(course: @course)
|
|
enroll_user_in_group(@group_foo, {user: @student})
|
|
end
|
|
it "should not keep the assignment visible even if there is a grade" do
|
|
@assignment.grade_student(@student, grade: 10, grader: @teacher)
|
|
@student.group_memberships.each(&:destroy!)
|
|
enroll_user_in_group(@group_bar, {user: @student})
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should not keep the assignment visible if there is no grade" do
|
|
@assignment.grade_student(@student, grade: nil, grader: @teacher)
|
|
@student.group_memberships.each(&:destroy!)
|
|
enroll_user_in_group(@group_bar, {user: @student})
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should not keep the assignment visible even if the grade is zero" do
|
|
@assignment.grade_student(@student, grade: 0, grader: @teacher)
|
|
@student.group_memberships.each(&:destroy!)
|
|
enroll_user_in_group(@group_bar, {user: @student})
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
end
|
|
|
|
context "user not in group with override" do
|
|
it "should hide the assignment from the user" do
|
|
# user not yet in group
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
end
|
|
|
|
context "user in group with override" do
|
|
before do
|
|
enroll_user_in_group(@group_foo, {user: @user})
|
|
end
|
|
|
|
it "should update when enrollments change" do
|
|
ensure_user_sees_assignment
|
|
@user.group_memberships.each(&:destroy!)
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
it "should update when the override is deleted" do
|
|
ensure_user_sees_assignment
|
|
@assignment.assignment_overrides.each(&:destroy!)
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
it "should not return duplicate visibilities with multiple visible sections" do
|
|
enroll_user_in_group(@group_bar, {user: @user})
|
|
give_group_due_date(@assignment, @group_bar)
|
|
visible_assignment_ids = AssignmentStudentVisibility.where(user_id: @user.id, course_id: @course.id)
|
|
expect(visible_assignment_ids.count).to eq 1
|
|
end
|
|
end
|
|
|
|
context "user in groups with and without override" do
|
|
before { enroller_user_in_both_groups(user: @user) }
|
|
it "should show the assignment to the user" do
|
|
ensure_user_sees_assignment
|
|
end
|
|
end
|
|
end
|
|
|
|
context "section overrides" do
|
|
before do
|
|
assignment_with_true_only_visible_to_overrides
|
|
give_section_due_date(@assignment, @section_foo)
|
|
end
|
|
context "user in section with override who then changes sections" do
|
|
before do
|
|
teacher_in_course(course: @course)
|
|
enroller_user_in_section(@section_foo)
|
|
end
|
|
|
|
it "should not keep the assignment visible even if there is a grade" do
|
|
@assignment.grade_student(@user, grade: 10, grader: @teacher)
|
|
Score.where(enrollment_id: @user.enrollments).each(&:destroy_permanently!)
|
|
@user.enrollments.each(&:destroy_permanently!)
|
|
enroller_user_in_section(@section_bar, {user: @user})
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should not keep the assignment visible if there is no grade" do
|
|
@assignment.grade_student(@user, grade: nil, grader: @teacher)
|
|
Score.where(enrollment_id: @user.enrollments).each(&:destroy_permanently!)
|
|
@user.enrollments.each(&:destroy_permanently!)
|
|
enroller_user_in_section(@section_bar, {user: @user})
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should not keep the assignment visible even if the grade is zero" do
|
|
@assignment.grade_student(@user, grade: 0, grader: @teacher)
|
|
Score.where(enrollment_id: @user.enrollments).each(&:destroy_permanently!)
|
|
@user.enrollments.each(&:destroy_permanently!)
|
|
enroller_user_in_section(@section_bar, {user: @user})
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
end
|
|
|
|
context "user in default section" do
|
|
it "should hide the assignment from the user" do
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
end
|
|
context "user in section with override" do
|
|
before{enroller_user_in_section(@section_foo)}
|
|
it "should show the assignment to the user" do
|
|
ensure_user_sees_assignment
|
|
end
|
|
|
|
it "should not show unpublished assignments" do
|
|
@assignment.workflow_state = "unpublished"
|
|
@assignment.save!
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should update when enrollments are destroyed" do
|
|
ensure_user_sees_assignment
|
|
enrollments = StudentEnrollment.where(:user_id => @user.id, :course_id => @course.id, :course_section_id => @section_foo.id)
|
|
enrollments.destroy_all
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should update when enrollments are inactive" do
|
|
ensure_user_sees_assignment
|
|
@user.enrollments.where(:course_id => @course.id, :course_section_id => @section_foo.id).first.deactivate
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should update when the override is deleted" do
|
|
ensure_user_sees_assignment
|
|
@assignment.assignment_overrides.each(&:destroy!)
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
|
|
it "should not return duplicate visibilities with multiple visible sections" do
|
|
enroller_user_in_section(@section_bar, {user: @user})
|
|
give_section_due_date(@assignment, @section_bar)
|
|
visible_assignment_ids = AssignmentStudentVisibility.where(user_id: @user.id, course_id: @course.id)
|
|
expect(visible_assignment_ids.count).to eq 1
|
|
end
|
|
end
|
|
context "user in section with no override" do
|
|
before{enroller_user_in_section(@section_bar)}
|
|
it "should hide the assignment from the user" do
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
end
|
|
context "user in section with override and one without override" do
|
|
before do
|
|
enroller_user_in_both_sections
|
|
end
|
|
it "should show the assignment to the user" do
|
|
ensure_user_sees_assignment
|
|
end
|
|
end
|
|
end
|
|
context "assignment with false only_visible_to_overrides" do
|
|
before do
|
|
assignment_with_false_only_visible_to_overrides
|
|
give_section_due_date(@assignment, @section_foo)
|
|
end
|
|
context "user in default section" do
|
|
it "should show the assignment to the user" do
|
|
ensure_user_sees_assignment
|
|
end
|
|
it "should not show deleted assignments" do
|
|
@assignment.destroy
|
|
ensure_user_does_not_see_assignment
|
|
end
|
|
end
|
|
context "user in section with override" do
|
|
before{enroller_user_in_section(@section_foo)}
|
|
it "should show the assignment to the user" do
|
|
ensure_user_sees_assignment
|
|
end
|
|
end
|
|
context "user in section with no override" do
|
|
before{enroller_user_in_section(@section_bar)}
|
|
it "should show the assignment to the user" do
|
|
ensure_user_sees_assignment
|
|
end
|
|
end
|
|
context "user in section with override and one without override" do
|
|
before do
|
|
enroller_user_in_both_sections
|
|
end
|
|
it "should show the assignment to the user" do
|
|
ensure_user_sees_assignment
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe AssignmentStudentVisibility do
|
|
let!(:course) do
|
|
course = Course.create!
|
|
course.enroll_student(first_student)
|
|
course.enroll_student(second_student)
|
|
course
|
|
end
|
|
|
|
let(:assignment) do
|
|
assignment = course.assignments.create!({
|
|
only_visible_to_overrides: false,
|
|
points_possible: 5,
|
|
submission_types: "online_text_entry",
|
|
title: "assignment"
|
|
})
|
|
assignment.publish
|
|
assignment.save!
|
|
assignment
|
|
end
|
|
let(:first_student) { User.create! }
|
|
let(:second_student) { User.create! }
|
|
let(:fake_student) { User.create! }
|
|
|
|
describe ".assignments_visible_to_all_students" do
|
|
let(:assignments_visible_to_all_students) do
|
|
AssignmentStudentVisibility.assignments_visible_to_all_students([assignment])
|
|
end
|
|
|
|
it "returns a hash with an empty visibility array for each assignment" do
|
|
expect(assignments_visible_to_all_students).to eq({ assignment.id => [] })
|
|
end
|
|
end
|
|
|
|
describe ".assignments_with_user_visibilities" do
|
|
let(:assignment_only_visible_to_overrides) do
|
|
assignment = course.assignments.create!({
|
|
only_visible_to_overrides: true,
|
|
points_possible: 5,
|
|
submission_types: "online_text_entry",
|
|
title: "assignment only visible to overrides"
|
|
})
|
|
override = assignment.assignment_overrides.create!(set_type: "ADHOC")
|
|
override.assignment_override_students.create!(user: first_student)
|
|
assignment
|
|
end
|
|
|
|
let(:assignments_with_visibilities) do
|
|
AssignmentStudentVisibility
|
|
.assignments_with_user_visibilities(course, [assignment, assignment_only_visible_to_overrides])
|
|
end
|
|
|
|
it "returns a hash with assignment ids and their associated user ids " \
|
|
"(or an empty array if the assignment is visible to everyone)" do
|
|
expected_visibilities = {
|
|
assignment.id => [],
|
|
assignment_only_visible_to_overrides.id => [first_student.id]
|
|
}
|
|
expect(assignments_with_visibilities).to eq expected_visibilities
|
|
end
|
|
|
|
it "excludes student ids for deleted enrollments" do
|
|
expected_visibilities = {
|
|
assignment.id => [],
|
|
assignment_only_visible_to_overrides.id => []
|
|
}
|
|
course.enrollments.find_by(user_id: first_student).destroy
|
|
expect(assignments_with_visibilities).to eq expected_visibilities
|
|
end
|
|
|
|
it "does not call AssignmentStudentVisibility.users_with_visibility_by_assignment " \
|
|
"if all assignments are visible to everyone" do
|
|
expect(AssignmentStudentVisibility).to receive(:users_with_visibility_by_assignment).never
|
|
# change this assignment so that it is visible to all students
|
|
assignment_only_visible_to_overrides.only_visible_to_overrides = false
|
|
assignment_only_visible_to_overrides.save!
|
|
assignments_with_visibilities
|
|
end
|
|
end
|
|
end
|
|
|
|
end
|