From d17d4bcec131d42652db77e90aae84dc8f096eff Mon Sep 17 00:00:00 2001 From: Paul Abumov Date: Mon, 14 Aug 2023 17:10:34 -0400 Subject: [PATCH 1/2] Added unittests to cover prolific_utils module --- .../providers/prolific/prolific_utils.py | 2 + .../providers/prolific/test_prolific_utils.py | 601 ++++++++++++++++++ 2 files changed, 603 insertions(+) diff --git a/mephisto/abstractions/providers/prolific/prolific_utils.py b/mephisto/abstractions/providers/prolific/prolific_utils.py index a1e052c95..2b3bf00bf 100644 --- a/mephisto/abstractions/providers/prolific/prolific_utils.py +++ b/mephisto/abstractions/providers/prolific/prolific_utils.py @@ -541,6 +541,8 @@ def add_workers_to_qualification( ) raise + return None + def give_worker_qualification( client: ProlificClient, diff --git a/test/abstractions/providers/prolific/test_prolific_utils.py b/test/abstractions/providers/prolific/test_prolific_utils.py index 93c52b1cd..1650019e7 100644 --- a/test/abstractions/providers/prolific/test_prolific_utils.py +++ b/test/abstractions/providers/prolific/test_prolific_utils.py @@ -16,9 +16,13 @@ from omegaconf import DictConfig from mephisto.abstractions.providers.prolific.api import constants +from mephisto.abstractions.providers.prolific.api.data_models import BonusPayments +from mephisto.abstractions.providers.prolific.api.data_models import ListSubmission +from mephisto.abstractions.providers.prolific.api.data_models import Participant from mephisto.abstractions.providers.prolific.api.data_models import ParticipantGroup from mephisto.abstractions.providers.prolific.api.data_models import Project from mephisto.abstractions.providers.prolific.api.data_models import Study +from mephisto.abstractions.providers.prolific.api.data_models import Submission from mephisto.abstractions.providers.prolific.api.data_models import User from mephisto.abstractions.providers.prolific.api.data_models import Workspace from mephisto.abstractions.providers.prolific.api.data_models import WorkspaceBalance @@ -30,8 +34,14 @@ from mephisto.abstractions.providers.prolific.prolific_utils import _find_prolific_project from mephisto.abstractions.providers.prolific.prolific_utils import _find_prolific_workspace from mephisto.abstractions.providers.prolific.prolific_utils import _find_qualification +from mephisto.abstractions.providers.prolific.prolific_utils import _find_submission +from mephisto.abstractions.providers.prolific.prolific_utils import _get_block_list_qualification from mephisto.abstractions.providers.prolific.prolific_utils import _get_external_study_url from mephisto.abstractions.providers.prolific.prolific_utils import _is_ec2_architect +from mephisto.abstractions.providers.prolific.prolific_utils import add_workers_to_qualification +from mephisto.abstractions.providers.prolific.prolific_utils import approve_work +from mephisto.abstractions.providers.prolific.prolific_utils import block_worker +from mephisto.abstractions.providers.prolific.prolific_utils import calculate_pay_amount from mephisto.abstractions.providers.prolific.prolific_utils import check_balance from mephisto.abstractions.providers.prolific.prolific_utils import check_credentials from mephisto.abstractions.providers.prolific.prolific_utils import create_qualification @@ -45,13 +55,20 @@ from mephisto.abstractions.providers.prolific.prolific_utils import find_or_create_qualification from mephisto.abstractions.providers.prolific.prolific_utils import get_authenticated_client from mephisto.abstractions.providers.prolific.prolific_utils import get_study +from mephisto.abstractions.providers.prolific.prolific_utils import get_submission +from mephisto.abstractions.providers.prolific.prolific_utils import give_worker_qualification from mephisto.abstractions.providers.prolific.prolific_utils import ( increase_total_available_places_for_study, ) from mephisto.abstractions.providers.prolific.prolific_utils import is_study_expired +from mephisto.abstractions.providers.prolific.prolific_utils import is_worker_blocked +from mephisto.abstractions.providers.prolific.prolific_utils import pay_bonus from mephisto.abstractions.providers.prolific.prolific_utils import publish_study +from mephisto.abstractions.providers.prolific.prolific_utils import reject_work +from mephisto.abstractions.providers.prolific.prolific_utils import remove_worker_qualification from mephisto.abstractions.providers.prolific.prolific_utils import setup_credentials from mephisto.abstractions.providers.prolific.prolific_utils import stop_study +from mephisto.abstractions.providers.prolific.prolific_utils import unblock_worker from mephisto.data_model.requester import RequesterArgs MOCK_PROLIFIC_CONFIG_DIR = "/tmp/" @@ -942,6 +959,590 @@ def test_is_study_expired(self, *args): self.assertFalse(result_just_with_awaiting_review_status) self.assertTrue(result_with_completed_status_and_internal_name) + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.add_participants_to_group' + ) + def test_add_workers_to_qualification_success(self, *args): + worker_ids = ['test', 'test2'] + participant_group_id = 'test3' + + result = add_workers_to_qualification(self.client, worker_ids, participant_group_id) + + self.assertIsNone(result) + + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.add_participants_to_group' + ) + def test_add_workers_to_qualification_exception(self, mock_add_participants_to_group, *args): + worker_ids = ['test', 'test2'] + participant_group_id = 'test3' + + exception_message = 'Error' + mock_add_participants_to_group.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + add_workers_to_qualification(self.client, worker_ids, participant_group_id) + + self.assertEqual(cm.exception.message, exception_message) + + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.add_participants_to_group' + ) + def test_give_worker_qualification_success(self, mock_add_participants_to_group, *args): + worker_id = 'test' + participant_group_id = 'test3' + + result = give_worker_qualification(self.client, worker_id, participant_group_id) + + mock_add_participants_to_group.assert_called_once_with( + id=participant_group_id, participant_ids=[worker_id], + ) + self.assertIsNone(result) + + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.remove_participants_from_group' + ) + def test_remove_worker_qualification_success(self, *args): + worker_id = 'test' + participant_group_id = 'test2' + + result = remove_worker_qualification(self.client, worker_id, participant_group_id) + + self.assertIsNone(result) + + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.remove_participants_from_group' + ) + def test_remove_worker_qualification_exception( + self, mock_remove_participants_from_group, *args, + ): + worker_id = 'test' + participant_group_id = 'test2' + + exception_message = 'Error' + mock_remove_participants_from_group.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + remove_worker_qualification(self.client, worker_id, participant_group_id) + + self.assertEqual(cm.exception.message, exception_message) + + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') + @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + def test_pay_bonus_empty_balance(self, mock_check_balance, mock_set_up, mock_pay, *args): + worker_id = 'test' + study_id = 'test2' + bonus_amount = 1000 + + mock_check_balance.return_value = False + + result = pay_bonus(self.client, mock_task_run_args, worker_id, bonus_amount, study_id) + + mock_set_up.assert_not_called() + mock_pay.assert_not_called() + self.assertFalse(result) + + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') + @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + def test_pay_bonus_empty_balance(self, mock_check_balance, mock_set_up, mock_pay, *args): + worker_id = 'test' + study_id = 'test2' + bonus_amount = 1000 + + mock_check_balance.return_value = False + + result = pay_bonus(self.client, mock_task_run_args, worker_id, bonus_amount, study_id) + + mock_set_up.assert_not_called() + mock_pay.assert_not_called() + self.assertFalse(result) + + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') + @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + def test_pay_bonus_success(self, mock_check_balance, mock_set_up, mock_pay, *args): + worker_id = 'test' + study_id = 'test2' + bonus_payments_id = 'test3' + bonus_amount = 1000 + + mock_bonus_payments = BonusPayments() + mock_bonus_payments.id = bonus_payments_id + mock_check_balance.return_value = True + mock_set_up.return_value = mock_bonus_payments + + result = pay_bonus(self.client, mock_task_run_args, worker_id, bonus_amount, study_id) + + mock_set_up.assert_called_once_with(study_id, f'{worker_id},{bonus_amount / 100}') + mock_pay.assert_called_once_with(mock_bonus_payments.id) + self.assertTrue(result) + + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') + @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + def test_pay_bonus_set_up_exception(self, mock_check_balance, mock_set_up, mock_pay, *args): + worker_id = 'test' + study_id = 'test2' + bonus_amount = 1000 + + mock_check_balance.return_value = True + + exception_message = 'Error' + mock_set_up.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + pay_bonus(self.client, mock_task_run_args, worker_id, bonus_amount, study_id) + + self.assertEqual(cm.exception.message, exception_message) + mock_set_up.assert_called_once_with(study_id, f'{worker_id},{bonus_amount / 100}') + mock_pay.assert_not_called() + + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') + @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') + @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + def test_pay_bonus_pay_exception(self, mock_check_balance, mock_set_up, mock_pay, *args): + worker_id = 'test' + study_id = 'test2' + bonus_payments_id = 'test3' + bonus_amount = 1000 + + mock_bonus_payments = BonusPayments() + mock_bonus_payments.id = bonus_payments_id + mock_check_balance.return_value = True + mock_set_up.return_value = mock_bonus_payments + + exception_message = 'Error' + mock_pay.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + pay_bonus(self.client, mock_task_run_args, worker_id, bonus_amount, study_id) + + self.assertEqual(cm.exception.message, exception_message) + mock_set_up.assert_called_once_with(study_id, f'{worker_id},{bonus_amount / 100}') + mock_pay.assert_called_once_with(mock_bonus_payments.id) + + @patch('mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_qualification') + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' + ) + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' + ) + def test__get_block_list_qualification( + self, + mock_find_or_create_prolific_workspace, + mock_find_or_create_prolific_project, + mock_find_or_create_qualification, + *args, + ): + mock_workspace = Workspace() + mock_workspace.id = 'test' + mock_project = Project() + mock_project.id = 'test2' + mock_project.title = 'test2_title' + mock_participant_group = ParticipantGroup() + mock_participant_group.id = 'test3' + mock_participant_group.name = 'test3_name' + + mock_find_or_create_prolific_workspace.return_value = mock_workspace + mock_find_or_create_prolific_project.return_value = mock_project + mock_find_or_create_qualification.return_value = mock_participant_group + + result = _get_block_list_qualification(self.client, mock_task_run_args) + + self.assertEqual(mock_participant_group, result) + + @patch('mephisto.abstractions.providers.prolific.prolific_utils.add_workers_to_qualification') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._get_block_list_qualification') + def test_block_worker_success( + self, mock__get_block_list_qualification, mock_add_workers_to_qualification, *args, + ): + worker_id = 'test' + + mock_participant_group = ParticipantGroup() + mock_participant_group.id = 'test2' + + mock__get_block_list_qualification.return_value = mock_participant_group + + result = block_worker(self.client, mock_task_run_args, worker_id) + + self.assertIsNone(result) + mock_add_workers_to_qualification.called_once_with( + self.client, workers_ids=[worker_id], qualification_id=mock_participant_group.id, + ) + + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.remove_participants_from_group' + ) + @patch('mephisto.abstractions.providers.prolific.prolific_utils._get_block_list_qualification') + def test_unblock_worker_success( + self, mock__get_block_list_qualification, mock_remove_participants_from_group, *args, + ): + worker_id = 'test' + + mock_participant_group = ParticipantGroup() + mock_participant_group.id = 'test2' + + mock__get_block_list_qualification.return_value = mock_participant_group + + result = unblock_worker(self.client, mock_task_run_args, worker_id) + + self.assertIsNone(result) + mock_remove_participants_from_group.called_once_with( + id=mock_participant_group.id, participant_ids=[worker_id], + ) + + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' + ) + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' + ) + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.list_participants_for_group' + ) + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_qualification') + def test_is_worker_blocked_no_block_list_qualification( + self, mock__find_qualification, mock_list_participants_for_group, *args, + ): + worker_id = 'test' + + mock__find_qualification.return_value = (False, None) + + result = is_worker_blocked(self.client, mock_task_run_args, worker_id) + + self.assertFalse(result) + mock_list_participants_for_group.assert_not_called() + + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' + ) + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' + ) + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.list_participants_for_group' + ) + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_qualification') + def test_is_worker_blocked_success_true( + self, mock__find_qualification, mock_list_participants_for_group, *args, + ): + worker_id = 'test' + + mock_participant_group = ParticipantGroup() + mock_participant_group.id = 'test2' + mock_participant = Participant() + mock_participant.participant_id = worker_id + + mock__find_qualification.return_value = (True, mock_participant_group) + mock_list_participants_for_group.return_value = [mock_participant] + + result = is_worker_blocked(self.client, mock_task_run_args, worker_id) + + self.assertTrue(result) + mock_list_participants_for_group.assert_called_once() + + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' + ) + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' + ) + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.list_participants_for_group' + ) + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_qualification') + def test_is_worker_blocked_success_false( + self, mock__find_qualification, mock_list_participants_for_group, *args, + ): + worker_id = 'test' + + mock_participant_group = ParticipantGroup() + mock_participant_group.id = 'test2' + mock_participant = Participant() + mock_participant.participant_id = 'test3' + + mock__find_qualification.return_value = (True, mock_participant_group) + mock_list_participants_for_group.return_value = [mock_participant] + + result = is_worker_blocked(self.client, mock_task_run_args, worker_id) + + self.assertFalse(result) + mock_list_participants_for_group.assert_called_once() + + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' + ) + @patch( + 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' + ) + @patch( + 'mephisto.abstractions.providers.prolific.api.' + 'participant_groups.ParticipantGroups.list_participants_for_group' + ) + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_qualification') + def test_is_worker_blocked_exception( + self, mock__find_qualification, mock_list_participants_for_group, *args, + ): + worker_id = 'test' + + mock_participant_group = ParticipantGroup() + mock_participant_group.id = 'test2' + mock_participant = Participant() + mock_participant.participant_id = worker_id + + mock__find_qualification.return_value = (True, mock_participant_group) + + exception_message = 'Error' + mock_list_participants_for_group.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + is_worker_blocked(self.client, mock_task_run_args, worker_id) + + self.assertEqual(cm.exception.message, exception_message) + mock_list_participants_for_group.assert_called_once() + + @patch('mephisto.abstractions.providers.prolific.api.studies.Studies.calculate_cost') + def test_calculate_pay_amount_success(self, mock_calculate_cost, *args): + task_amount = 1000 + total_available_places = 2 + + mock_calculate_cost.return_value = task_amount * total_available_places + + result = calculate_pay_amount(self.client, task_amount, total_available_places) + + self.assertEqual(task_amount * total_available_places, result) + + @patch('mephisto.abstractions.providers.prolific.api.studies.Studies.calculate_cost') + def test_calculate_pay_amount_exception(self, mock_calculate_cost, *args): + task_amount = 1000 + total_available_places = 2 + + exception_message = 'Error' + mock_calculate_cost.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + calculate_pay_amount(self.client, task_amount, total_available_places) + + self.assertEqual(cm.exception.message, exception_message) + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.list') + def test__find_submission_success_found(self, mock_list, *args): + worker_id = 'test' + study_id = 'test2' + + mock_list_submission = ListSubmission() + mock_list_submission.id = 'test3' + mock_list_submission.participant_id = worker_id + + mock_list.return_value = [mock_list_submission] + + result = _find_submission(self.client, study_id, worker_id) + + self.assertEqual(mock_list_submission, result) + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.list') + def test__find_submission_success_not_found(self, mock_list, *args): + worker_id = 'test' + study_id = 'test2' + + mock_list_submission = ListSubmission() + mock_list_submission.id = 'test3' + mock_list_submission.participant_id = 'test4' + + mock_list.return_value = [mock_list_submission] + + result = _find_submission(self.client, study_id, worker_id) + + self.assertEqual(None, result) + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.list') + def test__find_submission_exception(self, mock_list, *args): + worker_id = 'test' + study_id = 'test2' + + exception_message = 'Error' + mock_list.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + _find_submission(self.client, study_id, worker_id) + + self.assertEqual(cm.exception.message, exception_message) + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.retrieve') + def test_get_submission_success(self, mock_retrieve, *args): + submission_id = 'test' + + mock_submission = Submission() + mock_submission.id = 'test2' + + mock_retrieve.return_value = mock_submission + + result = get_submission(self.client, submission_id) + + self.assertEqual(mock_submission, result) + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.retrieve') + def test_get_submission_exception(self, mock_retrieve, *args): + submission_id = 'test' + + exception_message = 'Error' + mock_retrieve.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + get_submission(self.client, submission_id) + + self.assertEqual(cm.exception.message, exception_message) + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.approve') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + def test_approve_work_submission_not_found(self, mock__find_submission, mock_approve, *args): + worker_id = 'test' + study_id = 'test2' + + mock__find_submission.return_value = False + + result = approve_work(self.client, study_id, worker_id) + + self.assertIsNone(result) + mock_approve.assert_not_called() + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.approve') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + def test_approve_work_success(self, mock__find_submission, mock_approve, *args): + worker_id = 'test' + study_id = 'test2' + + mock_submission = Submission() + mock_submission.id = 'test3' + mock_submission.status = constants.SubmissionStatus.AWAITING_REVIEW + + mock_submission_approved = Submission() + mock_submission_approved.id = 'test4' + mock_submission_approved.status = constants.SubmissionStatus.APPROVED + + mock__find_submission.return_value = mock_submission + mock_approve.return_value = mock_submission_approved + + result = approve_work(self.client, study_id, worker_id) + + self.assertEqual(mock_submission_approved, result) + mock_approve.assert_called_once() + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.approve') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + def test_approve_work_incorrect_submission_status( + self, mock__find_submission, mock_approve, *args, + ): + worker_id = 'test' + study_id = 'test2' + + mock_submission = Submission() + mock_submission.id = 'test3' + mock_submission.status = constants.SubmissionStatus.ACTIVE + + mock__find_submission.return_value = mock_submission + + result = approve_work(self.client, study_id, worker_id) + + self.assertIsNone(result) + mock_approve.assert_not_called() + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.approve') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + def test_approve_work_exception(self, mock__find_submission, mock_approve, *args): + worker_id = 'test' + study_id = 'test2' + + mock_submission = Submission() + mock_submission.id = 'test3' + mock_submission.status = constants.SubmissionStatus.AWAITING_REVIEW + + mock__find_submission.return_value = mock_submission + + exception_message = 'Error' + mock_approve.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + approve_work(self.client, study_id, worker_id) + + self.assertEqual(cm.exception.message, exception_message) + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.reject') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + def test_reject_work_submission_not_found(self, mock__find_submission, mock_reject, *args): + worker_id = 'test' + study_id = 'test2' + + mock__find_submission.return_value = False + + result = reject_work(self.client, study_id, worker_id) + + self.assertIsNone(result) + mock_reject.assert_not_called() + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.reject') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + def test_reject_work_success(self, mock__find_submission, mock_reject, *args): + worker_id = 'test' + study_id = 'test2' + + mock_submission = Submission() + mock_submission.id = 'test3' + mock_submission.status = constants.SubmissionStatus.AWAITING_REVIEW + + mock_submission_rejected = Submission() + mock_submission_rejected.id = 'test4' + mock_submission_rejected.status = constants.SubmissionStatus.REJECTED + + mock__find_submission.return_value = mock_submission + mock_reject.return_value = mock_submission_rejected + + result = reject_work(self.client, study_id, worker_id) + + self.assertEqual(mock_submission_rejected, result) + mock_reject.assert_called_once() + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.reject') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + def test_reject_work_incorrect_submission_status( + self, mock__find_submission, mock_reject, *args, + ): + worker_id = 'test' + study_id = 'test2' + + mock_submission = Submission() + mock_submission.id = 'test3' + mock_submission.status = constants.SubmissionStatus.ACTIVE + + mock__find_submission.return_value = mock_submission + + result = reject_work(self.client, study_id, worker_id) + + self.assertIsNone(result) + mock_reject.assert_not_called() + + @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.reject') + @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + def test_reject_work_exception(self, mock__find_submission, mock_reject, *args): + worker_id = 'test' + study_id = 'test2' + + mock_submission = Submission() + mock_submission.id = 'test3' + mock_submission.status = constants.SubmissionStatus.AWAITING_REVIEW + + mock__find_submission.return_value = mock_submission + + exception_message = 'Error' + mock_reject.side_effect = ProlificRequestError(exception_message) + with self.assertRaises(ProlificRequestError) as cm: + reject_work(self.client, study_id, worker_id) + + self.assertEqual(cm.exception.message, exception_message) + if __name__ == "__main__": unittest.main() From 1a0d0d4e9c80979a49aa20ea503810eda1b66bfb Mon Sep 17 00:00:00 2001 From: Paul Abumov Date: Tue, 15 Aug 2023 20:29:16 -0400 Subject: [PATCH 2/2] Added unittests for prolific_utils and base_api_resource --- .../prolific/api/base_api_resource.py | 5 + .../providers/prolific/api/status.py | 1 + .../providers/prolific/__init__.py | 5 + .../providers/prolific/api/__init__.py | 5 + .../prolific/api/test_base_api_resourse.py | 362 ++++++++++++ .../providers/prolific/test_prolific_utils.py | 555 ++++++++---------- 6 files changed, 638 insertions(+), 295 deletions(-) create mode 100644 test/abstractions/providers/prolific/api/__init__.py create mode 100644 test/abstractions/providers/prolific/api/test_base_api_resourse.py diff --git a/mephisto/abstractions/providers/prolific/api/base_api_resource.py b/mephisto/abstractions/providers/prolific/api/base_api_resource.py index 85ec0bcde..959144051 100644 --- a/mephisto/abstractions/providers/prolific/api/base_api_resource.py +++ b/mephisto/abstractions/providers/prolific/api/base_api_resource.py @@ -89,6 +89,7 @@ def _base_request( elif method == HTTPMethod.DELETE: response = requests.delete(url, headers=headers, json=params) + else: raise ProlificException("Invalid HTTP method.") @@ -102,6 +103,10 @@ def _base_request( return result + except ProlificException: + # Reraise these errors further to avoid catching them in the latest `except` condition + raise + except requests.exceptions.HTTPError as err: logger.error(f"{log_prefix} Request error: {err}. Response text: `{err.response.text}`") if err.response.status_code == status.HTTP_401_UNAUTHORIZED: diff --git a/mephisto/abstractions/providers/prolific/api/status.py b/mephisto/abstractions/providers/prolific/api/status.py index 12d1156e1..d746c5e23 100644 --- a/mephisto/abstractions/providers/prolific/api/status.py +++ b/mephisto/abstractions/providers/prolific/api/status.py @@ -5,6 +5,7 @@ # LICENSE file in the root directory of this source tree. # 2xx +HTTP_200_OK = 200 HTTP_204_NO_CONTENT = 204 # 4xx diff --git a/test/abstractions/providers/prolific/__init__.py b/test/abstractions/providers/prolific/__init__.py index e69de29bb..240697e32 100644 --- a/test/abstractions/providers/prolific/__init__.py +++ b/test/abstractions/providers/prolific/__init__.py @@ -0,0 +1,5 @@ +#!/usr/bin/env python3 + +# Copyright (c) Facebook, Inc. and its affiliates. +# This source code is licensed under the MIT license found in the +# LICENSE file in the root directory of this source tree. diff --git a/test/abstractions/providers/prolific/api/__init__.py b/test/abstractions/providers/prolific/api/__init__.py new file mode 100644 index 000000000..240697e32 --- /dev/null +++ b/test/abstractions/providers/prolific/api/__init__.py @@ -0,0 +1,5 @@ +#!/usr/bin/env python3 + +# Copyright (c) Facebook, Inc. and its affiliates. +# This source code is licensed under the MIT license found in the +# LICENSE file in the root directory of this source tree. diff --git a/test/abstractions/providers/prolific/api/test_base_api_resourse.py b/test/abstractions/providers/prolific/api/test_base_api_resourse.py new file mode 100644 index 000000000..164cefadd --- /dev/null +++ b/test/abstractions/providers/prolific/api/test_base_api_resourse.py @@ -0,0 +1,362 @@ +#!/usr/bin/env python3 + +# Copyright (c) Facebook, Inc. and its affiliates. +# This source code is licensed under the MIT license found in the +# LICENSE file in the root directory of this source tree. + +import json +import unittest +from unittest.mock import patch +from urllib.parse import urljoin + +import pytest +from requests import Response +from requests.exceptions import HTTPError + +from mephisto.abstractions.providers.prolific.api import status +from mephisto.abstractions.providers.prolific.api.base_api_resource import BaseAPIResource +from mephisto.abstractions.providers.prolific.api.base_api_resource import HTTPMethod +from mephisto.abstractions.providers.prolific.api.exceptions import ProlificAPIKeyError +from mephisto.abstractions.providers.prolific.api.exceptions import ProlificAuthenticationError +from mephisto.abstractions.providers.prolific.api.exceptions import ProlificException +from mephisto.abstractions.providers.prolific.api.exceptions import ProlificRequestError + + +API_KEY_PATH = "mephisto.abstractions.providers.prolific.api.base_api_resource.PROLIFIC_API_KEY" +API_KEY = "test" + + +class TestApiResource(BaseAPIResource): + pass + + +@pytest.mark.prolific +class TestBaseAPIResource(unittest.TestCase): + @patch("requests.get") + def test__base_request_success(self, mock_requests_get, *args): + method = HTTPMethod.GET + api_endpoint = "test/" + params = { + "param": "test", + } + headers = { + "header": "test", + } + content = b'{"id": "60d9aadeb86739de712faee0","name": "Study"}' + + mock_response = Response() + mock_response.raise_for_status = lambda: None + mock_response.status_code = status.HTTP_200_OK + mock_response._content = content + + mock_requests_get.return_value = mock_response + + result = TestApiResource._base_request( + method=method, + api_endpoint=api_endpoint, + params=params, + headers=headers, + api_key=API_KEY, + ) + + self.assertEqual(json.loads(content), result) + mock_requests_get.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={**headers, **{"Authorization": f"Token {API_KEY}"}}, + params=params, + ) + + @patch("requests.get") + def test__base_request_success_no_content(self, mock_requests_get, *args): + method = HTTPMethod.GET + api_endpoint = "test/" + params = { + "param": "test", + } + headers = { + "header": "test", + } + content = None + + mock_response = Response() + mock_response.raise_for_status = lambda: None + mock_response.status_code = status.HTTP_204_NO_CONTENT + mock_response._content = content + + mock_requests_get.return_value = mock_response + + result = TestApiResource._base_request( + method=method, + api_endpoint=api_endpoint, + params=params, + headers=headers, + api_key=API_KEY, + ) + + self.assertEqual(None, result) + mock_requests_get.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={**headers, **{"Authorization": f"Token {API_KEY}"}}, + params=params, + ) + + @patch(API_KEY_PATH, "") + @patch("requests.get") + def test__base_request_no_api_key(self, mock_requests_get, *args): + method = HTTPMethod.GET + api_endpoint = "test/" + params = { + "param": "test", + } + headers = { + "header": "test", + } + + with self.assertRaises(ProlificAPIKeyError) as cm: + TestApiResource._base_request( + method=method, + api_endpoint=api_endpoint, + params=params, + headers=headers, + ) + + self.assertEqual(cm.exception.message, ProlificAPIKeyError.default_message) + mock_requests_get.assert_not_called() + + @patch("requests.get") + def test__base_request_incorrect_request_method(self, mock_requests_get, *args): + method = "unreal_method" + api_endpoint = "test/" + params = { + "param": "test", + } + headers = { + "header": "test", + } + + with self.assertRaises(ProlificException) as cm: + TestApiResource._base_request( + method=method, + api_endpoint=api_endpoint, + params=params, + headers=headers, + api_key=API_KEY, + ) + + self.assertEqual(cm.exception.message, "Invalid HTTP method.") + mock_requests_get.assert_not_called() + + @patch("requests.get") + def test__base_request_request_httperror(self, mock_requests_get, *args): + method = HTTPMethod.GET + api_endpoint = "test/" + params = { + "param": "test", + } + headers = { + "header": "test", + } + content = b"test" + + mock_response = Response() + mock_response.raise_for_status = lambda: None + mock_response.status_code = status.HTTP_204_NO_CONTENT + mock_response._content = content + + error_message = "Error" + exception = HTTPError(error_message) + exception.response = mock_response + + mock_requests_get.side_effect = exception + + with self.assertRaises(ProlificRequestError) as cm: + TestApiResource._base_request( + method=method, + api_endpoint=api_endpoint, + params=params, + headers=headers, + api_key=API_KEY, + ) + + self.assertEqual(cm.exception.message, f'{error_message}. {content.decode("utf8")}') + mock_requests_get.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={**headers, **{"Authorization": f"Token {API_KEY}"}}, + params=params, + ) + + @patch("requests.get") + def test__base_request_request_httperror_unauthorized(self, mock_requests_get, *args): + method = HTTPMethod.GET + api_endpoint = "test/" + params = { + "param": "test", + } + headers = { + "header": "test", + } + content = b"test" + + mock_response = Response() + mock_response.raise_for_status = lambda: None + mock_response.status_code = status.HTTP_401_UNAUTHORIZED + mock_response._content = content + + error_message = "Error" + exception = HTTPError(error_message) + exception.response = mock_response + + mock_requests_get.side_effect = exception + + with self.assertRaises(ProlificAuthenticationError) as cm: + TestApiResource._base_request( + method=method, + api_endpoint=api_endpoint, + params=params, + headers=headers, + api_key=API_KEY, + ) + + self.assertEqual(cm.exception.message, ProlificAuthenticationError.default_message) + self.assertEqual(cm.exception.status_code, ProlificAuthenticationError.status_code) + mock_requests_get.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={**headers, **{"Authorization": f"Token {API_KEY}"}}, + params=params, + ) + + @patch("requests.get") + def test__base_request_unexpected_exception(self, mock_requests_get, *args): + method = HTTPMethod.GET + api_endpoint = "test/" + params = { + "param": "test", + } + headers = { + "header": "test", + } + + error_message = "Error" + exception = ValueError(error_message) + + mock_requests_get.side_effect = exception + + with self.assertRaises(ProlificException) as cm: + TestApiResource._base_request( + method=method, + api_endpoint=api_endpoint, + params=params, + headers=headers, + api_key=API_KEY, + ) + + self.assertEqual(cm.exception.message, ProlificException.default_message) + mock_requests_get.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={**headers, **{"Authorization": f"Token {API_KEY}"}}, + params=params, + ) + + @patch(API_KEY_PATH, API_KEY) + @patch("requests.get") + def test_get(self, mock_requests_get, *args): + api_endpoint = "test-get/" + params = { + "param": "test", + } + content = b'{"id": "60d9aadeb86739de712faee0","name": "Study"}' + + mock_response = Response() + mock_response.raise_for_status = lambda: None + mock_response.status_code = status.HTTP_200_OK + mock_response._content = content + + mock_requests_get.return_value = mock_response + + result = TestApiResource.get(api_endpoint=api_endpoint, params=params) + + self.assertEqual(json.loads(content), result) + mock_requests_get.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={"Authorization": f"Token {API_KEY}"}, + params=params, + ) + + @patch(API_KEY_PATH, API_KEY) + @patch("requests.post") + def test_post(self, mock_requests_post, *args): + api_endpoint = "test-post/" + params = { + "param": "test", + } + content = b'{"id": "60d9aadeb86739de712faee0","name": "Study"}' + + mock_response = Response() + mock_response.raise_for_status = lambda: None + mock_response.status_code = status.HTTP_200_OK + mock_response._content = content + + mock_requests_post.return_value = mock_response + + result = TestApiResource.post(api_endpoint=api_endpoint, params=params) + + self.assertEqual(json.loads(content), result) + mock_requests_post.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={"Authorization": f"Token {API_KEY}"}, + params=params, + ) + + @patch(API_KEY_PATH, API_KEY) + @patch("requests.patch") + def test_patch(self, mock_requests_patch, *args): + api_endpoint = "test-patch/" + params = { + "param": "test", + } + content = b'{"id": "60d9aadeb86739de712faee0","name": "Study"}' + + mock_response = Response() + mock_response.raise_for_status = lambda: None + mock_response.status_code = status.HTTP_200_OK + mock_response._content = content + + mock_requests_patch.return_value = mock_response + + result = TestApiResource.patch(api_endpoint=api_endpoint, params=params) + + self.assertEqual(json.loads(content), result) + mock_requests_patch.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={"Authorization": f"Token {API_KEY}"}, + params=params, + ) + + @patch(API_KEY_PATH, API_KEY) + @patch("requests.delete") + def test_delete(self, mock_requests_delete, *args): + api_endpoint = "test-delete/" + params = { + "param": "test", + } + content = b'{"id": "60d9aadeb86739de712faee0","name": "Study"}' + + mock_response = Response() + mock_response.raise_for_status = lambda: None + mock_response.status_code = status.HTTP_200_OK + mock_response._content = content + + mock_requests_delete.return_value = mock_response + + result = TestApiResource.delete(api_endpoint=api_endpoint, params=params) + + self.assertEqual(json.loads(content), result) + mock_requests_delete.called_once_with( + urljoin("https://api.prolific.co/api/v1/", api_endpoint), + headers={"Authorization": f"Token {API_KEY}"}, + params=params, + ) + + +if __name__ == "__main__": + unittest.main() diff --git a/test/abstractions/providers/prolific/test_prolific_utils.py b/test/abstractions/providers/prolific/test_prolific_utils.py index 1650019e7..5b02af505 100644 --- a/test/abstractions/providers/prolific/test_prolific_utils.py +++ b/test/abstractions/providers/prolific/test_prolific_utils.py @@ -74,6 +74,9 @@ MOCK_PROLIFIC_CONFIG_DIR = "/tmp/" MOCK_PROLIFIC_CONFIG_PATH = "/tmp/test_conf_credentials" +API_PATH = "mephisto.abstractions.providers.prolific.api" +UTILS_PATH = "mephisto.abstractions.providers.prolific.prolific_utils" + @dataclass class MockProlificRequesterArgs(RequesterArgs): @@ -135,26 +138,20 @@ def remove_credentials_file(): if os.path.exists(MOCK_PROLIFIC_CONFIG_PATH): os.remove(MOCK_PROLIFIC_CONFIG_PATH) - @patch("mephisto.abstractions.providers.prolific.api.users.Users.me") + @patch(f"{API_PATH}.users.Users.me") def test_check_credentials_true(self, mock_prolific_users_me, *args): mock_prolific_users_me.return_value = User(id="test") result = check_credentials() self.assertTrue(result) - @patch("mephisto.abstractions.providers.prolific.api.users.Users.me") + @patch(f"{API_PATH}.users.Users.me") def test_check_credentials_false(self, mock_prolific_users_me, *args): mock_prolific_users_me.side_effect = ProlificRequestError() result = check_credentials() self.assertFalse(result) - @patch( - "mephisto.abstractions.providers.prolific.prolific_utils.CREDENTIALS_CONFIG_DIR", - MOCK_PROLIFIC_CONFIG_DIR, - ) - @patch( - "mephisto.abstractions.providers.prolific.prolific_utils.CREDENTIALS_CONFIG_PATH", - MOCK_PROLIFIC_CONFIG_PATH, - ) + @patch(f"{UTILS_PATH}.CREDENTIALS_CONFIG_DIR", MOCK_PROLIFIC_CONFIG_DIR) + @patch(f"{UTILS_PATH}.CREDENTIALS_CONFIG_PATH", MOCK_PROLIFIC_CONFIG_PATH) def test_setup_credentials(self, *args): self.remove_credentials_file() self.assertFalse(os.path.exists(MOCK_PROLIFIC_CONFIG_PATH)) @@ -250,8 +247,8 @@ def test__convert_eligibility_requirements(self, *args): ], ) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.get_balance") - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_workspace") + @patch(f"{API_PATH}.workspaces.Workspaces.get_balance") + @patch(f"{UTILS_PATH}._find_prolific_workspace") def test_check_balance_success(self, mock__find_prolific_workspace, mock_get_balance, *args): expected_value = 9999 @@ -265,14 +262,14 @@ def test_check_balance_success(self, mock__find_prolific_workspace, mock_get_bal balance = check_balance(self.client, workspace_name="test") self.assertEqual(expected_value, balance) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.get_balance") - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_workspace") + @patch(f"{API_PATH}.workspaces.Workspaces.get_balance") + @patch(f"{UTILS_PATH}._find_prolific_workspace") def test_check_balance_no_workspace_name(self, *args): balance = check_balance(self.client) self.assertEqual(None, balance) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.get_balance") - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_workspace") + @patch(f"{API_PATH}.workspaces.Workspaces.get_balance") + @patch(f"{UTILS_PATH}._find_prolific_workspace") def test_check_balance_found_no_workspace(self, mock__find_prolific_workspace, *args): mock_workspace = Workspace() mock_workspace.id = "test" @@ -281,8 +278,8 @@ def test_check_balance_found_no_workspace(self, mock__find_prolific_workspace, * balance = check_balance(self.client, workspace_name="test") self.assertEqual(None, balance) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.get_balance") - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_workspace") + @patch(f"{API_PATH}.workspaces.Workspaces.get_balance") + @patch(f"{UTILS_PATH}._find_prolific_workspace") def test_check_balance_get_balance_exception( self, mock__find_prolific_workspace, @@ -305,7 +302,7 @@ def test_check_balance_get_balance_exception( self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.retrieve") + @patch(f"{API_PATH}.workspaces.Workspaces.retrieve") def test__find_prolific_workspace_with_id_success(self, mock_retrieve, *args): expected_id = "test" expected_title = "test" @@ -319,7 +316,7 @@ def test__find_prolific_workspace_with_id_success(self, mock_retrieve, *args): result = _find_prolific_workspace(self.client, title="", id=expected_id) self.assertEqual((True, mock_workspace), result) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.retrieve") + @patch(f"{API_PATH}.workspaces.Workspaces.retrieve") def test__find_prolific_workspace_with_id_exception(self, mock_retrieve, *args): expected_id = "test" @@ -330,7 +327,7 @@ def test__find_prolific_workspace_with_id_exception(self, mock_retrieve, *args): self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.list") + @patch(f"{API_PATH}.workspaces.Workspaces.list") def test__find_prolific_workspace_with_title_success(self, mock_list, *args): expected_title = "test" @@ -342,7 +339,7 @@ def test__find_prolific_workspace_with_title_success(self, mock_list, *args): result = _find_prolific_workspace(self.client, title=expected_title) self.assertEqual((True, mock_workspace), result) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.list") + @patch(f"{API_PATH}.workspaces.Workspaces.list") def test__find_prolific_workspace_with_title_success_no_result(self, mock_list, *args): expected_title = "test" @@ -354,7 +351,7 @@ def test__find_prolific_workspace_with_title_success_no_result(self, mock_list, result = _find_prolific_workspace(self.client, title=expected_title) self.assertEqual((False, None), result) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.list") + @patch(f"{API_PATH}.workspaces.Workspaces.list") def test__find_prolific_workspace_with_title_exception(self, mock_list, *args): expected_title = "test" @@ -365,7 +362,7 @@ def test__find_prolific_workspace_with_title_exception(self, mock_list, *args): self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_workspace") + @patch(f"{UTILS_PATH}._find_prolific_workspace") def test_find_or_create_prolific_workspace_success_find( self, mock__find_prolific_workspace, @@ -382,8 +379,8 @@ def test_find_or_create_prolific_workspace_success_find( self.assertEqual(mock_workspace, result) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.create") - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_workspace") + @patch(f"{API_PATH}.workspaces.Workspaces.create") + @patch(f"{UTILS_PATH}._find_prolific_workspace") def test_find_or_create_prolific_workspace_success_create( self, mock__find_prolific_workspace, @@ -402,8 +399,8 @@ def test_find_or_create_prolific_workspace_success_create( self.assertEqual(mock_workspace, result) - @patch("mephisto.abstractions.providers.prolific.api.workspaces.Workspaces.create") - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_workspace") + @patch(f"{API_PATH}.workspaces.Workspaces.create") + @patch(f"{UTILS_PATH}._find_prolific_workspace") def test_find_or_create_prolific_workspace_create_exception( self, mock__find_prolific_workspace, @@ -424,7 +421,7 @@ def test_find_or_create_prolific_workspace_create_exception( self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.projects.Projects.list_for_workspace") + @patch(f"{API_PATH}.projects.Projects.list_for_workspace") def test__find_prolific_project_success_with_title(self, mock_list_for_workspace, *args): workspace_id = "test" project_title = "test2" @@ -439,7 +436,7 @@ def test__find_prolific_project_success_with_title(self, mock_list_for_workspace self.assertEqual((True, mock_project), result) self.assertFalse(hasattr(mock_project, "id")) - @patch("mephisto.abstractions.providers.prolific.api.projects.Projects.list_for_workspace") + @patch(f"{API_PATH}.projects.Projects.list_for_workspace") def test__find_prolific_project_success_with_id(self, mock_list_for_workspace, *args): workspace_id = "test" project_title = "test2" @@ -456,7 +453,7 @@ def test__find_prolific_project_success_with_id(self, mock_list_for_workspace, * self.assertEqual((True, mock_project), result) self.assertTrue(hasattr(mock_project, "id")) - @patch("mephisto.abstractions.providers.prolific.api.projects.Projects.list_for_workspace") + @patch(f"{API_PATH}.projects.Projects.list_for_workspace") def test__find_prolific_project_success_no_result(self, mock_list_for_workspace, *args): workspace_id = "test" project_title = "test2" @@ -470,7 +467,7 @@ def test__find_prolific_project_success_no_result(self, mock_list_for_workspace, self.assertEqual((False, None), result) - @patch("mephisto.abstractions.providers.prolific.api.projects.Projects.list_for_workspace") + @patch(f"{API_PATH}.projects.Projects.list_for_workspace") def test__find_prolific_project_exception(self, mock_list_for_workspace, *args): workspace_id = "test" project_title = "test2" @@ -482,7 +479,7 @@ def test__find_prolific_project_exception(self, mock_list_for_workspace, *args): self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_project") + @patch(f"{UTILS_PATH}._find_prolific_project") def test_find_or_create_prolific_project_success_find(self, mock__find_prolific_project, *args): workspace_id = "test" project_title = "test2" @@ -496,8 +493,8 @@ def test_find_or_create_prolific_project_success_find(self, mock__find_prolific_ self.assertEqual(mock_project, result) - @patch("mephisto.abstractions.providers.prolific.api.projects.Projects.create_for_workspace") - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_project") + @patch(f"{API_PATH}.projects.Projects.create_for_workspace") + @patch(f"{UTILS_PATH}._find_prolific_project") def test_find_or_create_prolific_project_success_create( self, mock__find_prolific_project, @@ -517,8 +514,8 @@ def test_find_or_create_prolific_project_success_create( self.assertEqual(mock_project, result) - @patch("mephisto.abstractions.providers.prolific.api.projects.Projects.create_for_workspace") - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_prolific_project") + @patch(f"{API_PATH}.projects.Projects.create_for_workspace") + @patch(f"{UTILS_PATH}._find_prolific_project") def test_find_or_create_prolific_project_create_exception( self, mock__find_prolific_project, @@ -537,9 +534,7 @@ def test_find_or_create_prolific_project_create_exception( self.assertEqual(cm.exception.message, exception_message) - @patch( - "mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.remove" - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.remove") def test_delete_qualification_success(self, mock_remove, *args): prolific_participant_group_id = "test" @@ -549,9 +544,7 @@ def test_delete_qualification_success(self, mock_remove, *args): self.assertTrue(result) - @patch( - "mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.remove" - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.remove") def test_delete_qualification_exception(self, mock_remove, *args): prolific_participant_group_id = "test" @@ -562,7 +555,7 @@ def test_delete_qualification_exception(self, mock_remove, *args): self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.list") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.list") def test__find_qualification_success(self, mock_participant_groups_list, *args): prolific_project_id = uuid4().hex[:24] qualification_name = "test" @@ -579,7 +572,7 @@ def test__find_qualification_success(self, mock_participant_groups_list, *args): _, q = _find_qualification(self.client, prolific_project_id, qualification_name) self.assertEqual(q.id, expected_qualification_id) - @patch("mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.list") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.list") def test__find_qualification_no_qualification(self, mock_participant_groups_list, *args): prolific_project_id = uuid4().hex[:24] qualification_name = "test" @@ -587,7 +580,7 @@ def test__find_qualification_no_qualification(self, mock_participant_groups_list result = _find_qualification(self.client, prolific_project_id, qualification_name) self.assertEqual(result, (False, None)) - @patch("mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.list") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.list") def test__find_qualification_error(self, mock_participant_groups_list, *args): prolific_project_id = uuid4().hex[:24] qualification_name = "test" @@ -597,9 +590,7 @@ def test__find_qualification_error(self, mock_participant_groups_list, *args): _find_qualification(self.client, prolific_project_id, qualification_name) self.assertEqual(cm.exception.message, exception_message) - @patch( - "mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.create" - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.create") def test_create_qualification_success(self, mock_create, *args): prolific_project_id = "test" qualification_name = "test2" @@ -615,9 +606,7 @@ def test_create_qualification_success(self, mock_create, *args): self.assertEqual(mock_participant_group, result) - @patch( - "mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.create" - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.create") def test_create_qualification_exception(self, mock_create, *args): prolific_project_id = "test" qualification_name = "test2" @@ -629,7 +618,7 @@ def test_create_qualification_exception(self, mock_create, *args): self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_qualification") + @patch(f"{UTILS_PATH}._find_qualification") def test_find_or_create_qualification_found_one(self, mock_find_qualification, *args): prolific_project_id = uuid4().hex[:24] qualification_name = "test" @@ -644,10 +633,8 @@ def test_find_or_create_qualification_found_one(self, mock_find_qualification, * ) self.assertEqual(result.id, expected_qualification_id) - @patch( - "mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.create" - ) - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_qualification") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.create") + @patch(f"{UTILS_PATH}._find_qualification") def test_find_or_create_qualification_created_new( self, mock_find_qualification, @@ -670,10 +657,8 @@ def test_find_or_create_qualification_created_new( ) self.assertEqual(result.id, expected_qualification_id) - @patch( - "mephisto.abstractions.providers.prolific.api.participant_groups.ParticipantGroups.create" - ) - @patch("mephisto.abstractions.providers.prolific.prolific_utils._find_qualification") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.create") + @patch(f"{UTILS_PATH}._find_qualification") def test_find_or_create_qualification_error( self, mock_find_qualification, @@ -739,8 +724,8 @@ def test__get_external_study_url(self, mock_get_full_domain, *args): result_ec2_architect, ) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.update") - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.create") + @patch(f"{API_PATH}.studies.Studies.update") + @patch(f"{API_PATH}.studies.Studies.create") def test_create_study_success(self, mock_study_create, mock_study_update, *args): project_id = uuid4().hex[:24] expected_study_id = uuid4().hex[:24] @@ -769,8 +754,8 @@ def test_create_study_success(self, mock_study_create, mock_study_update, *args) ) self.assertEqual(study.id, expected_study_id) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.update") - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.create") + @patch(f"{API_PATH}.studies.Studies.update") + @patch(f"{API_PATH}.studies.Studies.create") def test_create_study_error(self, mock_study_create, *args): project_id = uuid4().hex[:24] exception_message = "Error" @@ -783,8 +768,8 @@ def test_create_study_error(self, mock_study_create, *args): ) self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.update") - @patch("mephisto.abstractions.providers.prolific.prolific_utils.get_study") + @patch(f"{API_PATH}.studies.Studies.update") + @patch(f"{UTILS_PATH}.get_study") def test_increase_total_available_places_for_study_success(self, mock_get_study, *args): study_id = "test" @@ -798,8 +783,8 @@ def test_increase_total_available_places_for_study_success(self, mock_get_study, self.assertEqual(mock_study, result) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.update") - @patch("mephisto.abstractions.providers.prolific.prolific_utils.get_study") + @patch(f"{API_PATH}.studies.Studies.update") + @patch(f"{UTILS_PATH}.get_study") def test_increase_total_available_places_for_study_exception( self, mock_get_study, @@ -821,7 +806,7 @@ def test_increase_total_available_places_for_study_exception( self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.retrieve") + @patch(f"{API_PATH}.studies.Studies.retrieve") def test_get_study_success(self, mock_retrieve, *args): study_id = "test" @@ -834,7 +819,7 @@ def test_get_study_success(self, mock_retrieve, *args): self.assertEqual(mock_study, result) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.retrieve") + @patch(f"{API_PATH}.studies.Studies.retrieve") def test_get_study_exception(self, mock_retrieve, *args): study_id = "test" @@ -848,7 +833,7 @@ def test_get_study_exception(self, mock_retrieve, *args): self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.publish") + @patch(f"{API_PATH}.studies.Studies.publish") def test_publish_study_success(self, mock_publish, *args): study_id = "test" @@ -861,7 +846,7 @@ def test_publish_study_success(self, mock_publish, *args): self.assertEqual(study_id, result) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.publish") + @patch(f"{API_PATH}.studies.Studies.publish") def test_publish_study_exception(self, mock_publish, *args): study_id = "test" @@ -875,7 +860,7 @@ def test_publish_study_exception(self, mock_publish, *args): self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.stop") + @patch(f"{API_PATH}.studies.Studies.stop") def test_stop_study_success(self, mock_stop, *args): study_id = "test" @@ -888,7 +873,7 @@ def test_stop_study_success(self, mock_stop, *args): self.assertEqual(mock_study, result) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.stop") + @patch(f"{API_PATH}.studies.Studies.stop") def test_stop_study_exception(self, mock_stop, *args): study_id = "test" @@ -902,9 +887,9 @@ def test_stop_study_exception(self, mock_stop, *args): self.assertEqual(cm.exception.message, exception_message) - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.update") - @patch("mephisto.abstractions.providers.prolific.api.studies.Studies.stop") - @patch("mephisto.abstractions.providers.prolific.prolific_utils.get_study") + @patch(f"{API_PATH}.studies.Studies.update") + @patch(f"{API_PATH}.studies.Studies.stop") + @patch(f"{UTILS_PATH}.get_study") def test_expire_study_success(self, mock_get_study, mock_stop, *args): study_id = "test" @@ -919,7 +904,7 @@ def test_expire_study_success(self, mock_get_study, mock_stop, *args): self.assertEqual(mock_study, result) - @patch("mephisto.abstractions.providers.prolific.prolific_utils.get_study") + @patch(f"{UTILS_PATH}.get_study") def test_expire_study_exception(self, mock_get_study, *args): study_id = "test" @@ -959,83 +944,71 @@ def test_is_study_expired(self, *args): self.assertFalse(result_just_with_awaiting_review_status) self.assertTrue(result_with_completed_status_and_internal_name) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.add_participants_to_group' - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.add_participants_to_group") def test_add_workers_to_qualification_success(self, *args): - worker_ids = ['test', 'test2'] - participant_group_id = 'test3' + worker_ids = ["test", "test2"] + participant_group_id = "test3" result = add_workers_to_qualification(self.client, worker_ids, participant_group_id) self.assertIsNone(result) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.add_participants_to_group' - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.add_participants_to_group") def test_add_workers_to_qualification_exception(self, mock_add_participants_to_group, *args): - worker_ids = ['test', 'test2'] - participant_group_id = 'test3' + worker_ids = ["test", "test2"] + participant_group_id = "test3" - exception_message = 'Error' + exception_message = "Error" mock_add_participants_to_group.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: add_workers_to_qualification(self.client, worker_ids, participant_group_id) self.assertEqual(cm.exception.message, exception_message) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.add_participants_to_group' - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.add_participants_to_group") def test_give_worker_qualification_success(self, mock_add_participants_to_group, *args): - worker_id = 'test' - participant_group_id = 'test3' + worker_id = "test" + participant_group_id = "test3" result = give_worker_qualification(self.client, worker_id, participant_group_id) mock_add_participants_to_group.assert_called_once_with( - id=participant_group_id, participant_ids=[worker_id], + id=participant_group_id, + participant_ids=[worker_id], ) self.assertIsNone(result) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.remove_participants_from_group' - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.remove_participants_from_group") def test_remove_worker_qualification_success(self, *args): - worker_id = 'test' - participant_group_id = 'test2' + worker_id = "test" + participant_group_id = "test2" result = remove_worker_qualification(self.client, worker_id, participant_group_id) self.assertIsNone(result) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.remove_participants_from_group' - ) + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.remove_participants_from_group") def test_remove_worker_qualification_exception( - self, mock_remove_participants_from_group, *args, + self, + mock_remove_participants_from_group, + *args, ): - worker_id = 'test' - participant_group_id = 'test2' + worker_id = "test" + participant_group_id = "test2" - exception_message = 'Error' + exception_message = "Error" mock_remove_participants_from_group.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: remove_worker_qualification(self.client, worker_id, participant_group_id) self.assertEqual(cm.exception.message, exception_message) - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') - @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + @patch(f"{API_PATH}.bonuses.Bonuses.pay") + @patch(f"{API_PATH}.bonuses.Bonuses.set_up") + @patch(f"{UTILS_PATH}.check_balance") def test_pay_bonus_empty_balance(self, mock_check_balance, mock_set_up, mock_pay, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" bonus_amount = 1000 mock_check_balance.return_value = False @@ -1046,12 +1019,12 @@ def test_pay_bonus_empty_balance(self, mock_check_balance, mock_set_up, mock_pay mock_pay.assert_not_called() self.assertFalse(result) - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') - @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + @patch(f"{API_PATH}.bonuses.Bonuses.pay") + @patch(f"{API_PATH}.bonuses.Bonuses.set_up") + @patch(f"{UTILS_PATH}.check_balance") def test_pay_bonus_empty_balance(self, mock_check_balance, mock_set_up, mock_pay, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" bonus_amount = 1000 mock_check_balance.return_value = False @@ -1062,13 +1035,13 @@ def test_pay_bonus_empty_balance(self, mock_check_balance, mock_set_up, mock_pay mock_pay.assert_not_called() self.assertFalse(result) - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') - @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + @patch(f"{API_PATH}.bonuses.Bonuses.pay") + @patch(f"{API_PATH}.bonuses.Bonuses.set_up") + @patch(f"{UTILS_PATH}.check_balance") def test_pay_bonus_success(self, mock_check_balance, mock_set_up, mock_pay, *args): - worker_id = 'test' - study_id = 'test2' - bonus_payments_id = 'test3' + worker_id = "test" + study_id = "test2" + bonus_payments_id = "test3" bonus_amount = 1000 mock_bonus_payments = BonusPayments() @@ -1078,36 +1051,36 @@ def test_pay_bonus_success(self, mock_check_balance, mock_set_up, mock_pay, *arg result = pay_bonus(self.client, mock_task_run_args, worker_id, bonus_amount, study_id) - mock_set_up.assert_called_once_with(study_id, f'{worker_id},{bonus_amount / 100}') + mock_set_up.assert_called_once_with(study_id, f"{worker_id},{bonus_amount / 100}") mock_pay.assert_called_once_with(mock_bonus_payments.id) self.assertTrue(result) - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') - @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + @patch(f"{API_PATH}.bonuses.Bonuses.pay") + @patch(f"{API_PATH}.bonuses.Bonuses.set_up") + @patch(f"{UTILS_PATH}.check_balance") def test_pay_bonus_set_up_exception(self, mock_check_balance, mock_set_up, mock_pay, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" bonus_amount = 1000 mock_check_balance.return_value = True - exception_message = 'Error' + exception_message = "Error" mock_set_up.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: pay_bonus(self.client, mock_task_run_args, worker_id, bonus_amount, study_id) self.assertEqual(cm.exception.message, exception_message) - mock_set_up.assert_called_once_with(study_id, f'{worker_id},{bonus_amount / 100}') + mock_set_up.assert_called_once_with(study_id, f"{worker_id},{bonus_amount / 100}") mock_pay.assert_not_called() - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.pay') - @patch('mephisto.abstractions.providers.prolific.api.bonuses.Bonuses.set_up') - @patch('mephisto.abstractions.providers.prolific.prolific_utils.check_balance') + @patch(f"{API_PATH}.bonuses.Bonuses.pay") + @patch(f"{API_PATH}.bonuses.Bonuses.set_up") + @patch(f"{UTILS_PATH}.check_balance") def test_pay_bonus_pay_exception(self, mock_check_balance, mock_set_up, mock_pay, *args): - worker_id = 'test' - study_id = 'test2' - bonus_payments_id = 'test3' + worker_id = "test" + study_id = "test2" + bonus_payments_id = "test3" bonus_amount = 1000 mock_bonus_payments = BonusPayments() @@ -1115,22 +1088,18 @@ def test_pay_bonus_pay_exception(self, mock_check_balance, mock_set_up, mock_pay mock_check_balance.return_value = True mock_set_up.return_value = mock_bonus_payments - exception_message = 'Error' + exception_message = "Error" mock_pay.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: pay_bonus(self.client, mock_task_run_args, worker_id, bonus_amount, study_id) self.assertEqual(cm.exception.message, exception_message) - mock_set_up.assert_called_once_with(study_id, f'{worker_id},{bonus_amount / 100}') + mock_set_up.assert_called_once_with(study_id, f"{worker_id},{bonus_amount / 100}") mock_pay.assert_called_once_with(mock_bonus_payments.id) - @patch('mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_qualification') - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' - ) - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' - ) + @patch(f"{UTILS_PATH}.find_or_create_qualification") + @patch(f"{UTILS_PATH}.find_or_create_prolific_project") + @patch(f"{UTILS_PATH}.find_or_create_prolific_workspace") def test__get_block_list_qualification( self, mock_find_or_create_prolific_workspace, @@ -1139,13 +1108,13 @@ def test__get_block_list_qualification( *args, ): mock_workspace = Workspace() - mock_workspace.id = 'test' + mock_workspace.id = "test" mock_project = Project() - mock_project.id = 'test2' - mock_project.title = 'test2_title' + mock_project.id = "test2" + mock_project.title = "test2_title" mock_participant_group = ParticipantGroup() - mock_participant_group.id = 'test3' - mock_participant_group.name = 'test3_name' + mock_participant_group.id = "test3" + mock_participant_group.name = "test3_name" mock_find_or_create_prolific_workspace.return_value = mock_workspace mock_find_or_create_prolific_project.return_value = mock_project @@ -1155,15 +1124,18 @@ def test__get_block_list_qualification( self.assertEqual(mock_participant_group, result) - @patch('mephisto.abstractions.providers.prolific.prolific_utils.add_workers_to_qualification') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._get_block_list_qualification') + @patch(f"{UTILS_PATH}.add_workers_to_qualification") + @patch(f"{UTILS_PATH}._get_block_list_qualification") def test_block_worker_success( - self, mock__get_block_list_qualification, mock_add_workers_to_qualification, *args, + self, + mock__get_block_list_qualification, + mock_add_workers_to_qualification, + *args, ): - worker_id = 'test' + worker_id = "test" mock_participant_group = ParticipantGroup() - mock_participant_group.id = 'test2' + mock_participant_group.id = "test2" mock__get_block_list_qualification.return_value = mock_participant_group @@ -1171,21 +1143,23 @@ def test_block_worker_success( self.assertIsNone(result) mock_add_workers_to_qualification.called_once_with( - self.client, workers_ids=[worker_id], qualification_id=mock_participant_group.id, + self.client, + workers_ids=[worker_id], + qualification_id=mock_participant_group.id, ) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.remove_participants_from_group' - ) - @patch('mephisto.abstractions.providers.prolific.prolific_utils._get_block_list_qualification') + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.remove_participants_from_group") + @patch(f"{UTILS_PATH}._get_block_list_qualification") def test_unblock_worker_success( - self, mock__get_block_list_qualification, mock_remove_participants_from_group, *args, + self, + mock__get_block_list_qualification, + mock_remove_participants_from_group, + *args, ): - worker_id = 'test' + worker_id = "test" mock_participant_group = ParticipantGroup() - mock_participant_group.id = 'test2' + mock_participant_group.id = "test2" mock__get_block_list_qualification.return_value = mock_participant_group @@ -1193,24 +1167,21 @@ def test_unblock_worker_success( self.assertIsNone(result) mock_remove_participants_from_group.called_once_with( - id=mock_participant_group.id, participant_ids=[worker_id], + id=mock_participant_group.id, + participant_ids=[worker_id], ) - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' - ) - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' - ) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.list_participants_for_group' - ) - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_qualification') + @patch(f"{UTILS_PATH}.find_or_create_prolific_project") + @patch(f"{UTILS_PATH}.find_or_create_prolific_workspace") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.list_participants_for_group") + @patch(f"{UTILS_PATH}._find_qualification") def test_is_worker_blocked_no_block_list_qualification( - self, mock__find_qualification, mock_list_participants_for_group, *args, + self, + mock__find_qualification, + mock_list_participants_for_group, + *args, ): - worker_id = 'test' + worker_id = "test" mock__find_qualification.return_value = (False, None) @@ -1219,24 +1190,20 @@ def test_is_worker_blocked_no_block_list_qualification( self.assertFalse(result) mock_list_participants_for_group.assert_not_called() - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' - ) - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' - ) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.list_participants_for_group' - ) - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_qualification') + @patch(f"{UTILS_PATH}.find_or_create_prolific_project") + @patch(f"{UTILS_PATH}.find_or_create_prolific_workspace") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.list_participants_for_group") + @patch(f"{UTILS_PATH}._find_qualification") def test_is_worker_blocked_success_true( - self, mock__find_qualification, mock_list_participants_for_group, *args, + self, + mock__find_qualification, + mock_list_participants_for_group, + *args, ): - worker_id = 'test' + worker_id = "test" mock_participant_group = ParticipantGroup() - mock_participant_group.id = 'test2' + mock_participant_group.id = "test2" mock_participant = Participant() mock_participant.participant_id = worker_id @@ -1248,26 +1215,22 @@ def test_is_worker_blocked_success_true( self.assertTrue(result) mock_list_participants_for_group.assert_called_once() - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' - ) - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' - ) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.list_participants_for_group' - ) - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_qualification') + @patch(f"{UTILS_PATH}.find_or_create_prolific_project") + @patch(f"{UTILS_PATH}.find_or_create_prolific_workspace") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.list_participants_for_group") + @patch(f"{UTILS_PATH}._find_qualification") def test_is_worker_blocked_success_false( - self, mock__find_qualification, mock_list_participants_for_group, *args, + self, + mock__find_qualification, + mock_list_participants_for_group, + *args, ): - worker_id = 'test' + worker_id = "test" mock_participant_group = ParticipantGroup() - mock_participant_group.id = 'test2' + mock_participant_group.id = "test2" mock_participant = Participant() - mock_participant.participant_id = 'test3' + mock_participant.participant_id = "test3" mock__find_qualification.return_value = (True, mock_participant_group) mock_list_participants_for_group.return_value = [mock_participant] @@ -1277,30 +1240,26 @@ def test_is_worker_blocked_success_false( self.assertFalse(result) mock_list_participants_for_group.assert_called_once() - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_project' - ) - @patch( - 'mephisto.abstractions.providers.prolific.prolific_utils.find_or_create_prolific_workspace' - ) - @patch( - 'mephisto.abstractions.providers.prolific.api.' - 'participant_groups.ParticipantGroups.list_participants_for_group' - ) - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_qualification') + @patch(f"{UTILS_PATH}.find_or_create_prolific_project") + @patch(f"{UTILS_PATH}.find_or_create_prolific_workspace") + @patch(f"{API_PATH}.participant_groups.ParticipantGroups.list_participants_for_group") + @patch(f"{UTILS_PATH}._find_qualification") def test_is_worker_blocked_exception( - self, mock__find_qualification, mock_list_participants_for_group, *args, + self, + mock__find_qualification, + mock_list_participants_for_group, + *args, ): - worker_id = 'test' + worker_id = "test" mock_participant_group = ParticipantGroup() - mock_participant_group.id = 'test2' + mock_participant_group.id = "test2" mock_participant = Participant() mock_participant.participant_id = worker_id mock__find_qualification.return_value = (True, mock_participant_group) - exception_message = 'Error' + exception_message = "Error" mock_list_participants_for_group.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: is_worker_blocked(self.client, mock_task_run_args, worker_id) @@ -1308,7 +1267,7 @@ def test_is_worker_blocked_exception( self.assertEqual(cm.exception.message, exception_message) mock_list_participants_for_group.assert_called_once() - @patch('mephisto.abstractions.providers.prolific.api.studies.Studies.calculate_cost') + @patch(f"{API_PATH}.studies.Studies.calculate_cost") def test_calculate_pay_amount_success(self, mock_calculate_cost, *args): task_amount = 1000 total_available_places = 2 @@ -1319,25 +1278,25 @@ def test_calculate_pay_amount_success(self, mock_calculate_cost, *args): self.assertEqual(task_amount * total_available_places, result) - @patch('mephisto.abstractions.providers.prolific.api.studies.Studies.calculate_cost') + @patch(f"{API_PATH}.studies.Studies.calculate_cost") def test_calculate_pay_amount_exception(self, mock_calculate_cost, *args): task_amount = 1000 total_available_places = 2 - exception_message = 'Error' + exception_message = "Error" mock_calculate_cost.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: calculate_pay_amount(self.client, task_amount, total_available_places) self.assertEqual(cm.exception.message, exception_message) - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.list') + @patch(f"{API_PATH}.submissions.Submissions.list") def test__find_submission_success_found(self, mock_list, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock_list_submission = ListSubmission() - mock_list_submission.id = 'test3' + mock_list_submission.id = "test3" mock_list_submission.participant_id = worker_id mock_list.return_value = [mock_list_submission] @@ -1346,14 +1305,14 @@ def test__find_submission_success_found(self, mock_list, *args): self.assertEqual(mock_list_submission, result) - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.list') + @patch(f"{API_PATH}.submissions.Submissions.list") def test__find_submission_success_not_found(self, mock_list, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock_list_submission = ListSubmission() - mock_list_submission.id = 'test3' - mock_list_submission.participant_id = 'test4' + mock_list_submission.id = "test3" + mock_list_submission.participant_id = "test4" mock_list.return_value = [mock_list_submission] @@ -1361,24 +1320,24 @@ def test__find_submission_success_not_found(self, mock_list, *args): self.assertEqual(None, result) - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.list') + @patch(f"{API_PATH}.submissions.Submissions.list") def test__find_submission_exception(self, mock_list, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" - exception_message = 'Error' + exception_message = "Error" mock_list.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: _find_submission(self.client, study_id, worker_id) self.assertEqual(cm.exception.message, exception_message) - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.retrieve') + @patch(f"{API_PATH}.submissions.Submissions.retrieve") def test_get_submission_success(self, mock_retrieve, *args): - submission_id = 'test' + submission_id = "test" mock_submission = Submission() - mock_submission.id = 'test2' + mock_submission.id = "test2" mock_retrieve.return_value = mock_submission @@ -1386,22 +1345,22 @@ def test_get_submission_success(self, mock_retrieve, *args): self.assertEqual(mock_submission, result) - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.retrieve') + @patch(f"{API_PATH}.submissions.Submissions.retrieve") def test_get_submission_exception(self, mock_retrieve, *args): - submission_id = 'test' + submission_id = "test" - exception_message = 'Error' + exception_message = "Error" mock_retrieve.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: get_submission(self.client, submission_id) self.assertEqual(cm.exception.message, exception_message) - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.approve') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + @patch(f"{API_PATH}.submissions.Submissions.approve") + @patch(f"{UTILS_PATH}._find_submission") def test_approve_work_submission_not_found(self, mock__find_submission, mock_approve, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock__find_submission.return_value = False @@ -1410,18 +1369,18 @@ def test_approve_work_submission_not_found(self, mock__find_submission, mock_app self.assertIsNone(result) mock_approve.assert_not_called() - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.approve') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + @patch(f"{API_PATH}.submissions.Submissions.approve") + @patch(f"{UTILS_PATH}._find_submission") def test_approve_work_success(self, mock__find_submission, mock_approve, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock_submission = Submission() - mock_submission.id = 'test3' + mock_submission.id = "test3" mock_submission.status = constants.SubmissionStatus.AWAITING_REVIEW mock_submission_approved = Submission() - mock_submission_approved.id = 'test4' + mock_submission_approved.id = "test4" mock_submission_approved.status = constants.SubmissionStatus.APPROVED mock__find_submission.return_value = mock_submission @@ -1432,16 +1391,19 @@ def test_approve_work_success(self, mock__find_submission, mock_approve, *args): self.assertEqual(mock_submission_approved, result) mock_approve.assert_called_once() - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.approve') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + @patch(f"{API_PATH}.submissions.Submissions.approve") + @patch(f"{UTILS_PATH}._find_submission") def test_approve_work_incorrect_submission_status( - self, mock__find_submission, mock_approve, *args, + self, + mock__find_submission, + mock_approve, + *args, ): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock_submission = Submission() - mock_submission.id = 'test3' + mock_submission.id = "test3" mock_submission.status = constants.SubmissionStatus.ACTIVE mock__find_submission.return_value = mock_submission @@ -1451,30 +1413,30 @@ def test_approve_work_incorrect_submission_status( self.assertIsNone(result) mock_approve.assert_not_called() - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.approve') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + @patch(f"{API_PATH}.submissions.Submissions.approve") + @patch(f"{UTILS_PATH}._find_submission") def test_approve_work_exception(self, mock__find_submission, mock_approve, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock_submission = Submission() - mock_submission.id = 'test3' + mock_submission.id = "test3" mock_submission.status = constants.SubmissionStatus.AWAITING_REVIEW mock__find_submission.return_value = mock_submission - exception_message = 'Error' + exception_message = "Error" mock_approve.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: approve_work(self.client, study_id, worker_id) self.assertEqual(cm.exception.message, exception_message) - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.reject') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + @patch(f"{API_PATH}.submissions.Submissions.reject") + @patch(f"{UTILS_PATH}._find_submission") def test_reject_work_submission_not_found(self, mock__find_submission, mock_reject, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock__find_submission.return_value = False @@ -1483,18 +1445,18 @@ def test_reject_work_submission_not_found(self, mock__find_submission, mock_reje self.assertIsNone(result) mock_reject.assert_not_called() - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.reject') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + @patch(f"{API_PATH}.submissions.Submissions.reject") + @patch(f"{UTILS_PATH}._find_submission") def test_reject_work_success(self, mock__find_submission, mock_reject, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock_submission = Submission() - mock_submission.id = 'test3' + mock_submission.id = "test3" mock_submission.status = constants.SubmissionStatus.AWAITING_REVIEW mock_submission_rejected = Submission() - mock_submission_rejected.id = 'test4' + mock_submission_rejected.id = "test4" mock_submission_rejected.status = constants.SubmissionStatus.REJECTED mock__find_submission.return_value = mock_submission @@ -1505,16 +1467,19 @@ def test_reject_work_success(self, mock__find_submission, mock_reject, *args): self.assertEqual(mock_submission_rejected, result) mock_reject.assert_called_once() - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.reject') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + @patch(f"{API_PATH}.submissions.Submissions.reject") + @patch(f"{UTILS_PATH}._find_submission") def test_reject_work_incorrect_submission_status( - self, mock__find_submission, mock_reject, *args, + self, + mock__find_submission, + mock_reject, + *args, ): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock_submission = Submission() - mock_submission.id = 'test3' + mock_submission.id = "test3" mock_submission.status = constants.SubmissionStatus.ACTIVE mock__find_submission.return_value = mock_submission @@ -1524,19 +1489,19 @@ def test_reject_work_incorrect_submission_status( self.assertIsNone(result) mock_reject.assert_not_called() - @patch('mephisto.abstractions.providers.prolific.api.submissions.Submissions.reject') - @patch('mephisto.abstractions.providers.prolific.prolific_utils._find_submission') + @patch(f"{API_PATH}.submissions.Submissions.reject") + @patch(f"{UTILS_PATH}._find_submission") def test_reject_work_exception(self, mock__find_submission, mock_reject, *args): - worker_id = 'test' - study_id = 'test2' + worker_id = "test" + study_id = "test2" mock_submission = Submission() - mock_submission.id = 'test3' + mock_submission.id = "test3" mock_submission.status = constants.SubmissionStatus.AWAITING_REVIEW mock__find_submission.return_value = mock_submission - exception_message = 'Error' + exception_message = "Error" mock_reject.side_effect = ProlificRequestError(exception_message) with self.assertRaises(ProlificRequestError) as cm: reject_work(self.client, study_id, worker_id)