Skip to content

Commit

Permalink
Add flag to School model to allow use of subdomains; default True
Browse files Browse the repository at this point in the history
  • Loading branch information
davepeck committed May 17, 2024
1 parent 712f148 commit 69556df
Show file tree
Hide file tree
Showing 4 changed files with 77 additions and 2 deletions.
16 changes: 14 additions & 2 deletions server/utils/email.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,11 +18,17 @@ class Domains:
aliases: tuple[str, ...]


def subdomain_of(domain: str, parent: str) -> bool:
"""Check if the `domain` is a subdomain of `parent`."""
return domain == parent or domain.endswith(f".{parent}")


def normalize_email(
address: str,
tag: str | None = "+",
dots: bool = True,
domains: Domains | None = None,
allow_subdomains: bool = True,
) -> str:
"""
Normalize an email address.
Expand All @@ -32,6 +38,7 @@ def normalize_email(
- If provided, remove the `tag` character (+) and everything after it
- If requested, remove dots (.) from the address
- If provided, replace the domain with the primary domain if it is an alias
- If requested, remove subdomains from the domain
You must have previously validated the email address.
Expand All @@ -45,8 +52,13 @@ def normalize_email(
local = local.split(tag, maxsplit=1)[0]
if dots:
local = local.replace(".", "")
if domains and domain in domains.aliases:
domain = domains.primary
if domains:
if allow_subdomains and any(
subdomain_of(domain, d) for d in [domains.primary, *domains.aliases]
):
domain = domains.primary
elif not allow_subdomains and domain in domains.aliases:
domain = domains.primary
# FORCE ascii for now (yes, this is absurd).
local = local.encode("ascii", "ignore").decode("ascii")
domain = domain.encode("ascii", "ignore").decode("ascii")
Expand Down
40 changes: 40 additions & 0 deletions server/utils/test_email.py
Original file line number Diff line number Diff line change
Expand Up @@ -77,3 +77,43 @@ def test_force_ascii(self):
expected = "[email protected]"
result = e.normalize_email(email)
self.assertEqual(result, expected)

def test_allow_subdomains(self):
"""Test an email address with a subdomain."""
email = "[email protected]"
expected = "[email protected]"
domains = e.Domains("example.com", ())
result = e.normalize_email(email, domains=domains, allow_subdomains=True)
self.assertEqual(result, expected)

def test_allow_submdomains_invalid(self):
"""Test an email address with a non-subdomain."""
email = "[email protected]"
expected = "[email protected]"
domains = e.Domains("example.com", ())
result = e.normalize_email(email, domains=domains, allow_subdomains=True)
self.assertEqual(result, expected)

def test_allow_subdomains_aliases(self):
"""Test an email address with domain aliases and subdomains."""
email = "[email protected]"
expected = "[email protected]"
domains = e.Domains("example.com", ("example.edu",))
result = e.normalize_email(email, domains=domains, allow_subdomains=True)
self.assertEqual(result, expected)

def test_disallow_subdomains(self):
"""Test an email address with a subdomain."""
email = "[email protected]"
expected = "[email protected]"
domains = e.Domains("example.com", ())
result = e.normalize_email(email, domains=domains, allow_subdomains=False)
self.assertEqual(result, expected)

def test_disallow_subdomains_aliases(self):
"""Test an email address with domain aliases and subdomains."""
email = "[email protected]"
expected = "[email protected]"
domains = e.Domains("example.com", ("example.edu",))
result = e.normalize_email(email, domains=domains, allow_subdomains=False)
self.assertEqual(result, expected)
18 changes: 18 additions & 0 deletions server/vb/migrations/0013_add_subdomains_flag.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
# Generated by Django 5.0.3 on 2024-05-17 17:20

from django.db import migrations, models


class Migration(migrations.Migration):

dependencies = [
('vb', '0012_add_percent_voted'),
]

operations = [
migrations.AddField(
model_name='school',
name='allow_subdomains',
field=models.BooleanField(default=True, help_text='Whether to allow arbitrary subdomains in school emails.'),
),
]
5 changes: 5 additions & 0 deletions server/vb/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,10 @@ class School(models.Model):
default=True,
help_text="Whether to remove dots from the local part of school emails.",
) # noqa
allow_subdomains = models.BooleanField(
default=True,
help_text="Whether to allow arbitrary subdomains in school emails.",
)

logo: "Logo"
contests: "ContestManager"
Expand All @@ -64,6 +68,7 @@ def normalize_email(self, address: str) -> str:
tag=self.mail_tag if self.mail_tag else None,
dots=self.mail_dots,
domains=domains,
allow_subdomains=self.allow_subdomains,
)

def hash_email(self, address: str) -> str:
Expand Down

0 comments on commit 69556df

Please sign in to comment.