Skip to content
Snippets Groups Projects
Commit 6cc761af authored by Martin Juhás's avatar Martin Juhás
Browse files

Merge branch '234-sending-emails-to-multiple-recipients' into 'main'

Resolve "Sending Emails to Multiple Recipients"

Closes #234

See merge request inject/backend!249
parents 82942c55 0239b8e8
No related branches found
No related tags found
No related merge requests found
...@@ -35,7 +35,8 @@ disable_error_code = ["annotation-unchecked"] ...@@ -35,7 +35,8 @@ disable_error_code = ["annotation-unchecked"]
exclude = [ exclude = [
"migrations", "migrations",
"tests", "tests",
"dev" "dev",
"scripts",
] ]
[tool.black] [tool.black]
......
...@@ -67,47 +67,39 @@ def send_email( ...@@ -67,47 +67,39 @@ def send_email(
def _validate_participants( def _validate_participants(
participant_addresses: List[str], exercise: Exercise participant_addresses: List[str], exercise: Exercise
) -> List[EmailParticipant]: ) -> QuerySet[EmailParticipant]:
if len(set(participant_addresses)) < 2: distinct_addresses = set(participant_addresses)
if len(distinct_addresses) < 2:
raise RunningExerciseOperationException( raise RunningExerciseOperationException(
"A thread must have at least 2 participants" "A thread must have at least 2 participants"
) )
email_between_teams = exercise.config.has_enabled( participants = EmailParticipant.objects.filter(
Feature.EMAIL_BETWEEN_TEAMS exercise_id=exercise.id, address__in=distinct_addresses
) )
participants: List[EmailParticipant] = []
previous_definition_participant: Optional[EmailParticipant] = None if len(participants) != len(distinct_addresses):
has_team = False # how much do we care about the error message?
raise RunningExerciseOperationException(
for address in participant_addresses: f"Some address/es from this list `{distinct_addresses}` don't exist in the exercise"
# TODO: Handle email address that is not included in the definition,
# needs to be logged somewhere, however action logs do not seem as
# the best choice, because I am not sure if this should be included
# in the trainee visible logs
# maybe add a new action log type for errors, which can be used for filtering
participant = get_model(
EmailParticipant, exercise=exercise, address=address
) )
if participant.is_definition():
if previous_definition_participant is not None: team_participants_count = sum(
raise RunningExerciseOperationException( map(lambda participant: participant.team_id is not None, participants)
f"Cannot send email to these addresses at the same time: " )
f"'{previous_definition_participant.address}',"
f" '{participant.address}'" if team_participants_count == 0:
) raise RunningExerciseOperationException(
previous_definition_participant = participant f"A thread must contain at least 1 team address"
else: )
# this is probably an extremely rare edge case, however there is a possibility
# that a team randomly guesses another teams address and without this check if team_participants_count > 1 and not exercise.config.has_enabled(
# the 'email_between_teams' feature could be bypassed Feature.EMAIL_BETWEEN_TEAMS
if has_team and not email_between_teams: ):
raise RunningExerciseOperationException( raise RunningExerciseOperationException(
"Email communication between teams is disabled" f"Email between teams is not enabled"
) )
has_team = True
participants.append(participant)
return participants return participants
...@@ -118,17 +110,21 @@ def _activate_thread_definition_milestones( ...@@ -118,17 +110,21 @@ def _activate_thread_definition_milestones(
This method handles proper milestone activation for milestones specified in This method handles proper milestone activation for milestones specified in
EmailAddress EmailAddress
""" """
team_email_sent = ( # Instructor emails should not activate milestones
thread.emails.filter(sender__definition_address__isnull=True).count() if sender.is_definition():
> 0 return
)
if team_email_sent or sender.is_definition(): # If a team already sent an email to this thread, do not activate milestones again
team_email_sent = thread.emails.filter(
sender__definition_address__isnull=True
).exists()
if team_email_sent:
return return
if ( for definition_participant in thread.participants.filter(
definition_participant := thread.get_definition_participant() definition_address__isnull=False
) is not None: ).select_related("definition_address__control"):
_update_thread_milestones( _update_thread_milestones(
thread, definition_participant.definition_address.control thread, definition_participant.definition_address.control
) )
...@@ -147,9 +143,7 @@ class EmailClient: ...@@ -147,9 +143,7 @@ class EmailClient:
) )
new_thread.participants.add(*participants) new_thread.participants.add(*participants)
SubscriptionHandler.broadcast_email_thread( SubscriptionHandler.broadcast_email_thread(new_thread, exercise)
new_thread, participants[0].exercise
)
return new_thread return new_thread
@staticmethod @staticmethod
......
...@@ -353,17 +353,20 @@ class Query(graphene.ObjectType): ...@@ -353,17 +353,20 @@ class Query(graphene.ObjectType):
@protected(User.AuthGroup.INSTRUCTOR) @protected(User.AuthGroup.INSTRUCTOR)
def resolve_thread_templates( def resolve_thread_templates(
self, info, thread_id: int self, info, thread_id: int
) -> List[EmailTemplate]: ) -> QuerySet[EmailTemplate]:
thread = get_model(EmailThread, id=thread_id) thread = get_model(EmailThread, id=thread_id)
exercise_access(info.context, thread.exercise_id) exercise_access(info.context, thread.exercise_id)
definition_participant = thread.get_definition_participant() definition_participants = thread.participants.filter(
if definition_participant is None: definition_address__isnull=False
return [] ).select_related("definition_address")
return list( templates = EmailTemplate.objects.none()
definition_participant.get_definition_address().templates.all()
) for participant in definition_participants:
templates |= participant.definition_address.templates.all()
return templates
@protected(User.AuthGroup.INSTRUCTOR) @protected(User.AuthGroup.INSTRUCTOR)
def resolve_thread_template(self, info, template_id: str) -> EmailTemplate: def resolve_thread_template(self, info, template_id: str) -> EmailTemplate:
......
...@@ -128,22 +128,15 @@ class EmailTests(TestCase): ...@@ -128,22 +128,15 @@ class EmailTests(TestCase):
participants, self.subject, str(self.exercise.id) participants, self.subject, str(self.exercise.id)
) )
participants.append("team-1@mail.com")
participants.append("nonexistentaddress@mail.com") participants.append("nonexistentaddress@mail.com")
with self.assertRaises(ModelNotFoundException) as _:
EmailClient.create_thread(
participants, self.subject, str(self.exercise.id)
)
# multiple definition addresses
participants[1] = "test@mail.ex"
with self.assertRaises(RunningExerciseOperationException) as _: with self.assertRaises(RunningExerciseOperationException) as _:
EmailClient.create_thread( EmailClient.create_thread(
participants, self.subject, str(self.exercise.id) participants, self.subject, str(self.exercise.id)
) )
_disable_feature(self.exercise, Feature.EMAIL_BETWEEN_TEAMS) _disable_feature(self.exercise, Feature.EMAIL_BETWEEN_TEAMS)
participants[1] = "team-1@mail.com" participants[0] = "team-2@mail.com"
participants.append("team-2@mail.com")
with self.assertRaises(RunningExerciseOperationException) as _: with self.assertRaises(RunningExerciseOperationException) as _:
EmailClient.create_thread( EmailClient.create_thread(
participants, self.subject, str(self.exercise.id) participants, self.subject, str(self.exercise.id)
...@@ -190,6 +183,17 @@ class EmailTests(TestCase): ...@@ -190,6 +183,17 @@ class EmailTests(TestCase):
self.subject, self.subject,
) )
# thread with multiple definition addresses
participants = ["team-1@mail.com", "test@mail.ex", "doe@mail.ex"]
thread = EmailClient.create_thread(
participants, self.subject, str(self.exercise.id)
)
self._compare_expected_participants(
participants, thread.participants.all()
)
self.assertEqual(thread.subject, self.subject)
def test_get_thread(self): def test_get_thread(self):
# each combination consist of a definition and a team address, # each combination consist of a definition and a team address,
# the team address is first # the team address is first
...@@ -355,8 +359,11 @@ class EmailTests(TestCase): ...@@ -355,8 +359,11 @@ class EmailTests(TestCase):
) )
def test_team_email(self): def test_team_email(self):
participants = ["doe@mail.ex", "team-1@mail.com"] participants = ["doe@mail.ex", "test@mail.ex", "team-1@mail.com"]
milestone = self.definition.milestones.get(name="email_doe") email_doe = self.definition.milestones.get(name="email_doe")
email_sent_to_test = self.definition.milestones.get(
name="email_sent_to_test"
)
team_sender = self.participants["team-1@mail.com"] team_sender = self.participants["team-1@mail.com"]
definition_sender = self.participants["doe@mail.ex"] definition_sender = self.participants["doe@mail.ex"]
email_channel = self.definition.channels.get(type=InjectTypes.EMAIL) email_channel = self.definition.channels.get(type=InjectTypes.EMAIL)
...@@ -365,7 +372,10 @@ class EmailTests(TestCase): ...@@ -365,7 +372,10 @@ class EmailTests(TestCase):
) )
# check that milestones from the definition address were not activated on thread create # check that milestones from the definition address were not activated on thread create
self.assertFalse(is_milestone_reached(team_sender.team, milestone)) self.assertFalse(is_milestone_reached(team_sender.team, email_doe))
self.assertFalse(
is_milestone_reached(team_sender.team, email_sent_to_test)
)
file_info = self.definition.files.first() file_info = self.definition.files.first()
action_logs = EmailClient.send_email( action_logs = EmailClient.send_email(
create_input( create_input(
...@@ -392,7 +402,10 @@ class EmailTests(TestCase): ...@@ -392,7 +402,10 @@ class EmailTests(TestCase):
) )
# check that after first email was sent the thread milestones were not activated # check that after first email was sent the thread milestones were not activated
# because it was not sent by a team # because it was not sent by a team
self.assertFalse(is_milestone_reached(team_sender.team, milestone)) self.assertFalse(is_milestone_reached(team_sender.team, email_doe))
self.assertFalse(
is_milestone_reached(team_sender.team, email_sent_to_test)
)
action_logs = EmailClient.send_email( action_logs = EmailClient.send_email(
create_input( create_input(
...@@ -411,9 +424,12 @@ class EmailTests(TestCase): ...@@ -411,9 +424,12 @@ class EmailTests(TestCase):
f"'{sent_email.sender=}' and '{team_sender}'", f"'{sent_email.sender=}' and '{team_sender}'",
) )
self.assertTrue(is_milestone_reached(team_sender.team, milestone)) self.assertTrue(is_milestone_reached(team_sender.team, email_doe))
self.assertTrue(
is_milestone_reached(team_sender.team, email_sent_to_test)
)
deactivate_milestones(team_sender.team, [milestone]) deactivate_milestones(team_sender.team, [email_doe, email_sent_to_test])
action_logs = EmailClient.send_email( action_logs = EmailClient.send_email(
create_input( create_input(
...@@ -426,4 +442,7 @@ class EmailTests(TestCase): ...@@ -426,4 +442,7 @@ class EmailTests(TestCase):
self._check_action_logs_channel(action_logs, email_channel) self._check_action_logs_channel(action_logs, email_channel)
# check that further emails do not activate the thread milestones again # check that further emails do not activate the thread milestones again
self.assertFalse(is_milestone_reached(team_sender.team, milestone)) self.assertFalse(is_milestone_reached(team_sender.team, email_doe))
self.assertFalse(
is_milestone_reached(team_sender.team, email_sent_to_test)
)
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment