from __future__ import annotations __copyright__ = "Copyright (C) 2018 Dong Zhuang, Zesheng Wang, Andreas Kloeckner" __license__ = """ Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. """ import datetime import io import unittest import pytest from django.test import Client, TestCase from django.urls import reverse from django.utils.timezone import now, timedelta from course import constants, grades, models from course.constants import ( grade_aggregation_strategy as g_strategy, grade_state_change_types as g_state, participation_permission as pperm, ) from course.flow import reopen_session from course.grades import ( get_single_grade_changes_and_state_machine as get_gc_and_machine, ) from relate.utils import local_now from tests import factories from tests.base_test_mixins import ( HackRepoMixin, MockAddMessageMixing, SingleCoursePageTestMixin, SingleCourseQuizPageTestMixin, ) from tests.constants import QUIZ_FLOW_ID from tests.utils import mock def get_session_grading_rule_use_last_activity_as_cmplt_time_side_effect( session, flow_desc, now_datetime): # The testing flow "quiz-test" didn't set the attribute from course.utils import get_session_grading_rule actual_grading_rule = get_session_grading_rule(session, flow_desc, now_datetime) actual_grading_rule.use_last_activity_as_completion_time = True return actual_grading_rule class GradesTestMixin(SingleCoursePageTestMixin, MockAddMessageMixing): time = now() - timedelta(days=10) @classmethod def setUpTestData(cls): super().setUpTestData() cls.gopp = factories.GradingOpportunityFactory( course=cls.course, aggregation_strategy=g_strategy.use_latest) def setUp(self): super().setUp() self.gopp.refresh_from_db() def use_default_setup(self): self.session1 = factories.FlowSessionFactory.create( participation=self.student_participation, completion_time=self.time) self.time_increment() self.gc_main_1 = factories.GradeChangeFactory.create(**(self.gc(points=5))) self.gc_session1 = factories.GradeChangeFactory.create(**(self.gc(points=0, flow_session=self.session1))) self.session2 = factories.FlowSessionFactory.create( participation=self.student_participation, completion_time=self.time) self.gc_main_2 = factories.GradeChangeFactory.create(**(self.gc(points=7))) self.gc_session2 = factories.GradeChangeFactory.create(**(self.gc(points=6, flow_session=self.session2))) assert models.GradingOpportunity.objects.count() == 1 assert models.GradeChange.objects.count() == 4 assert models.FlowSession.objects.count() == 2 def time_increment(self, minute_delta=10): self.time += timedelta(minutes=minute_delta) @classmethod def gc(cls, opportunity=None, state=None, attempt_id=None, points=None, max_points=None, comment=None, due_time=None, grade_time=None, flow_session=None, null_attempt_id=False, **kwargs): if attempt_id is None: if flow_session is None: if not null_attempt_id: attempt_id = "main" else: from course.flow import get_flow_session_attempt_id attempt_id = get_flow_session_attempt_id(flow_session) gc_kwargs = { "opportunity": opportunity or cls.gopp, "participation": cls.student_participation, "state": state or g_state.graded, "attempt_id": attempt_id, "points": points, "max_points": max_points or 100, "comment": comment, "due_time": due_time, "grade_time": grade_time or cls.time, "flow_session": flow_session, } cls.time += timedelta(minutes=10) gc_kwargs.update(kwargs) return gc_kwargs def get_gc_machine(self, gopp=None, participation=None): if not gopp: gopp = self.gopp if not participation: participation = self.student_participation _, machine = get_gc_and_machine(gopp, participation) return machine def get_gc_stringify_machine_readable_state(self): machine = self.get_gc_machine() return machine.stringify_machine_readable_state() def get_gc_stringify_state(self): machine = self.get_gc_machine() return machine.stringify_state() def update_gopp_strategy(self, strategy=None): if not strategy: return else: self.gopp.aggregation_strategy = strategy self.gopp.save() self.gopp.refresh_from_db() def assertGradeChangeStateEqual(self, expected_state_string=None): # noqa # targeting stringify_state state_string = self.get_gc_stringify_state() from django.utils.encoding import force_str self.assertEqual(force_str(state_string), expected_state_string) def assertGradeChangeMachineReadableStateEqual(self, expected_state_string=None): # noqa # targeting stringify_machine_readable_state state_string = self.get_gc_stringify_machine_readable_state() from decimal import Decimal, InvalidOperation try: percentage = Decimal(state_string, ) except InvalidOperation: percentage = None try: expected_percentage = Decimal(expected_state_string) except InvalidOperation: expected_percentage = None not_equal_msg = ( f"{state_string} does not have equal value with '{expected_percentage!s}'" ) if percentage is not None and expected_percentage is not None: self.assertTrue( abs(percentage - expected_percentage) < 1e-4, msg=not_equal_msg) else: if type(percentage) is not type(expected_percentage): self.fail(not_equal_msg) if percentage is None and expected_percentage is None: self.assertEqual(state_string, expected_state_string) def append_gc(self, gc): return factories.GradeChangeFactory.create(**gc) def update_gc(self, gc_object, update_time=True, **kwargs): # This is alter GradeChange objects via db. gc_dict = gc_object.__dict__ gc_dict.update(**kwargs) if update_time: gc_dict["grade_time"] = now() gc_object.save() gc_object.refresh_from_db() class ViewParticipantGradesTest(GradesTestMixin, TestCase): # test grades.view_participant_grades def test_pctx_no_participation(self): with self.temporarily_switch_to_user(None): resp = self.get_my_grades_view() self.assertEqual(resp.status_code, 403) def test_view_others_grades_no_perm(self): other_participation = factories.ParticipationFactory(course=self.course) with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_participant_grades(other_participation.pk) self.assertEqual(resp.status_code, 403) def test_view_others_grades_no_pctx(self): other_participation = factories.ParticipationFactory(course=self.course) with self.temporarily_switch_to_user(None): resp = self.get_view_participant_grades(other_participation.pk) self.assertEqual(resp.status_code, 403) def test_view_my_participation_grades(self): with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_participant_grades(self.student_participation.pk) self.assertEqual(resp.status_code, 200) def test_view_my_grades(self): with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_my_grades_view() self.assertEqual(resp.status_code, 200) def test_ta_view_student_grades(self): with self.temporarily_switch_to_user(self.ta_participation.user): resp = self.get_view_participant_grades(self.student_participation.pk) self.assertEqual(resp.status_code, 200) def test_view(self): # {{{ gopps. Notice: there is another gopp created in setUp # shown for all (gopp with no gchanges) # shown for all shown_gopp = factories.GradingOpportunityFactory( course=self.course, identifier="1shown", name="SHOWN") # hidden for all view hidden_gopp_all = factories.GradingOpportunityFactory( course=self.course, identifier="2hidden_all", name="HIDDEN_ALL", shown_in_grade_book=False) # hidden if is_privileged_view is True hidden_gopp = factories.GradingOpportunityFactory( course=self.course, identifier="3hidden", name="HIDDEN", shown_in_participant_grade_book=False) # result hidden shown_gopp_result_hidden = factories.GradingOpportunityFactory( course=self.course, identifier="4shown_result_hidden", name="SHOWN_RESULT_HIDDEN", result_shown_in_participant_grade_book=False) # }}} # {{{ gchanges # this will be consumed gchange_shown1 = factories.GradeChangeFactory( **self.gc( opportunity=shown_gopp, attempt_id="main", points=0, max_points=10)) # this won't be consumed gchange_hidden_all = factories.GradeChangeFactory( **self.gc( opportunity=hidden_gopp_all, attempt_id="hidden_all", points=1, max_points=5)) # this will be consumed gchange_result_hidden = factories.GradeChangeFactory( **self.gc( opportunity=shown_gopp_result_hidden, attempt_id="shown_result_hidden", points=15, max_points=15)) # this will be consumed only when is_privileged_view is True gchange_hidden = factories.GradeChangeFactory( **self.gc( opportunity=hidden_gopp, attempt_id="hidden", points=2, max_points=5)) # this will be consumed gchange_shown2 = factories.GradeChangeFactory( **self.gc( opportunity=shown_gopp, attempt_id="main", points=10, max_points=10)) # this will be consumed gchange_shown3 = factories.GradeChangeFactory( **self.gc( opportunity=shown_gopp, attempt_id="main", points=6, max_points=10)) # }}} user = self.student_participation.user with self.temporarily_switch_to_user(user): with self.subTest(user=user): with mock.patch( "course.models.GradeStateMachine.consume") as mock_consume: resp = self.get_my_grades_view() self.assertEqual(resp.status_code, 200) # testing call of GradeStateMachine consume expected_called = [ [gchange_shown1, gchange_shown2, gchange_shown3], [gchange_result_hidden]] # no expected to be consumed not_expected_called = [ [gchange_hidden_all], [gchange_hidden]] actually_called = [] for call in mock_consume.call_args_list: arg, _ = call for not_expected in not_expected_called: self.assertNotIn(not_expected, arg) if len(arg[0]): actually_called.append(arg[0]) self.assertListEqual(actually_called, expected_called) # non mock call resp = self.get_my_grades_view() self.assertEqual(resp.status_code, 200) self.assertEqual( len(resp.context["grading_opportunities"]), 3) self.assertEqual(len(resp.context["grade_table"]), 3) self.assertFalse(resp.context["is_privileged_view"]) self.assertContains(resp, "60.0%", count=1) # for shown_gopp self.assertNotContains(resp, "40.0%") # for hidden_gopp self.assertContains(resp, "(not released)", count=2) # for hidden_gopp # noqa user = self.ta_participation.user with self.temporarily_switch_to_user(user): with self.subTest(user=user): with mock.patch( "course.models.GradeStateMachine.consume") as mock_consume: resp = self.get_view_participant_grades( self.student_participation.pk) self.assertEqual(resp.status_code, 200) # testing call of GradeStateMachine consume expected_called = [ [gchange_shown1, gchange_shown2, gchange_shown3], [gchange_hidden], [gchange_result_hidden]] # no expected to be consumed not_expected_called = [ [gchange_hidden_all]] actually_called = [] for call in mock_consume.call_args_list: arg, _ = call for not_expected in not_expected_called: self.assertNotIn(not_expected, arg) if len(arg[0]): actually_called.append(arg[0]) self.assertListEqual(actually_called, expected_called) # non mock call resp = self.get_view_participant_grades( self.student_participation.pk) self.assertEqual( len(resp.context["grading_opportunities"]), 4) self.assertEqual(len(resp.context["grade_table"]), 4) self.assertTrue(resp.context["is_privileged_view"]) self.assertContains(resp, "60.0%", count=1) # for shown_gopp self.assertContains(resp, "40.0%", count=1) # for hidden_gopp @pytest.mark.slow class GetGradeTableTest(GradesTestMixin, TestCase): # test grades.get_grade_table @classmethod def setUpTestData(cls): super().setUpTestData() # 2 more participations (cls.ptpt1, cls.ptpt2) = factories.ParticipationFactory.create_batch( size=2, course=cls.course) # this make sure it filtered by participation status active factories.ParticipationFactory( course=cls.course, status=constants.participation_status.dropped) # another course and a gopp, this make sure it filtered by course another_course = factories.CourseFactory(identifier="another-course") factories.GradingOpportunityFactory(course=another_course) def run_test(self): super().setUp() # {{{ gopps. Notice: there is another gopp created in setUp # shown for all (gopp with no gchanges) # shown for all shown_gopp = factories.GradingOpportunityFactory( course=self.course, identifier="1shown", name="SHOWN") # hidden for all view hidden_gopp_all = factories.GradingOpportunityFactory( course=self.course, identifier="2hidden_all", name="HIDDEN_ALL", shown_in_grade_book=False) # hidden if is_privileged_view is True hidden_gopp = factories.GradingOpportunityFactory( course=self.course, identifier="3hidden", name="HIDDEN", shown_in_participant_grade_book=False) # }}} # {{{ gchanges for stu stu_shown_gopp_gchanges = [ self.gc( participation=self.student_participation, opportunity=shown_gopp, attempt_id="main", points=0, max_points=10), self.gc( participation=self.student_participation, opportunity=shown_gopp, attempt_id="main", points=2, max_points=10), self.gc( participation=self.student_participation, opportunity=shown_gopp, attempt_id="main", points=5, max_points=10), self.gc( participation=self.student_participation, opportunity=shown_gopp, attempt_id="main", points=4, max_points=10), ] # expecting 40% stu_hidden_gopp_gchanges = [ self.gc( participation=self.student_participation, opportunity=hidden_gopp, attempt_id="hidden", points=5, max_points=10), self.gc( participation=self.student_participation, opportunity=hidden_gopp, attempt_id="hidden", points=4, max_points=10), self.gc( participation=self.student_participation, opportunity=hidden_gopp, attempt_id="hidden", points=3, max_points=10), self.gc( participation=self.student_participation, opportunity=hidden_gopp, attempt_id="hidden", points=2, max_points=10), ] # expecting 20% stu_hidden_all_gopp_gchanges = [ self.gc( participation=self.student_participation, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=5, max_points=10), self.gc( participation=self.student_participation, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=4, max_points=10), self.gc( participation=self.student_participation, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=3, max_points=10), self.gc( participation=self.student_participation, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=2, max_points=10), ] # no result expected # }}} # {{{ gchanges for ptpt1 ptpt1shown_gopp_gchanges = [ self.gc( participation=self.ptpt1, opportunity=shown_gopp, attempt_id="main", points=1, max_points=10), self.gc( participation=self.ptpt1, opportunity=shown_gopp, attempt_id="main", points=3, max_points=10), self.gc( participation=self.ptpt1, opportunity=shown_gopp, attempt_id="main", points=6, max_points=10), self.gc( participation=self.ptpt1, opportunity=shown_gopp, attempt_id="main", points=9, max_points=10), ] # expecting 90% ptpt1_hidden_gopp_gchanges = [ self.gc( participation=self.ptpt1, opportunity=hidden_gopp, attempt_id="hidden", points=10, max_points=10), self.gc( participation=self.ptpt1, opportunity=hidden_gopp, attempt_id="hidden", points=9, max_points=10), self.gc( participation=self.ptpt1, opportunity=hidden_gopp, attempt_id="hidden", points=8, max_points=10), self.gc( participation=self.ptpt1, opportunity=hidden_gopp, attempt_id="hidden", points=7, max_points=10), ] # expecting 70% ptpt1_hidden_all_gopp_gchanges = [ self.gc( participation=self.ptpt1, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=3, max_points=10), self.gc( participation=self.ptpt1, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=2, max_points=10), self.gc( participation=self.ptpt1, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=1, max_points=10), self.gc( participation=self.ptpt1, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=1.5, max_points=10), ] # no result expected # {{{ gchanges for ptpt2 ptpt2_shown_gopp_gchanges = [ self.gc( participation=self.ptpt2, opportunity=shown_gopp, attempt_id="main", points=10, max_points=10), self.gc( participation=self.ptpt2, opportunity=shown_gopp, attempt_id="main", points=1, max_points=10), self.gc( participation=self.ptpt2, opportunity=shown_gopp, attempt_id="main", points=2, max_points=10), self.gc( participation=self.ptpt2, opportunity=shown_gopp, attempt_id="main", points=3.5, max_points=10), ] # expecting 35% ptpt2_hidden_gopp_gchanges = [ self.gc( participation=self.ptpt2, opportunity=hidden_gopp, attempt_id="hidden", points=2, max_points=10), self.gc( participation=self.ptpt2, opportunity=hidden_gopp, attempt_id="hidden", points=4, max_points=10), self.gc( participation=self.ptpt2, opportunity=hidden_gopp, attempt_id="hidden", points=3, max_points=10), self.gc( participation=self.ptpt2, opportunity=hidden_gopp, attempt_id="hidden", points=6.5, max_points=10), ] # expecting 65% ptpt2_hidden_all_gopp_gchanges = [ self.gc( participation=self.ptpt2, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=5, max_points=10), self.gc( participation=self.ptpt2, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=4, max_points=10), self.gc( participation=self.ptpt2, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=3, max_points=10), self.gc( participation=self.ptpt2, opportunity=hidden_gopp_all, attempt_id="hidden_all", points=6, max_points=10), ] # no result expected # }}} gchange_kwargs_lists = [ stu_hidden_all_gopp_gchanges, stu_hidden_gopp_gchanges, stu_shown_gopp_gchanges, ptpt1_hidden_all_gopp_gchanges, ptpt1_hidden_gopp_gchanges, ptpt1shown_gopp_gchanges, ptpt2_hidden_all_gopp_gchanges, ptpt2_hidden_gopp_gchanges, ptpt2_shown_gopp_gchanges] from random import shuffle while True: gchange_kwargs_lists = [lst for lst in gchange_kwargs_lists if len(lst)] if not gchange_kwargs_lists: break shuffle(gchange_kwargs_lists) kwarg_list = gchange_kwargs_lists[0] gchange_kwargs = kwarg_list.pop(0) factories.GradeChangeFactory(**gchange_kwargs) participations, grading_opps, grade_table = ( grades.get_grade_table(self.course)) self.assertEqual( participations, [ self.instructor_participation, self.ta_participation, self.student_participation, self.ptpt1, self.ptpt2]) # hidden_gopp_all not shown # ordered by identifier self.assertListEqual(grading_opps, [shown_gopp, hidden_gopp, self.gopp]) self.assertEqual(len(grade_table), 5) for i in range(5): self.assertEqual(len(grade_table[i]), 3) self.assertEqual(grade_table[2][0].grade_state_machine.percentage(), 40) self.assertEqual(grade_table[2][1].grade_state_machine.percentage(), 20) self.assertEqual(grade_table[2][2].grade_state_machine.percentage(), None) self.assertEqual(grade_table[3][0].grade_state_machine.percentage(), 90) self.assertEqual(grade_table[3][1].grade_state_machine.percentage(), 70) self.assertEqual(grade_table[3][2].grade_state_machine.percentage(), None) self.assertEqual(grade_table[4][0].grade_state_machine.percentage(), 35) self.assertEqual(grade_table[4][1].grade_state_machine.percentage(), 65) self.assertEqual(grade_table[4][2].grade_state_machine.percentage(), None) def test(self): for _i in range(10): self.run_test() factories.UserFactory.reset_sequence(0) self.setUp() fake_access_rules_tag = "fake_tag" fake_task_id = "abcdef123" class MockAsyncRes: def __init__(self): self.id = fake_task_id class ViewGradesByOpportunityTest(GradesTestMixin, TestCase): # test grades.view_grades_by_opportunity def setUp(self): super().setUp() # create 2 flow sessions, one with access_rules_tag factories.FlowSessionFactory( participation=self.student_participation, flow_id=QUIZ_FLOW_ID, in_progress=False, page_count=15) factories.FlowSessionFactory( participation=self.student_participation, access_rules_tag=fake_access_rules_tag, page_count=15) fake_expire_in_progress_sessions = mock.patch( "course.tasks.expire_in_progress_sessions.delay", return_value=MockAsyncRes()) self.mock_expire_in_progress_sessions = ( fake_expire_in_progress_sessions.start()) self.addCleanup(fake_expire_in_progress_sessions.stop) fake_finish_in_progress_sessions = mock.patch( "course.tasks.finish_in_progress_sessions.delay", return_value=MockAsyncRes()) self.mock_finish_in_progress_sessions = ( fake_finish_in_progress_sessions.start()) self.addCleanup(fake_finish_in_progress_sessions.stop) fake_regrade_flow_sessions = mock.patch( "course.tasks.regrade_flow_sessions.delay", return_value=MockAsyncRes()) self.mock_regrade_flow_sessions = fake_regrade_flow_sessions.start() self.addCleanup(fake_regrade_flow_sessions.stop) fake_recalculate_ended_sessions = mock.patch( "course.tasks.recalculate_ended_sessions.delay", return_value=MockAsyncRes()) self.mock_recalculate_ended_sessions = ( fake_recalculate_ended_sessions.start()) self.addCleanup(fake_recalculate_ended_sessions.stop) gopp_id = "la_quiz" def test_no_permission(self): with self.temporarily_switch_to_user(None): resp = self.get_gradebook_by_opp_view( self.gopp_id, force_login_instructor=False) self.assertEqual(resp.status_code, 403) resp = self.post_gradebook_by_opp_view( self.gopp_id, {}, force_login_instructor=False) self.assertEqual(resp.status_code, 403) with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_gradebook_by_opp_view( self.gopp_id, force_login_instructor=False) self.assertEqual(resp.status_code, 403) resp = self.post_gradebook_by_opp_view( self.gopp_id, {}, force_login_instructor=False) self.assertEqual(resp.status_code, 403) def test_gopp_does_not_exist(self): with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.client.get(self.get_gradebook_url_by_opp_id("2")) self.assertEqual(resp.status_code, 404) def test_gopp_course_not_match(self): another_course = factories.CourseFactory(identifier="another-course") another_course_gopp = factories.GradingOpportunityFactory( course=another_course, identifier=self.gopp_id) with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.client.get(self.get_gradebook_url_by_opp_id( another_course_gopp.id)) self.assertEqual(resp.status_code, 400) def test_batch_op_no_permission(self): with self.temporarily_switch_to_user(self.ta_participation.user): for op in ["expire", "end", "regrade", "recalculate"]: with self.subTest(user=self.ta_participation.user, op=op): resp = self.post_gradebook_by_opp_view( self.gopp_id, post_data={"rule_tag": grades.RULE_TAG_NONE_STRING, "past_due_only": True, op: ""}, force_login_instructor=False) # because post is neglected for user without those pperms self.assertEqual(resp.status_code, 200) self.assertEqual( self.mock_expire_in_progress_sessions.call_count, 0) self.assertEqual( self.mock_finish_in_progress_sessions.call_count, 0) self.assertEqual( self.mock_regrade_flow_sessions.call_count, 0) self.assertEqual( self.mock_recalculate_ended_sessions.call_count, 0) def test_batch_op_no_permission2(self): # with partial permission permission_ops = [ (pperm.batch_end_flow_session, "end"), (pperm.batch_impose_flow_session_deadline, "expire"), (pperm.batch_regrade_flow_session, "regrade"), (pperm.batch_recalculate_flow_session_grade, "recalculate")] from itertools import combinations comb = list(combinations(permission_ops, 2)) comb += [reversed(c) for c in comb] with self.temporarily_switch_to_user(self.ta_participation.user): for po in comb: allowed, not_allowed = po pp = models.ParticipationPermission( participation=self.ta_participation, permission=allowed[0]) pp.save() op = not_allowed[1] with self.subTest(user=self.ta_participation.user, op=op): resp = self.post_gradebook_by_opp_view( self.gopp_id, post_data={"rule_tag": grades.RULE_TAG_NONE_STRING, "past_due_only": True, op: ""}, force_login_instructor=False) self.assertEqual(resp.status_code, 403) # revoke permission pp.delete() self.assertEqual( self.mock_expire_in_progress_sessions.call_count, 0) self.assertEqual( self.mock_finish_in_progress_sessions.call_count, 0) self.assertEqual( self.mock_regrade_flow_sessions.call_count, 0) self.assertEqual( self.mock_recalculate_ended_sessions.call_count, 0) def test_batch_op(self): for op in ["expire", "end", "regrade", "recalculate"]: for rule_tag in [fake_access_rules_tag, grades.RULE_TAG_NONE_STRING]: with self.subTest(user=self.instructor_participation.user, op=op): resp = self.post_gradebook_by_opp_view( self.gopp_id, post_data={"rule_tag": rule_tag, "past_due_only": True, op: ""}) self.assertRedirects( resp, reverse( "relate-monitor_task", kwargs={"task_id": fake_task_id}), fetch_redirect_response=False) self.assertEqual( self.mock_expire_in_progress_sessions.call_count, 2) self.assertEqual( self.mock_finish_in_progress_sessions.call_count, 2) self.assertEqual( self.mock_regrade_flow_sessions.call_count, 2) self.assertEqual( self.mock_recalculate_ended_sessions.call_count, 2) def test_invalid_batch_op(self): resp = self.post_gradebook_by_opp_view( self.gopp_id, post_data={"rule_tag": grades.RULE_TAG_NONE_STRING, "past_due_only": True, "invalid_op": ""}) # because post is neglected for user without those pperms self.assertEqual(resp.status_code, 400) self.assertEqual( self.mock_expire_in_progress_sessions.call_count, 0) self.assertEqual( self.mock_finish_in_progress_sessions.call_count, 0) self.assertEqual( self.mock_regrade_flow_sessions.call_count, 0) self.assertEqual( self.mock_recalculate_ended_sessions.call_count, 0) def test_post_form_invalid(self): with mock.patch( "course.grades.ModifySessionsForm.is_valid") as mock_form_valid: mock_form_valid.return_value = False resp = self.post_gradebook_by_opp_view( self.gopp_id, post_data={"rule_tag": grades.RULE_TAG_NONE_STRING, "past_due_only": True, "end": ""}) # just ignore self.assertEqual(resp.status_code, 200) self.assertEqual( self.mock_expire_in_progress_sessions.call_count, 0) self.assertEqual( self.mock_finish_in_progress_sessions.call_count, 0) self.assertEqual( self.mock_regrade_flow_sessions.call_count, 0) self.assertEqual( self.mock_recalculate_ended_sessions.call_count, 0) def test_get_flow_status(self): factories.FlowSessionFactory(participation=self.student_participation, in_progress=True, page_count=13) # There're 3 participations, student has 2 finished session, # 1 in-progress session not_started = 'not started' finished = 'finished' unfinished = 'unfinished' resp = self.get_gradebook_by_opp_view(self.gopp_id) self.assertEqual(resp.status_code, 200) # The instructor and ta didn't start the session self.assertContains(resp, not_started, count=2, html=True) self.assertContains(resp, finished, count=2, html=True) self.assertContains(resp, unfinished, count=1, html=True) resp = self.get_gradebook_by_opp_view(self.gopp_id, view_page_grades=True) self.assertEqual(resp.status_code, 200) # The student_participation has 2 session in setUp self.assertContains(resp, finished, count=2, html=True) self.assertContains(resp, unfinished, count=1, html=True) # no "not started" when view_page_grades self.assertContains(resp, not_started, count=0, html=True) # remove all flow sessions for fs in models.FlowSession.objects.all(): fs.delete() resp = self.get_gradebook_by_opp_view(self.gopp_id) self.assertEqual(resp.status_code, 200) self.assertContains(resp, not_started, count=3, html=True) self.assertContains(resp, finished, count=0, html=True) self.assertContains(resp, unfinished, count=0, html=True) resp = self.get_gradebook_by_opp_view(self.gopp_id, view_page_grades=True) self.assertEqual(resp.status_code, 200) # no "not started" when view_page_grades self.assertContains(resp, not_started, count=0, html=True) self.assertContains(resp, finished, count=0, html=True) self.assertContains(resp, unfinished, count=0, html=True) def test_get_with_multiple_flow_sessions(self): factories.FlowSessionFactory( participation=self.student_participation, flow_id=QUIZ_FLOW_ID, in_progress=True) resp = self.get_gradebook_by_opp_view(self.gopp_id) self.assertEqual(resp.status_code, 200) def test_get_with_multiple_flow_sessions_view_page_grade(self): factories.FlowSessionFactory( participation=self.student_participation, flow_id=QUIZ_FLOW_ID, in_progress=True, page_count=12 ) resp = self.get_gradebook_by_opp_view(self.gopp_id, view_page_grades=True) self.assertEqual(resp.status_code, 200) def test_non_session_gopp(self): gopp = factories.GradingOpportunityFactory( course=self.course, identifier="another_gopp", flow_id=None) factories.GradeChangeFactory(**self.gc(opportunity=gopp)) resp = self.get_gradebook_by_opp_view(gopp.identifier, view_page_grades=True) self.assertEqual(resp.status_code, 200) resp = self.get_gradebook_by_opp_view(gopp.identifier) self.assertEqual(resp.status_code, 200) class GradesChangeStateMachineTest(GradesTestMixin, TestCase): def test_no_gradechange(self): # when no grade change object exists with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", None) self.assertResponseContextEqual(resp, "avg_grade_population", 0) def test_default_setup(self): self.use_default_setup() self.assertGradeChangeMachineReadableStateEqual(6) self.assertGradeChangeStateEqual("6.0% (/3)") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", 6) self.assertResponseContextEqual(resp, "avg_grade_population", 1) def test_change_aggregate_strategy_average(self): self.use_default_setup() self.update_gopp_strategy(g_strategy.avg_grade) self.assertGradeChangeMachineReadableStateEqual(4.333) self.assertGradeChangeStateEqual("4.3% (/3)") def test_change_aggregate_strategy_earliest(self): self.use_default_setup() self.update_gopp_strategy(g_strategy.use_earliest) self.assertGradeChangeMachineReadableStateEqual(0) self.assertGradeChangeStateEqual("0.0% (/3)") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", 0) self.assertResponseContextEqual(resp, "avg_grade_population", 1) def test_change_aggregate_strategy_max(self): self.use_default_setup() self.update_gopp_strategy(g_strategy.max_grade) self.assertGradeChangeMachineReadableStateEqual(7) self.assertGradeChangeStateEqual("7.0% (/3)") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", 7) self.assertResponseContextEqual(resp, "avg_grade_population", 1) def test_change_aggregate_strategy_max_none(self): # when no grade change has percentage self.update_gopp_strategy(g_strategy.max_grade) self.assertGradeChangeMachineReadableStateEqual("NONE") self.assertGradeChangeStateEqual("- ∅ -") factories.GradeChangeFactory.create(**(self.gc(points=None))) self.assertGradeChangeMachineReadableStateEqual("NONE") self.assertGradeChangeStateEqual("- ∅ -") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", None) self.assertResponseContextEqual(resp, "avg_grade_population", 0) def test_change_aggregate_strategy_min(self): self.use_default_setup() self.update_gopp_strategy(g_strategy.min_grade) self.assertGradeChangeMachineReadableStateEqual(0) self.assertGradeChangeStateEqual("0.0% (/3)") def test_change_aggregate_strategy_min_none(self): # when no grade change has percentage self.update_gopp_strategy(g_strategy.min_grade) self.assertGradeChangeMachineReadableStateEqual("NONE") self.assertGradeChangeStateEqual("- ∅ -") factories.GradeChangeFactory.create(**(self.gc(points=None))) self.assertGradeChangeMachineReadableStateEqual("NONE") self.assertGradeChangeStateEqual("- ∅ -") def test_change_aggregate_strategy_invalid(self): self.use_default_setup() self.update_gopp_strategy("invalid_strategy") with self.assertRaises(ValueError): self.get_gc_stringify_machine_readable_state() def test_average_grade_value(self): # Other tests for course.grades.average_grade self.use_default_setup() self.assertGradeChangeMachineReadableStateEqual(6) self.assertGradeChangeStateEqual("6.0% (/3)") # make sure participations with pperm.included_in_grade_statistics # are not included factories.GradeChangeFactory.create(**(self.gc( participation=self.instructor_participation, points=2))) factories.GradeChangeFactory.create(**(self.gc( participation=self.ta_participation, points=3))) with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", 6) self.assertResponseContextEqual(resp, "avg_grade_population", 1) temp_ptcp = factories.ParticipationFactory.create( course=self.course) factories.GradeChangeFactory.create( **(self.gc(participation=temp_ptcp, points=3))) with self.temporarily_switch_to_user(temp_ptcp.user): resp = self.get_view_single_grade(temp_ptcp, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", 4.5) self.assertResponseContextEqual(resp, "avg_grade_population", 2) def test_append_gc(self): self.use_default_setup() self.append_gc(self.gc(points=8, flow_session=self.session2)) self.assertGradeChangeMachineReadableStateEqual(8) self.assertGradeChangeStateEqual("8.0% (/3)") self.append_gc(self.gc(points=0, flow_session=self.session2)) self.assertGradeChangeMachineReadableStateEqual(0) self.assertGradeChangeStateEqual("0.0% (/3)") def test_update_latest_gc_of_latest_finished_session(self): self.use_default_setup() self.assertGradeChangeMachineReadableStateEqual(6) self.update_gc(self.gc_session2, points=10) self.assertGradeChangeMachineReadableStateEqual(10) self.assertGradeChangeStateEqual("10.0% (/3)") def test_update_earliest_gc_of_earlier_finished_session(self): self.use_default_setup() self.assertGradeChangeMachineReadableStateEqual(6) self.update_gc(self.gc_main_2, update_time=False, points=15) self.assertGradeChangeMachineReadableStateEqual(6) self.assertGradeChangeStateEqual("6.0% (/3)") def test_gc_without_attempt_id(self): # TODO: Is it a bug? percentage of GradeChanges without attempt_id are # put at the beginning of the valid_percentages list. # Uncomment the following to see the failure # self.use_default_setup() # self.assertGradeChangeMachineReadableStateEqual(6) # print(self.gc_main_1.grade_time) # # self.time_increment() # create a gc without attempt_id gc = factories.GradeChangeFactory.create( **(self.gc(points=8.5, null_attempt_id=True))) # print(gc.grade_time) machine = self.get_gc_machine() self.assertGradeChangeMachineReadableStateEqual(8.5) self.assertEqual(machine.valid_percentages, [8.5]) def test_gc_unavailable(self): factories.GradeChangeFactory.create(**(self.gc(points=9.1))) factories.GradeChangeFactory.create( **(self.gc(points=0, state=g_state.unavailable))) machine = self.get_gc_machine() self.assertGradeChangeMachineReadableStateEqual("OTHER_STATE") self.assertEqual(machine.valid_percentages, []) self.assertGradeChangeStateEqual("(other state)") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", None) self.assertResponseContextEqual(resp, "avg_grade_population", 0) # failure when unavailable gc follows another grade change factories.GradeChangeFactory.create(**(self.gc(points=5))) with self.assertRaises(ValueError) as e: self.get_gc_stringify_machine_readable_state() self.assertIn("cannot accept grade once opportunity has been " "marked 'unavailable'", e.exception) def test_gc_exempt(self): factories.GradeChangeFactory.create(**(self.gc(points=6))) factories.GradeChangeFactory.create( **(self.gc(points=0, state=g_state.exempt))) machine = self.get_gc_machine() self.assertGradeChangeMachineReadableStateEqual("EXEMPT") self.assertEqual(machine.valid_percentages, []) self.assertGradeChangeStateEqual("(exempt)") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", None) self.assertResponseContextEqual(resp, "avg_grade_population", 0) # failure when exempt gc follows another grade change factories.GradeChangeFactory.create(**(self.gc(points=5))) with self.assertRaises(ValueError) as e: self.get_gc_stringify_machine_readable_state() self.assertIn("cannot accept grade once opportunity has been " "marked 'exempt'", e.exception) def test_gc_do_over(self): factories.GradeChangeFactory.create(**(self.gc(points=6))) # This creates a GradeChange object with no attempt_id factories.GradeChangeFactory.create( **(self.gc(points=0, state=g_state.do_over, null_attempt_id=True))) machine = self.get_gc_machine() self.assertGradeChangeMachineReadableStateEqual("NONE") self.assertEqual(machine.valid_percentages, []) self.assertGradeChangeStateEqual("- ∅ -") # This make sure new grade change objects following do_over gc is # consumed without problem factories.GradeChangeFactory.create(**(self.gc(points=5))) self.assertGradeChangeMachineReadableStateEqual("5") machine = self.get_gc_machine() self.assertEqual(machine.valid_percentages, [5]) self.assertGradeChangeStateEqual("5.0%") def test_gc_do_over_average_grade_value(self): self.use_default_setup() factories.GradeChangeFactory.create( **(self.gc(points=None, state=g_state.do_over, flow_session=self.session2))) with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade(self.student_participation, self.gopp) self.assertResponseContextEqual(resp, "avg_grade_percentage", None) self.assertResponseContextEqual(resp, "avg_grade_population", 0) def test_gc_report_sent(self): factories.GradeChangeFactory.create(**(self.gc(points=6))) gc2 = factories.GradeChangeFactory.create( **(self.gc(points=0, state=g_state.report_sent))) machine = self.get_gc_machine() self.assertGradeChangeMachineReadableStateEqual("6") self.assertGradeChangeStateEqual("6.0%") self.assertEqual(machine.last_report_time, gc2.grade_time) def test_gc_extension(self): factories.GradeChangeFactory.create(**(self.gc(points=6))) gc2 = factories.GradeChangeFactory.create( **(self.gc(points=0, state=g_state.extension, due_time=self.time + timedelta(days=1)))) machine = self.get_gc_machine() self.assertGradeChangeMachineReadableStateEqual("6") self.assertGradeChangeStateEqual("6.0%") self.assertEqual(machine.due_time, gc2.due_time) def test_gc_grading_started(self): factories.GradeChangeFactory.create(**(self.gc(points=6))) factories.GradeChangeFactory.create( **(self.gc(points=0, state=g_state.grading_started))) self.assertGradeChangeMachineReadableStateEqual("6") self.assertGradeChangeStateEqual("6.0%") def test_gc_retrieved(self): factories.GradeChangeFactory.create(**(self.gc(points=6))) factories.GradeChangeFactory.create( **(self.gc(points=0, state=g_state.retrieved))) self.assertGradeChangeMachineReadableStateEqual("6") self.assertGradeChangeStateEqual("6.0%") def test_gc_non_exist_state(self): factories.GradeChangeFactory.create(**(self.gc(points=6))) factories.GradeChangeFactory.create( **(self.gc(points=0, state="some_state"))) with self.assertRaises(RuntimeError): self.get_gc_stringify_machine_readable_state() def test_gc_non_point(self): factories.GradeChangeFactory.create(**(self.gc(points=None))) self.assertGradeChangeMachineReadableStateEqual("NONE") self.assertGradeChangeStateEqual("- ∅ -") # }}} class ViewParticipantGradesTest2(GradesTestMixin, TestCase): def setUp(self): super().setUp() self.use_default_setup() self.gopp_hidden_in_gradebook = factories.GradingOpportunityFactory( course=self.course, aggregation_strategy=g_strategy.use_latest, flow_id=None, shown_in_grade_book=False, identifier="hidden_in_instructor_grade_book") self.gopp_hidden_in_gradebook = factories.GradingOpportunityFactory( course=self.course, aggregation_strategy=g_strategy.use_latest, flow_id=None, shown_in_grade_book=False, identifier="only_hidden_in_grade_book") self.gopp_hidden_in_participation_gradebook = ( factories.GradingOpportunityFactory( course=self.course, shown_in_participant_grade_book=False, aggregation_strategy=g_strategy.use_latest, flow_id=None, identifier="all_hidden_in_ptcp_gradebook")) self.gopp_result_hidden_in_participation_gradebook = ( factories.GradingOpportunityFactory( course=self.course, result_shown_in_participant_grade_book=False, aggregation_strategy=g_strategy.use_latest, flow_id=None, identifier="result_hidden_in_ptcp_gradebook")) self.gc_gopp_result_hidden = factories.GradeChangeFactory( **self.gc(points=66.67, opportunity=self.gopp_result_hidden_in_participation_gradebook, state=g_state.graded)) def test_view_my_grade(self): with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_my_grades() self.assertEqual(resp.status_code, 200) grade_table = self.get_response_context_value_by_name( resp, "grade_table") self.assertEqual((len(grade_table)), 2) self.assertEqual([g_info.opportunity.identifier for g_info in grade_table], [factories.DEFAULT_GRADE_IDENTIFIER, "result_hidden_in_ptcp_gradebook"]) # the grade is hidden self.assertNotContains(resp, 66.67) grade_participation = self.get_response_context_value_by_name( resp, "grade_participation") self.assertEqual(grade_participation.pk, self.student_participation.pk) # shown self.assertContains(resp, factories.DEFAULT_GRADE_IDENTIFIER) self.assertContains(resp, "result_hidden_in_ptcp_gradebook") # hidden self.assertNotContains(resp, "hidden_in_instructor_grade_book") self.assertNotContains(resp, "all_hidden_in_ptcp_gradebook") def test_view_participant_grades(self): with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.get_view_participant_grades(self.student_participation.id) self.assertEqual(resp.status_code, 200) grade_table = self.get_response_context_value_by_name( resp, "grade_table") self.assertEqual((len(grade_table)), 3) self.assertEqual([g_info.opportunity.identifier for g_info in grade_table], ["all_hidden_in_ptcp_gradebook", factories.DEFAULT_GRADE_IDENTIFIER, "result_hidden_in_ptcp_gradebook"]) # the grade hidden to participation is show to instructor # self.assertContains(resp, "66.67%(not released)") grade_participation = self.get_response_context_value_by_name( resp, "grade_participation") self.assertEqual(grade_participation.pk, self.student_participation.pk) # shown self.assertContains(resp, factories.DEFAULT_GRADE_IDENTIFIER) self.assertContains(resp, "result_hidden_in_ptcp_gradebook") self.assertContains(resp, "all_hidden_in_ptcp_gradebook") # hidden self.assertNotContains(resp, "hidden_in_instructor_grade_book") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_participant_grades( participation_id=self.instructor_participation.id) self.assertEqual(resp.status_code, 403) class ViewReopenSessionTest(GradesTestMixin, TestCase): # grades.view_reopen_session (currently for cases not covered by other tests) gopp_id = "la_quiz" def setUp(self): super().setUp() self.fs1 = factories.FlowSessionFactory( participation=self.student_participation, in_progress=False) self.fs2 = factories.FlowSessionFactory( participation=self.student_participation, in_progress=True) def test_flow_desc_not_exist(self): with mock.patch("course.content.get_flow_desc") as mock_get_flow_desc: from django.core.exceptions import ObjectDoesNotExist mock_get_flow_desc.side_effect = ObjectDoesNotExist resp = self.get_reopen_session_view( self.gopp_id, flow_session_id=self.fs1.pk) self.assertEqual(resp.status_code, 404) def test_already_in_progress(self): # not unsubmit, because we don't have previoius grade visit (which will # result in error) data = {"set_access_rules_tag": ["<<>>"], "comment": ["test reopen"], "reopen": ""} resp = self.post_reopen_session_view( self.gopp_id, flow_session_id=self.fs2.pk, data=data) self.assertEqual(resp.status_code, 200) self.assertAddMessageCallCount(1) self.assertAddMessageCalledWith( "Cannot reopen a session that's already in progress.") self.assertTrue(self.fs2.in_progress) def test_reopen_success(self): resp = self.get_reopen_session_view( self.gopp_id, flow_session_id=self.fs1.pk) self.assertEqual(resp.status_code, 200) # not unsubmit, because we don't have previoius grade visit (which will # result in error) data = {"set_access_rules_tag": ["<<>>"], "comment": ["test reopen"], "reopen": ""} resp = self.post_reopen_session_view( self.gopp_id, flow_session_id=self.fs1.pk, data=data) self.assertEqual(resp.status_code, 302) self.fs1.refresh_from_db() self.assertTrue(self.fs1.in_progress) def test_set_access_rule_tag(self): hacked_flow_desc = ( self.get_hacked_flow_desc_with_access_rule_tags(["blahblah"])) with mock.patch("course.content.get_flow_desc") as mock_get_flow_desc: mock_get_flow_desc.return_value = hacked_flow_desc # not unsubmit, because we don't have previoius grade visit (which will # result in error) data = {"set_access_rules_tag": ["blahblah"], "comment": ["test reopen"], "reopen": ""} resp = self.post_reopen_session_view( self.gopp_id, flow_session_id=self.fs1.pk, data=data) self.assertEqual(resp.status_code, 302) self.fs1.refresh_from_db() self.assertTrue(self.fs1.in_progress) self.assertEqual(self.fs1.access_rules_tag, "blahblah") class ViewSingleGradeTest(GradesTestMixin, TestCase): # grades.view_single_grade (currently for cases not covered by other tests) def setUp(self): super().setUp() fake_regrade_session = mock.patch("course.flow.regrade_session") self.mock_regrade_session = fake_regrade_session.start() self.addCleanup(fake_regrade_session.stop) fake_recalculate_session_grade = mock.patch( "course.flow.recalculate_session_grade") self.mock_recalculate_session_grade = fake_recalculate_session_grade.start() self.addCleanup(fake_recalculate_session_grade.stop) fake_expire_flow_session_standalone = mock.patch( "course.flow.expire_flow_session_standalone") self.mock_expire_flow_session_standalone = ( fake_expire_flow_session_standalone.start()) self.addCleanup(fake_expire_flow_session_standalone.stop) fake_finish_flow_session_standalone = mock.patch( "course.flow.finish_flow_session_standalone") self.mock_finish_flow_session_standalone = ( fake_finish_flow_session_standalone.start()) self.addCleanup(fake_finish_flow_session_standalone.stop) def test_participation_course_not_match(self): another_course_participation = factories.ParticipationFactory( course=factories.CourseFactory(identifier="another-course")) resp = self.get_view_single_grade(another_course_participation, self.gopp) self.assertEqual(resp.status_code, 400) def test_gopp_course_not_match(self): another_course_gopp = factories.GradingOpportunityFactory( course=factories.CourseFactory(identifier="another-course"), identifier=QUIZ_FLOW_ID) with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.client.get(self.get_single_grade_url( self.student_participation.pk, another_course_gopp.pk)) self.assertEqual(resp.status_code, 400) def test_view_other_single_grade_no_pperm(self): another_participation = factories.ParticipationFactory( course=self.course) with self.temporarily_switch_to_user(another_participation.user): resp = self.get_view_single_grade( self.student_participation, self.gopp, force_login_instructor=False) self.assertEqual(resp.status_code, 403) resp = self.post_view_single_grade( self.student_participation, self.gopp, data={}, force_login_instructor=False) self.assertEqual(resp.status_code, 403) def test_view_success(self): resp = self.get_view_single_grade( self.student_participation, self.gopp) self.assertEqual(resp.status_code, 200) def test_view_not_shown_in_grade_book(self): hidden_gopp = factories.GradingOpportunityFactory( course=self.course, identifier="hidden", shown_in_grade_book=False) resp = self.get_view_single_grade( self.student_participation, hidden_gopp) self.assertEqual(resp.status_code, 200) self.assertAddMessageCalledWith( "This grade is not shown in the grade book.") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade( self.student_participation, hidden_gopp, force_login_instructor=False) self.assertEqual(resp.status_code, 403) def test_view_not_shown_in_participant_grade_book(self): hidden_gopp = factories.GradingOpportunityFactory( course=self.course, identifier="hidden", shown_in_participant_grade_book=False) resp = self.get_view_single_grade( self.student_participation, hidden_gopp) self.assertEqual(resp.status_code, 200) self.assertAddMessageCalledWith( "This grade is not shown in the student grade book.") with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade( self.student_participation, hidden_gopp, force_login_instructor=False) self.assertEqual(resp.status_code, 403) def test_post_no_pperm(self): another_participation = factories.ParticipationFactory( course=self.course) # only view_gradebook pperm pp = models.ParticipationPermission( participation=another_participation, permission=pperm.view_gradebook) pp.save() fs = factories.FlowSessionFactory( participation=self.student_participation, flow_id=self.flow_id) for op in ["imposedl", "end", "regrade", "recalculate"]: with self.subTest(op=op): resp = self.post_view_single_grade( self.student_participation, self.gopp, data={"%s_%d" % (op, fs.pk): ""}, force_login_instructor=False) self.assertEqual(resp.status_code, 403) def test_post_no_action_match(self): resp = self.post_view_single_grade( self.student_participation, self.gopp, data={"blablabal": ""}) self.assertEqual(resp.status_code, 400) def test_post(self): fs = factories.FlowSessionFactory( participation=self.student_participation, flow_id=self.flow_id) tup = ( ("imposedl", self.mock_expire_flow_session_standalone, "Session deadline imposed."), ("end", self.mock_finish_flow_session_standalone, "Session ended."), ("regrade", self.mock_regrade_session, "Session regraded."), ("recalculate", self.mock_recalculate_session_grade, "Session grade recalculated.")) for op, mock_func, msg in tup: with self.subTest(op=op): resp = self.post_view_single_grade( self.student_participation, self.gopp, data={"%s_%d" % (op, fs.pk): ""}) self.assertEqual(resp.status_code, 200) self.assertEqual(mock_func.call_count, 1) self.assertAddMessageCalledWith(msg, reset=True) mock_func.reset_mock() def test_post_invalid_session_op(self): fs = factories.FlowSessionFactory( participation=self.student_participation, flow_id=self.flow_id) resp = self.post_view_single_grade( self.student_participation, self.gopp, data={"blablabal_%d" % fs.pk: ""}) self.assertEqual(resp.status_code, 400) def test_post_keyboard_interrupt(self): fs = factories.FlowSessionFactory( participation=self.student_participation, flow_id=self.flow_id) tup = ( ("imposedl", self.mock_expire_flow_session_standalone, "Session deadline imposed."), ("end", self.mock_finish_flow_session_standalone, "Session ended."), ("regrade", self.mock_regrade_session, "Session regraded."), ("recalculate", self.mock_recalculate_session_grade, "Session grade recalculated.")) err = "foo" self.mock_regrade_session.side_effect = KeyboardInterrupt(err) self.mock_recalculate_session_grade.side_effect = KeyboardInterrupt(err) self.mock_expire_flow_session_standalone.side_effect = KeyboardInterrupt(err) self.mock_finish_flow_session_standalone.side_effect = KeyboardInterrupt(err) for op, mock_func, msg in tup: with self.subTest(op=op): resp = self.post_view_single_grade( self.student_participation, self.gopp, data={"%s_%d" % (op, fs.pk): ""}) self.assertEqual(resp.status_code, 200) self.assertAddMessageNotCalledWith(msg, reset=False) self.assertAddMessageCalledWith( f"Error: KeyboardInterrupt {err}", reset=True) mock_func.reset_mock() def test_view_gopp_flow_desc_not_exist(self): with mock.patch("course.content.get_flow_desc") as mock_get_flow_desc: from django.core.exceptions import ObjectDoesNotExist mock_get_flow_desc.side_effect = ObjectDoesNotExist() resp = self.get_view_single_grade( self.student_participation, self.gopp) self.assertEqual(resp.status_code, 200) self.assertResponseContextIsNone( resp, "flow_sessions_and_session_properties") def test_view_gopp_no_flow_id(self): gopp = factories.GradingOpportunityFactory( course=self.course, identifier="no_flow_id", flow_id=None) factories.GradeChangeFactory( **self.gc( opportunity=gopp)) resp = self.get_view_single_grade( self.student_participation, gopp) self.assertEqual(resp.status_code, 200) self.assertResponseContextIsNone( resp, "flow_sessions_and_session_properties") def test_filter_out_pre_public_grade_changes(self): gopp = factories.GradingOpportunityFactory( course=self.course, identifier="no_flow_id", flow_id=None) # 5 gchanges factories.GradeChangeFactory(**self.gc( opportunity=gopp)) factories.GradeChangeFactory(**self.gc( opportunity=gopp)) factories.GradeChangeFactory(**self.gc( opportunity=gopp)) fourth_gc = factories.GradeChangeFactory(**self.gc( opportunity=gopp)) factories.GradeChangeFactory(**self.gc( opportunity=gopp)) resp = self.get_view_single_grade( self.student_participation, gopp) self.assertEqual(resp.status_code, 200) resp_gchanges = resp.context["grade_changes"] self.assertEqual(len(resp_gchanges), 5) # update_gopp gopp.hide_superseded_grade_history_before = ( fourth_gc.grade_time - timedelta(minutes=1)) gopp.save() # view by instructor resp = self.get_view_single_grade( self.student_participation, gopp) self.assertEqual(resp.status_code, 200) resp_gchanges = resp.context["grade_changes"] self.assertEqual(len(resp_gchanges), 5) # view by student with self.temporarily_switch_to_user(self.student_participation.user): resp = self.get_view_single_grade( self.student_participation, gopp, force_login_instructor=False) self.assertEqual(resp.status_code, 200) resp_gchanges = resp.context["grade_changes"] self.assertEqual(len(resp_gchanges), 2) class EditGradingOpportunityTest(GradesTestMixin, TestCase): # test grades.edit_grading_opportunity def get_edit_grading_opportunity_url(self, opp_id, course_identifier=None): course_identifier = course_identifier or self.get_default_course_identifier() kwargs = {"course_identifier": course_identifier, "opportunity_id": opp_id} return reverse("relate-edit_grading_opportunity", kwargs=kwargs) def get_edit_grading_opportunity_view(self, opp_id, course_identifier=None, force_login_instructor=True): course_identifier = course_identifier or self.get_default_course_identifier() if not force_login_instructor: user = self.get_logged_in_user() else: user = self.instructor_participation.user with self.temporarily_switch_to_user(user): return self.client.get( self.get_edit_grading_opportunity_url(opp_id, course_identifier)) def post_edit_grading_opportunity_view(self, opp_id, data, course_identifier=None, force_login_instructor=True): course_identifier = course_identifier or self.get_default_course_identifier() if not force_login_instructor: user = self.get_logged_in_user() else: user = self.instructor_participation.user with self.temporarily_switch_to_user(user): return self.client.post( self.get_edit_grading_opportunity_url(opp_id, course_identifier), data) def edit_grading_opportunity_post_data( self, name, identifier, page_scores_in_participant_gradebook=False, hide_superseded_grade_history_before=None, op="submit", shown_in_participant_grade_book=True, aggregation_strategy=constants.grade_aggregation_strategy.use_latest, shown_in_grade_book=True, result_shown_in_participant_grade_book=True, **kwargs): data = {"name": name, "identifier": identifier, op: "", "aggregation_strategy": aggregation_strategy} if page_scores_in_participant_gradebook: data["page_scores_in_participant_gradebook"] = "" if hide_superseded_grade_history_before: if isinstance(hide_superseded_grade_history_before, datetime.datetime): date_time_picker_time_format = "%Y-%m-%d %H:%M" hide_superseded_grade_history_before = ( hide_superseded_grade_history_before.strftime( date_time_picker_time_format)) data["hide_superseded_grade_history_before"] = ( hide_superseded_grade_history_before) if shown_in_participant_grade_book: data["shown_in_participant_grade_book"] = "" if shown_in_grade_book: data["shown_in_grade_book"] = "" if result_shown_in_participant_grade_book: data["result_shown_in_participant_grade_book"] = "" data.update(kwargs) return data def test_get_add_new(self): resp = self.get_edit_grading_opportunity_view(-1) self.assertEqual(resp.status_code, 200) def test_post_get_add_new(self): name = "my Gopp" identifier = "my_gopp" data = self.edit_grading_opportunity_post_data( name=name, identifier=identifier) resp = self.post_edit_grading_opportunity_view(-1, data=data) gopps = models.GradingOpportunity.objects.all() self.assertEqual(gopps.count(), 2) my_gopp = gopps.last() self.assertEqual(my_gopp.name, name) self.assertEqual(my_gopp.identifier, identifier) self.assertRedirects( resp, self.get_edit_grading_opportunity_url(my_gopp.pk), fetch_redirect_response=False) def test_course_not_match(self): another_course = factories.CourseFactory(identifier="another-course") another_course_gopp = factories.GradingOpportunityFactory( course=another_course) gopps = models.GradingOpportunity.objects.all() self.assertEqual(gopps.count(), 2) resp = self.get_edit_grading_opportunity_view( another_course_gopp.id, course_identifier=self.course.identifier) self.assertEqual(resp.status_code, 400) def test_view_edit_grading_opportunity(self): my_gopp = factories.GradingOpportunityFactory( course=self.course, identifier="another_gopp") data = self.edit_grading_opportunity_post_data( name=my_gopp.name, identifier=my_gopp.identifier, shown_in_grade_book=False) resp = self.post_edit_grading_opportunity_view(my_gopp.id, data=data) self.assertRedirects( resp, self.get_edit_grading_opportunity_url(my_gopp.pk), fetch_redirect_response=False) my_gopp.refresh_from_db() self.assertEqual(my_gopp.shown_in_grade_book, False) def test_view_edit_grading_opportunity_form_invalid(self): my_gopp = factories.GradingOpportunityFactory( course=self.course, identifier="another_gopp") data = self.edit_grading_opportunity_post_data( name=my_gopp.name, identifier=my_gopp.identifier, shown_in_grade_book=False) with mock.patch( "course.grades.EditGradingOpportunityForm.is_valid" ) as mock_form_is_valid: mock_form_is_valid.return_value = False resp = self.post_edit_grading_opportunity_view(my_gopp.id, data=data) self.assertEqual(resp.status_code, 200) my_gopp.refresh_from_db() self.assertEqual(my_gopp.shown_in_grade_book, True) class DownloadAllSubmissionsTest(SingleCourseQuizPageTestMixin, HackRepoMixin, TestCase): # test grades.download_all_submissions (for cases not covered by other tests) page_id = "half" my_access_rule_tag = "my_access_rule_tag" @classmethod def setUpTestData(cls): super().setUpTestData() # with this faked commit_sha, we may do multiple submissions cls.course.active_git_commit_sha = ( "my_fake_commit_sha_for_download_submissions") cls.course.save() client = Client() client.force_login(cls.student_participation.user) cls.start_flow(client, cls.flow_id) cls.submit_page_answer_by_page_id_and_test( client, cls.page_id, answer_data={"answer": 0.25}) cls.end_flow(client) fs = models.FlowSession.objects.first() fs.access_rules_tag = cls.my_access_rule_tag fs.save() cls.start_flow(client, cls.flow_id) cls.submit_page_answer_by_page_id_and_test(client, "proof_upload") cls.submit_page_answer_by_page_id_and_test(client, cls.page_id) cls.end_flow(client) # create an in_progress flow, with the same page submitted another_participation = factories.ParticipationFactory( course=cls.course) client.force_login(another_participation.user) cls.start_flow(client, cls.flow_id) cls.submit_page_answer_by_page_id_and_test(client, cls.page_id) # create a flow with no answers cls.start_flow(client, cls.flow_id) cls.end_flow(client) @property def group_page_id(self): _, group_id = self.get_page_ordinal_via_page_id( self.page_id, with_group_id=True) return f"{group_id}/{self.page_id}" def get_zip_file_buf_from_response(self, resp): return io.BytesIO(resp.content) def assertDownloadedFileZippedExtensionCount(self, resp, extensions, counts): # noqa assert isinstance(extensions, list) assert isinstance(counts, list) assert len(extensions) == len(counts) prefix, zip_file = resp["Content-Disposition"].split("=") self.assertEqual(prefix, "attachment; filename") self.assertEqual(resp.get("Content-Type"), "application/zip") buf = io.BytesIO(resp.content) import zipfile with zipfile.ZipFile(buf, "r") as zf: self.assertIsNone(zf.testzip()) for f in zf.filelist: self.assertTrue(f.file_size > 0) for i, ext in enumerate(extensions): self.assertEqual( len([f for f in zf.filelist if f.filename.endswith(ext)]), counts[i]) def test_no_rules_tag(self): hacked_flow_desc = self.get_hacked_flow_desc(del_rules=True) with mock.patch("course.content.get_flow_desc") as mock_get_flow_desc: mock_get_flow_desc.return_value = hacked_flow_desc with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.post_download_all_submissions_by_group_page_id( group_page_id=self.group_page_id, flow_id=self.flow_id) self.assertEqual(resp.status_code, 200) self.assertDownloadedFileZippedExtensionCount( resp, [".txt"], [1]) def test_download_first_attempt(self): with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.post_download_all_submissions_by_group_page_id( group_page_id=self.group_page_id, flow_id=self.flow_id, which_attempt="first") self.assertEqual(resp.status_code, 200) self.assertDownloadedFileZippedExtensionCount( resp, [".txt"], [1]) def test_download_all_attempts(self): with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.post_download_all_submissions_by_group_page_id( group_page_id=self.group_page_id, flow_id=self.flow_id, which_attempt="all") self.assertEqual(resp.status_code, 200) self.assertDownloadedFileZippedExtensionCount( resp, [".txt"], [2]) def test_download_include_feedback(self): with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.post_download_all_submissions_by_group_page_id( group_page_id=self.group_page_id, flow_id=self.flow_id, include_feedback=True) self.assertEqual(resp.status_code, 200) self.assertDownloadedFileZippedExtensionCount( resp, [".txt"], [2]) def test_download_include_feedback_no_feedback(self): with self.temporarily_switch_to_user(self.instructor_participation.user): another_group_page_id = ( self.group_page_id.replace(self.page_id, "proof_upload")) resp = self.post_download_all_submissions_by_group_page_id( group_page_id=another_group_page_id, flow_id=self.flow_id, include_feedback=True) self.assertEqual(resp.status_code, 200) self.assertDownloadedFileZippedExtensionCount( resp, [".pdf"], [1]) def test_download_include_extra_file(self): with self.temporarily_switch_to_user(self.instructor_participation.user): import os with open( os.path.join(os.path.dirname(__file__), "../resource", "test_file.pdf"), "rb") as extra_file: resp = self.post_download_all_submissions_by_group_page_id( group_page_id=self.group_page_id, flow_id=self.flow_id, extra_file=extra_file) self.assertEqual(resp.status_code, 200) self.assertDownloadedFileZippedExtensionCount( resp, [".txt", ".pdf"], [1, 1]) def test_download_in_progress(self): with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.post_download_all_submissions_by_group_page_id( group_page_id=self.group_page_id, flow_id=self.flow_id, non_in_progress_only=False) self.assertEqual(resp.status_code, 200) self.assertDownloadedFileZippedExtensionCount( resp, [".txt"], [2]) def test_download_other_access_rule_tags(self): hacked_flow_desc = ( self.get_hacked_flow_desc_with_access_rule_tags( [self.my_access_rule_tag, "blahblah"])) with mock.patch("course.content.get_flow_desc") as mock_get_flow_desc: mock_get_flow_desc.return_value = hacked_flow_desc with self.temporarily_switch_to_user(self.instructor_participation.user): resp = self.post_download_all_submissions_by_group_page_id( group_page_id=self.group_page_id, flow_id=self.flow_id, restrict_to_rules_tag=self.my_access_rule_tag) self.assertEqual(resp.status_code, 200) self.assertDownloadedFileZippedExtensionCount( resp, [".txt"], [1]) class PointsEqualTest(unittest.TestCase): # grades.points_equal def test(self): from decimal import Decimal self.assertTrue(grades.points_equal(None, None)) self.assertFalse(grades.points_equal(Decimal("1.11"), None)) self.assertFalse(grades.points_equal(None, Decimal("1.11"))) self.assertTrue(grades.points_equal(Decimal("1.11"), Decimal("1.11"))) self.assertFalse(grades.points_equal(Decimal("1.11"), Decimal("1.12"))) @unittest.SkipTest class FixingTest(GradesTestMixin, TestCase): # currently skipped def reopen_session1(self): existing_gc_count = models.GradeChange.objects.count() reopen_session(now_datetime=local_now(), session=self.session1, generate_grade_change=True, suppress_log=True) self.assertEqual(models.GradeChange.objects.count(), existing_gc_count+1) self.session1.refresh_from_db() def reopen_session2(self): existing_gc_count = models.GradeChange.objects.count() reopen_session(now_datetime=local_now(), session=self.session2, generate_grade_change=True, suppress_log=True) self.assertEqual(models.GradeChange.objects.count(), existing_gc_count+1) self.session2.refresh_from_db() def test_append_gc_with_session_after_reopen_session2(self): self.use_default_setup() self.reopen_session2() # append a grade change for session2 # grade_time need to be specified, because the faked gc # is using fake time, while reopen a session will create # an actual gc using the actual time. latest_gc = models.GradeChange.objects.all().order_by("-grade_time")[0] self.append_gc(self.gc(points=12, flow_session=self.session2, grade_time=now(), effective_time=latest_gc.effective_time)) self.assertGradeChangeMachineReadableStateEqual(12) self.assertGradeChangeStateEqual("12.00% (/3)") def test_append_nonsession_gc_after_reopen_session2(self): self.use_default_setup() self.reopen_session2() # Append a grade change without session # grade_time need to be specified, because the faked gc # is using fake time, while reopen a session will create # an actual gc using the actual time. self.append_gc(self.gc(points=11, grade_time=now())) self.assertGradeChangeMachineReadableStateEqual(11) self.assertGradeChangeStateEqual("11.00% (/3)") def test_new_gchange_created_when_finish_flow_use_last_has_activity(self): # With use_last_activity_as_completion_time = True, if a flow session HAS # last_activity, the expected effective_time of the new gchange should be # the last_activity() of the related flow_session. with self.temporarily_switch_to_user(self.instructor_participation.user): self.start_flow(QUIZ_FLOW_ID) # create a flow page visit, then there should be last_activity() for # the session. self.post_answer_by_ordinal(1, {"answer": ["0.5"]}) self.assertEqual( models.FlowPageVisit.objects.filter(answer__isnull=False).count(), 1) last_answered_visit = ( models.FlowPageVisit.objects.filter(answer__isnull=False).first()) last_answered_visit.visit_time = now() - timedelta(hours=1) last_answered_visit.save() self.assertEqual(models.GradeChange.objects.count(), 0) with mock.patch("course.flow.get_session_grading_rule") as \ mock_get_grading_rule: mock_get_grading_rule.side_effect = ( get_session_grading_rule_use_last_activity_as_cmplt_time_side_effect) resp = self.end_flow() self.assertEqual(resp.status_code, 200) self.assertEqual(models.GradeChange.objects.count(), 1) latest_gchange = models.GradeChange.objects.last() latest_flow_session = models.FlowSession.objects.last() self.assertIsNotNone(latest_flow_session.last_activity()) self.assertEqual(latest_flow_session.completion_time, latest_flow_session.last_activity()) self.assertEqual(latest_gchange.effective_time, latest_flow_session.last_activity()) # {{{ Fixed issue #263 and #417 def test_update_latest_gc_of_earlier_finished_session(self): self.use_default_setup() self.assertGradeChangeMachineReadableStateEqual(6) # Issue #263 and #417 # gc_session1 is the GradeChange object of session 1, update it's # value won't change the consumed state. self.update_gc(self.gc_session1, points=10) self.assertGradeChangeMachineReadableStateEqual(6) self.assertGradeChangeStateEqual("6.00% (/3)") def test_special_case(self): # https://github.com/inducer/relate/pull/423#discussion_r162121467 gc2015 = factories.GradeChangeFactory.create(**(self.gc(points=5))) session1 = factories.FlowSessionFactory.create( participation=self.student_participation, start_time=self.time-timedelta(days=17), completion_time=self.time-timedelta(days=14)) self.time_increment() gc2016 = factories.GradeChangeFactory.create( **(self.gc(points=0, flow_session=session1, grade_time=self.time))) gc2017 = factories.GradeChangeFactory.create(**(self.gc(points=7))) session2 = factories.FlowSessionFactory.create( participation=self.student_participation, start_time=self.time-timedelta(days=17), completion_time=self.time-timedelta(days=15)) self.time_increment() gc2018 = factories.GradeChangeFactory.create( **(self.gc(points=6, flow_session=session2))) assert models.GradingOpportunity.objects.count() == 1 assert models.GradeChange.objects.count() == 4 assert models.FlowSession.objects.count() == 2 self.assertTrue(session2.completion_time < session1.completion_time) self.assertTrue( gc2015.grade_time < gc2016.grade_time < gc2017.grade_time < gc2018.grade_time) self.assertGradeChangeMachineReadableStateEqual(gc2017.percentage()) # {{{ When two grade changes have the same grade_time # The expected behavior is GradeChange object with the larger pk # dominate. Fixed with #263 and #417 def test_gcs_have_same_grade_time1(self): gc1 = factories.GradeChangeFactory.create(**(self.gc(points=0))) session = factories.FlowSessionFactory.create( participation=self.student_participation, completion_time=gc1.grade_time-timedelta(days=1)) factories.GradeChangeFactory.create( **(self.gc(points=5, flow_session=session, grade_time=gc1.grade_time))) self.assertGradeChangeMachineReadableStateEqual(5) self.assertGradeChangeStateEqual("5.0% (/2)") def test_gc_have_same_grade_time2(self): session = factories.FlowSessionFactory.create( participation=self.student_participation, start_time=self.time-timedelta(days=1), completion_time=self.time) self.time_increment() gc1 = factories.GradeChangeFactory.create( **(self.gc(points=5, flow_session=session))) factories.GradeChangeFactory.create( **(self.gc(points=0, grade_time=gc1.grade_time))) self.assertGradeChangeMachineReadableStateEqual(0) self.assertGradeChangeStateEqual("0.00% (/2)") # }}} # {{{ Fix #430 def test_reopen_session2(self): self.use_default_setup() # original state self.assertGradeChangeMachineReadableStateEqual("6") n_gc = models.GradeChange.objects.count() self.reopen_session2() # A new GradeChange object is created, with state "do_over" expected_n_gc = models.GradeChange.objects.count() self.assertEqual(expected_n_gc, n_gc + 1) self.assertEqual( models.GradeChange.objects.order_by("grade_time").last().state, g_state.do_over) self.assertGradeChangeMachineReadableStateEqual("NONE") self.assertGradeChangeStateEqual("- ∅ - (/3)") def test_reopen_session_without_existing_gc(self): # This is rare, because a completed_session should had created # a GradeChange object. session_temp = factories.FlowSessionFactory.create( participation=self.student_participation, completion_time=self.time) existing_gc_count = models.GradeChange.objects.count() reopen_session(now_datetime=local_now(), session=session_temp, generate_grade_change=True, suppress_log=True) self.assertEqual(models.GradeChange.objects.count(), existing_gc_count) def test_reopen_session1(self): self.use_default_setup() self.assertGradeChangeMachineReadableStateEqual("6") n_gc = models.GradeChange.objects.count() self.reopen_session1() # A new GradeChange object is created, with state "do_over" expected_n_gc = models.GradeChange.objects.count() self.assertEqual(expected_n_gc, n_gc + 1) self.assertEqual( models.GradeChange.objects.order_by("grade_time").last().state, g_state.do_over) # session 1 is not the latest session self.assertGradeChangeMachineReadableStateEqual("6") self.assertGradeChangeStateEqual("6.00% (/3)") def _get_admin_flow_session_delete_url(self, args): return reverse("admin:course_flowsession_delete", args=args) def _delete_flow_session_admin(self, flow_session): exist_flow_session_count = models.FlowSession.objects.count() flow_session_delete_url = self._get_admin_flow_session_delete_url( args=(flow_session.id,)) delete_dict = {"post": "yes"} with self.temporarily_switch_to_user(self.superuser): resp = self.client.get(flow_session_delete_url) self.assertEqual(resp.status_code, 200) resp = self.client.post(flow_session_delete_url, data=delete_dict) self.assertEqual(resp.status_code, 302) self.assertEqual(exist_flow_session_count, models.FlowSession.objects.count() + 1) def test_delete_flow_session_admin_new_exempt_gradechange_created(self): self.use_default_setup() exist_grade_change_count = models.GradeChange.objects.count() # session1 has related grade changes, so a new grade change with 'exempt' is # created self._delete_flow_session_admin(self.session1) self.assertEqual(exist_grade_change_count + 1, models.GradeChange.objects.count()) last_gchange = ( models.GradeChange.objects .order_by("-grade_time").first()) self.assertIsNone(last_gchange.flow_session) self.assertEqual(last_gchange.state, g_state.exempt) def test_delete_flow_session_admin_no_new_gradechange_created(self): session_temp = factories.FlowSessionFactory.create( participation=self.student_participation, completion_time=self.time) exist_grade_change_count = models.GradeChange.objects.count() last_gchange_of_session_temp = ( models.GradeChange.objects .filter(flow_session=session_temp) .order_by("-grade_time")[:1]) self.assertEqual(last_gchange_of_session_temp.count(), 0) # session_temp has no related grade changes, so no new grade change # is created after deleted self._delete_flow_session_admin(session_temp) self.assertEqual(exist_grade_change_count, models.GradeChange.objects.count()) # }}} def test_backward_compatibility_merging_466(self): # this make sure after merging https://github.com/inducer/relate/pull/466 # gchanges are consumed without issue self.use_default_setup() self.gc_session2.effective_time = None self.gc_session2.save() self.gc_session2.refresh_from_db() # We are not using reopen_session(), because that will create new # gchange, which only happen after #466 was merged. self.session2.in_progress = True self.session2.save() self.session2.refresh_from_db() machine = self.get_gc_machine() # session2's gchange is excluded self.assertGradeChangeMachineReadableStateEqual(7) self.assertEqual(machine.valid_percentages, [0, 7]) self.assertGradeChangeStateEqual("7.00% (/2)") # {{{ test new gchange created when finishing flow def test_new_gchange_created_when_finish_flow_use_last_no_activity(self): # With use_last_activity_as_completion_time = True, if a flow session has # no last_activity, the expected effective_time of the new gchange should # be the completion time of the related flow_session. with self.temporarily_switch_to_user(self.student_participation.user): self.start_flow(QUIZ_FLOW_ID) self.assertEqual(models.GradeChange.objects.count(), 0) with mock.patch("course.flow.get_session_grading_rule") as \ mock_get_grading_rule: mock_get_grading_rule.side_effect = ( get_session_grading_rule_use_last_activity_as_cmplt_time_side_effect) resp = self.end_flow() self.assertEqual(resp.status_code, 200) self.assertEqual(models.GradeChange.objects.count(), 1) latest_gchange = models.GradeChange.objects.last() latest_flow_session = models.FlowSession.objects.last() self.assertIsNone(latest_flow_session.last_activity()) self.assertEqual(latest_gchange.effective_time, latest_flow_session.completion_time) def test_new_gchange_created_when_finish_flow_not_use_last_no_activity(self): # With use_last_activity_as_completion_time = False, if a flow session has # no last_activity, the expected effective_time of the new gchange should # be the completion time of the related flow_session. with self.temporarily_switch_to_user(self.student_participation.user): self.start_flow(QUIZ_FLOW_ID) self.assertEqual(models.GradeChange.objects.count(), 0) resp = self.end_flow() self.assertEqual(resp.status_code, 200) self.assertEqual(models.GradeChange.objects.count(), 1) latest_gchange = models.GradeChange.objects.last() latest_flow_session = models.FlowSession.objects.last() self.assertIsNone(latest_flow_session.last_activity()) self.assertEqual(latest_gchange.effective_time, latest_flow_session.completion_time) def test_new_gchange_created_when_finish_flow_not_use_last_has_activity(self): # With use_last_activity_as_completion_time = False, even if a flow session # HAS last_activity, the expected effective_time of the new gchange should # be the completion_time of the related flow_session. with self.temporarily_switch_to_user(self.instructor_participation.user): self.start_flow(QUIZ_FLOW_ID) # create a flow page visit, then there should be last_activity() for # the session. self.post_answer_by_ordinal(1, {"answer": ["0.5"]}) self.assertEqual( models.FlowPageVisit.objects.filter(answer__isnull=False).count(), 1) last_answered_visit = ( models.FlowPageVisit.objects.filter(answer__isnull=False).first()) last_answered_visit.visit_time = now() - timedelta(hours=1) last_answered_visit.save() self.assertEqual(models.GradeChange.objects.count(), 0) resp = self.end_flow() self.assertEqual(resp.status_code, 200) self.assertEqual(models.GradeChange.objects.count(), 1) latest_gchange = models.GradeChange.objects.last() latest_flow_session = models.FlowSession.objects.last() self.assertIsNotNone(latest_flow_session.last_activity()) self.assertNotEqual(latest_flow_session.completion_time, latest_flow_session.last_activity()) self.assertEqual(latest_gchange.effective_time, latest_flow_session.completion_time) # }}} # vim: fdm=marker