From fada7f903da587e8fcaf71f5d4a358a1a50f2c72 Mon Sep 17 00:00:00 2001 From: Zach Hancock Date: Wed, 4 Oct 2023 17:16:03 -0400 Subject: [PATCH 1/5] feat: exam reset producer --- edx_exams/apps/api/v1/views.py | 3 +- edx_exams/apps/core/api.py | 18 ++++++++ edx_exams/apps/core/signals/handlers.py | 15 +++++++ edx_exams/apps/core/signals/signals.py | 20 +++++++++ edx_exams/apps/core/tests/test_api.py | 59 ++++++++++++++++++++++++- 5 files changed, 113 insertions(+), 2 deletions(-) diff --git a/edx_exams/apps/api/v1/views.py b/edx_exams/apps/api/v1/views.py index ade39a7c..59c1fdbc 100644 --- a/edx_exams/apps/api/v1/views.py +++ b/edx_exams/apps/api/v1/views.py @@ -26,6 +26,7 @@ from edx_exams.apps.core.api import ( check_if_exam_timed_out, create_exam_attempt, + delete_exam_attempt, get_active_attempt_for_user, get_attempt_by_id, get_course_exams, @@ -626,7 +627,7 @@ def delete(self, request, attempt_id): error = {'detail': error_msg} return Response(status=status.HTTP_403_FORBIDDEN, data=error) - exam_attempt.delete() + delete_exam_attempt(exam_attempt, request.user) return Response(status=status.HTTP_204_NO_CONTENT) diff --git a/edx_exams/apps/core/api.py b/edx_exams/apps/core/api.py index b08a1e77..b0b10a86 100644 --- a/edx_exams/apps/core/api.py +++ b/edx_exams/apps/core/api.py @@ -20,6 +20,7 @@ from edx_exams.apps.core.signals.signals import ( emit_exam_attempt_errored_event, emit_exam_attempt_rejected_event, + emit_exam_attempt_reset_event, emit_exam_attempt_submitted_event, emit_exam_attempt_verified_event ) @@ -142,6 +143,23 @@ def update_attempt_status(attempt_id, to_status): return attempt_id +def delete_exam_attempt(attempt, requesting_user): + """ + Delete or 'reset' an exam attempt + """ + course_key = CourseKey.from_string(attempt.exam.course_id) + usage_key = UsageKey.from_string(attempt.exam.content_id) + + attempt.delete() + emit_exam_attempt_reset_event( + attempt.user, + course_key, + usage_key, + attempt.exam.exam_type, + requesting_user + ) + + def _allow_status_transition(attempt_obj, to_status): """ Helper method to assert that a given status transition is allowed diff --git a/edx_exams/apps/core/signals/handlers.py b/edx_exams/apps/core/signals/handlers.py index 962de7fc..5cce40e0 100644 --- a/edx_exams/apps/core/signals/handlers.py +++ b/edx_exams/apps/core/signals/handlers.py @@ -7,6 +7,7 @@ from openedx_events.learning.signals import ( EXAM_ATTEMPT_ERRORED, EXAM_ATTEMPT_REJECTED, + EXAM_ATTEMPT_RESET, EXAM_ATTEMPT_SUBMITTED, EXAM_ATTEMPT_VERIFIED ) @@ -68,3 +69,17 @@ def listen_for_exam_attempt_errored(sender, signal, **kwargs): # pylint: disabl event_data={'exam_attempt': kwargs['exam_attempt']}, event_metadata=kwargs['metadata'], ) + + +@receiver(EXAM_ATTEMPT_RESET) +def listen_for_exam_attempt_reset(sender, signal, **kwargs): # pylint: disable=unused-argument + """ + Publish EXAM_ATTEMPT_RESET signal onto the event bus + """ + get_producer().send( + signal=EXAM_ATTEMPT_RESET, + topic='exam-attempt-reset', + event_key_field='exam_attempt.course_key', + event_data={'exam_attempt': kwargs['exam_attempt']}, + event_metadata=kwargs['metadata'], + ) diff --git a/edx_exams/apps/core/signals/signals.py b/edx_exams/apps/core/signals/signals.py index 508da4b5..50f8747f 100644 --- a/edx_exams/apps/core/signals/signals.py +++ b/edx_exams/apps/core/signals/signals.py @@ -6,6 +6,7 @@ from openedx_events.learning.signals import ( EXAM_ATTEMPT_ERRORED, EXAM_ATTEMPT_REJECTED, + EXAM_ATTEMPT_RESET, EXAM_ATTEMPT_SUBMITTED, EXAM_ATTEMPT_VERIFIED ) @@ -95,3 +96,22 @@ def emit_exam_attempt_errored_event(user, course_key, usage_key, exam_type): exam_type=exam_type ) ) + + +def emit_exam_attempt_reset_event(user, course_key, usage_key, exam_type, requesting_user): + """ + Emit the EXAM_ATTEMPT_RESET Open edX event. + """ + user_data = _create_user_data(user) + requesting_user_data = _create_user_data(requesting_user) + + # .. event_implemented_name: EXAM_ATTEMPT_RESET + EXAM_ATTEMPT_RESET.send_event( + exam_attempt=ExamAttemptData( + student_user=user_data, + course_key=course_key, + usage_key=usage_key, + exam_type=exam_type, + requesting_user=requesting_user_data + ) + ) diff --git a/edx_exams/apps/core/tests/test_api.py b/edx_exams/apps/core/tests/test_api.py index c11dde76..0c5a4f85 100644 --- a/edx_exams/apps/core/tests/test_api.py +++ b/edx_exams/apps/core/tests/test_api.py @@ -17,6 +17,7 @@ from edx_exams.apps.core.api import ( check_if_exam_timed_out, create_exam_attempt, + delete_exam_attempt, get_active_attempt_for_user, get_attempt_by_id, get_attempt_for_user_with_attempt_number_and_resource_id, @@ -35,7 +36,7 @@ ) from edx_exams.apps.core.models import Exam, ExamAttempt from edx_exams.apps.core.statuses import ExamAttemptStatus -from edx_exams.apps.core.test_utils.factories import ExamAttemptFactory, ExamFactory +from edx_exams.apps.core.test_utils.factories import ExamAttemptFactory, ExamFactory, UserFactory test_start_time = datetime(2023, 11, 4, 11, 5, 23) test_time_limit_mins = 30 @@ -623,6 +624,62 @@ def test_exam_with_no_due_date(self): self.assertIsNotNone(ExamAttempt.objects.get(user_id=user_id, exam_id=exam_id)) +class TestDeleteExamAttempt(ExamsAPITestCase): + """ + Tests for the API utility function `delete_exam_attempt` + """ + def setUp(self): + super().setUp() + + self.exam = ExamFactory() + self.student_user = UserFactory() + self.exam_attempt = ExamAttemptFactory(user=self.student_user, exam=self.exam) + + def test_delete_exam_attempt(self): + """ + Test that an exam attempt is deleted + """ + delete_exam_attempt(self.exam_attempt, self.user) + self.assertFalse(ExamAttempt.objects.filter(id=self.exam_attempt.id).exists()) + + @patch('edx_exams.apps.core.signals.signals.EXAM_ATTEMPT_RESET.send_event') + def test_event_emitted(self, mock_event_send): + """ + Test that when an exam attempt is deleted, the EXAM_ATTEMPT_RESET event is emitted. + """ + delete_exam_attempt(self.exam_attempt, self.user) + + user_data = UserData( + id=self.student_user.id, + is_active=self.student_user.is_active, + pii=UserPersonalData( + username=self.student_user.username, + email=self.student_user.email, + name=self.student_user.full_name + ) + ) + requesting_user_data = UserData( + id=self.user.id, + is_active=self.user.is_active, + pii=UserPersonalData( + username=self.user.username, + email=self.user.email, + name=self.user.full_name + ) + ) + course_key = CourseKey.from_string(self.exam.course_id) + usage_key = UsageKey.from_string(self.exam.content_id) + + expected_data = ExamAttemptData( + student_user=user_data, + course_key=course_key, + usage_key=usage_key, + exam_type=self.exam.exam_type, + requesting_user=requesting_user_data, + ) + mock_event_send.assert_called_once_with(exam_attempt=expected_data) + + class TestGetExamByContentId(ExamsAPITestCase): """ Tests for the API utility function `get_exam_by_content_id` From 00891d6434f48a1aaeee7fdb216d6b7b5972997c Mon Sep 17 00:00:00 2001 From: Zach Hancock Date: Thu, 5 Oct 2023 13:35:22 -0400 Subject: [PATCH 2/5] style: better naming --- edx_exams/apps/api/v1/views.py | 4 ++-- edx_exams/apps/core/api.py | 4 ++-- edx_exams/apps/core/tests/test_api.py | 10 +++++----- 3 files changed, 9 insertions(+), 9 deletions(-) diff --git a/edx_exams/apps/api/v1/views.py b/edx_exams/apps/api/v1/views.py index 59c1fdbc..5b10979f 100644 --- a/edx_exams/apps/api/v1/views.py +++ b/edx_exams/apps/api/v1/views.py @@ -26,7 +26,6 @@ from edx_exams.apps.core.api import ( check_if_exam_timed_out, create_exam_attempt, - delete_exam_attempt, get_active_attempt_for_user, get_attempt_by_id, get_course_exams, @@ -37,6 +36,7 @@ get_exam_by_id, get_provider_by_exam_id, is_exam_passed_due, + reset_exam_attempt, update_attempt_status ) from edx_exams.apps.core.exam_types import get_exam_type @@ -627,7 +627,7 @@ def delete(self, request, attempt_id): error = {'detail': error_msg} return Response(status=status.HTTP_403_FORBIDDEN, data=error) - delete_exam_attempt(exam_attempt, request.user) + reset_exam_attempt(exam_attempt, request.user) return Response(status=status.HTTP_204_NO_CONTENT) diff --git a/edx_exams/apps/core/api.py b/edx_exams/apps/core/api.py index b0b10a86..84a66155 100644 --- a/edx_exams/apps/core/api.py +++ b/edx_exams/apps/core/api.py @@ -143,9 +143,9 @@ def update_attempt_status(attempt_id, to_status): return attempt_id -def delete_exam_attempt(attempt, requesting_user): +def reset_exam_attempt(attempt, requesting_user): """ - Delete or 'reset' an exam attempt + Reset an exam attempt """ course_key = CourseKey.from_string(attempt.exam.course_id) usage_key = UsageKey.from_string(attempt.exam.content_id) diff --git a/edx_exams/apps/core/tests/test_api.py b/edx_exams/apps/core/tests/test_api.py index 0c5a4f85..d5155fd2 100644 --- a/edx_exams/apps/core/tests/test_api.py +++ b/edx_exams/apps/core/tests/test_api.py @@ -17,7 +17,6 @@ from edx_exams.apps.core.api import ( check_if_exam_timed_out, create_exam_attempt, - delete_exam_attempt, get_active_attempt_for_user, get_attempt_by_id, get_attempt_for_user_with_attempt_number_and_resource_id, @@ -26,6 +25,7 @@ get_exam_by_content_id, get_exam_url_path, is_exam_passed_due, + reset_exam_attempt, update_attempt_status ) from edx_exams.apps.core.exceptions import ( @@ -624,9 +624,9 @@ def test_exam_with_no_due_date(self): self.assertIsNotNone(ExamAttempt.objects.get(user_id=user_id, exam_id=exam_id)) -class TestDeleteExamAttempt(ExamsAPITestCase): +class TestResetExamAttempt(ExamsAPITestCase): """ - Tests for the API utility function `delete_exam_attempt` + Tests for the API utility function `reset_exam_attempt` """ def setUp(self): super().setUp() @@ -635,7 +635,7 @@ def setUp(self): self.student_user = UserFactory() self.exam_attempt = ExamAttemptFactory(user=self.student_user, exam=self.exam) - def test_delete_exam_attempt(self): + def test_reset_exam_attempt(self): """ Test that an exam attempt is deleted """ @@ -645,7 +645,7 @@ def test_delete_exam_attempt(self): @patch('edx_exams.apps.core.signals.signals.EXAM_ATTEMPT_RESET.send_event') def test_event_emitted(self, mock_event_send): """ - Test that when an exam attempt is deleted, the EXAM_ATTEMPT_RESET event is emitted. + Test that when an exam attempt is reset, the EXAM_ATTEMPT_RESET event is emitted. """ delete_exam_attempt(self.exam_attempt, self.user) From fd0481803a21d11ebb45fad00f3aeec0ba9fb55d Mon Sep 17 00:00:00 2001 From: Zach Hancock Date: Thu, 5 Oct 2023 13:36:56 -0400 Subject: [PATCH 3/5] feat: update reset topic --- edx_exams/apps/core/signals/handlers.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/edx_exams/apps/core/signals/handlers.py b/edx_exams/apps/core/signals/handlers.py index 5cce40e0..f8573180 100644 --- a/edx_exams/apps/core/signals/handlers.py +++ b/edx_exams/apps/core/signals/handlers.py @@ -78,7 +78,7 @@ def listen_for_exam_attempt_reset(sender, signal, **kwargs): # pylint: disable= """ get_producer().send( signal=EXAM_ATTEMPT_RESET, - topic='exam-attempt-reset', + topic=topic_name, event_key_field='exam_attempt.course_key', event_data={'exam_attempt': kwargs['exam_attempt']}, event_metadata=kwargs['metadata'], From a401e3798072a255c2e0fcf5348fc45ab0b97aba Mon Sep 17 00:00:00 2001 From: Zach Hancock Date: Thu, 5 Oct 2023 14:12:58 -0400 Subject: [PATCH 4/5] test: missed a spot --- edx_exams/apps/core/tests/test_api.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/edx_exams/apps/core/tests/test_api.py b/edx_exams/apps/core/tests/test_api.py index d5155fd2..5d75f197 100644 --- a/edx_exams/apps/core/tests/test_api.py +++ b/edx_exams/apps/core/tests/test_api.py @@ -639,7 +639,7 @@ def test_reset_exam_attempt(self): """ Test that an exam attempt is deleted """ - delete_exam_attempt(self.exam_attempt, self.user) + reset_exam_attempt(self.exam_attempt, self.user) self.assertFalse(ExamAttempt.objects.filter(id=self.exam_attempt.id).exists()) @patch('edx_exams.apps.core.signals.signals.EXAM_ATTEMPT_RESET.send_event') @@ -647,7 +647,7 @@ def test_event_emitted(self, mock_event_send): """ Test that when an exam attempt is reset, the EXAM_ATTEMPT_RESET event is emitted. """ - delete_exam_attempt(self.exam_attempt, self.user) + reset_exam_attempt(self.exam_attempt, self.user) user_data = UserData( id=self.student_user.id, From 2375c54e5d97d6076f54b41bf97adb4bb6fc2d45 Mon Sep 17 00:00:00 2001 From: Zach Hancock Date: Thu, 5 Oct 2023 17:06:31 -0400 Subject: [PATCH 5/5] feat: logs --- edx_exams/apps/core/api.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/edx_exams/apps/core/api.py b/edx_exams/apps/core/api.py index 84a66155..6fa04d0f 100644 --- a/edx_exams/apps/core/api.py +++ b/edx_exams/apps/core/api.py @@ -150,6 +150,10 @@ def reset_exam_attempt(attempt, requesting_user): course_key = CourseKey.from_string(attempt.exam.course_id) usage_key = UsageKey.from_string(attempt.exam.content_id) + log.info( + f'Resetting exam attempt for user_id={attempt.user.id} in exam={attempt.exam.id} ' + ) + attempt.delete() emit_exam_attempt_reset_event( attempt.user,