Skip to content

Commit f219e26

Browse files
committed
feat: add push notifications
1 parent 1888e89 commit f219e26

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

53 files changed

+2196
-180
lines changed

.gitignore

+7
Original file line numberDiff line numberDiff line change
@@ -56,3 +56,10 @@ package-lock.json
5656
# Virtual environments
5757
venv/
5858
.venv/
59+
60+
# Webpush
61+
/keys/webpush/
62+
/keys/
63+
64+
# Keys
65+
*.pem

config/docker/initial_setup.sh

+3
Original file line numberDiff line numberDiff line change
@@ -49,3 +49,6 @@ python3 -u manage.py import_sports $(date +%m)
4949

5050
echo -e "${BLUE}${BOLD}Creating CSL apps...${CLEAR}"
5151
python3 -u manage.py dev_create_cslapps
52+
53+
echo -e "${BLUE}${BOLD}Generating vapid keys...${CLEAR}"
54+
python3 create_vapid_keys.py

config/scripts/create_vapid_keys.py

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
import base64
2+
import os
3+
4+
from cryptography.hazmat.primitives.serialization import Encoding, PublicFormat
5+
from py_vapid import Vapid
6+
7+
PROJECT_ROOT = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
8+
os.makedirs(os.path.join(PROJECT_ROOT, "keys", "webpush"))
9+
10+
# Generate VAPID key pair
11+
vapid = Vapid()
12+
vapid.generate_keys()
13+
14+
# Get public and private keys for the vapid key pair
15+
vapid.save_public_key(os.path.join(PROJECT_ROOT, "keys", "webpush", "public_key.pem"))
16+
public_key_bytes = vapid.public_key.public_bytes(Encoding.X962, PublicFormat.UncompressedPoint)
17+
18+
vapid.save_key(os.path.join(PROJECT_ROOT, "keys", "webpush", "private_key.pem"))
19+
20+
21+
# Convert the public key to applicationServerKey format
22+
application_server_key = base64.urlsafe_b64encode(public_key_bytes).replace(b"=", b"").decode("utf8")
23+
24+
with open(os.path.join(PROJECT_ROOT, "keys", "webpush", "ApplicationServerKey.key"), "w", encoding="utf-8") as f:
25+
f.write(application_server_key)

cron/eighth-absence.sh

+2-1
Original file line numberDiff line numberDiff line change
@@ -4,4 +4,5 @@
44
timestamp=$(date +"%Y-%m-%d-%H%M")
55
cd /usr/local/www/intranet3
66
./cron/env.sh ./manage.py absence_email --silent
7-
echo "Absence email sent at $timestamp." >> /var/log/ion/email.log
7+
./cron/env.sh ./manage.py absence_notify --silent
8+
echo "Absence email and push notification sent at $timestamp." >> /var/log/ion/email.log

docs/sourcedoc/intranet.apps.eighth.management.commands.rst

+8
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,14 @@ intranet.apps.eighth.management.commands.absence\_email module
1212
:undoc-members:
1313
:show-inheritance:
1414

15+
intranet.apps.eighth.management.commands.absence\_notify module
16+
---------------------------------------------------------------
17+
18+
.. automodule:: intranet.apps.eighth.management.commands.absence_notify
19+
:members:
20+
:undoc-members:
21+
:show-inheritance:
22+
1523
intranet.apps.eighth.management.commands.delete\_duplicate\_signups module
1624
--------------------------------------------------------------------------
1725

docs/sourcedoc/intranet.apps.notifications.rst

+40
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,14 @@ intranet.apps.notifications package
44
Submodules
55
----------
66

7+
intranet.apps.notifications.api module
8+
--------------------------------------
9+
10+
.. automodule:: intranet.apps.notifications.api
11+
:members:
12+
:undoc-members:
13+
:show-inheritance:
14+
715
intranet.apps.notifications.emails module
816
-----------------------------------------
917

@@ -12,6 +20,14 @@ intranet.apps.notifications.emails module
1220
:undoc-members:
1321
:show-inheritance:
1422

23+
intranet.apps.notifications.forms module
24+
----------------------------------------
25+
26+
.. automodule:: intranet.apps.notifications.forms
27+
:members:
28+
:undoc-members:
29+
:show-inheritance:
30+
1531
intranet.apps.notifications.models module
1632
-----------------------------------------
1733

@@ -20,6 +36,14 @@ intranet.apps.notifications.models module
2036
:undoc-members:
2137
:show-inheritance:
2238

39+
intranet.apps.notifications.serializers module
40+
----------------------------------------------
41+
42+
.. automodule:: intranet.apps.notifications.serializers
43+
:members:
44+
:undoc-members:
45+
:show-inheritance:
46+
2347
intranet.apps.notifications.tasks module
2448
----------------------------------------
2549

@@ -28,6 +52,14 @@ intranet.apps.notifications.tasks module
2852
:undoc-members:
2953
:show-inheritance:
3054

55+
intranet.apps.notifications.tests module
56+
----------------------------------------
57+
58+
.. automodule:: intranet.apps.notifications.tests
59+
:members:
60+
:undoc-members:
61+
:show-inheritance:
62+
3163
intranet.apps.notifications.urls module
3264
---------------------------------------
3365

@@ -36,6 +68,14 @@ intranet.apps.notifications.urls module
3668
:undoc-members:
3769
:show-inheritance:
3870

71+
intranet.apps.notifications.utils module
72+
----------------------------------------
73+
74+
.. automodule:: intranet.apps.notifications.utils
75+
:members:
76+
:undoc-members:
77+
:show-inheritance:
78+
3979
intranet.apps.notifications.views module
4080
----------------------------------------
4181

docs/sourcedoc/intranet.apps.polls.rst

+8
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,14 @@ intranet.apps.polls.models module
2828
:undoc-members:
2929
:show-inheritance:
3030

31+
intranet.apps.polls.notifications module
32+
----------------------------------------
33+
34+
.. automodule:: intranet.apps.polls.notifications
35+
:members:
36+
:undoc-members:
37+
:show-inheritance:
38+
3139
intranet.apps.polls.tests module
3240
--------------------------------
3341

intranet/apps/announcements/forms.py

+18-3
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,12 @@ def __init__(self, *args, **kwargs):
1212
super().__init__(*args, **kwargs)
1313
self.fields["expiration_date"].help_text = "By default, announcements expire after two weeks. To change this, click in the box above."
1414

15-
self.fields["notify_post"].help_text = "If this box is checked, students who have signed up for notifications will receive an email."
15+
self.fields["notify_post"].help_text = (
16+
"If this box is checked, students who have signed up for email "
17+
"notifications will receive an email "
18+
"and those who have signed up for push notifications will receive a "
19+
"push notification."
20+
)
1621

1722
self.fields["notify_email_all"].help_text = (
1823
"This will send an email notification to all of the users who can see this post. This option "
@@ -41,7 +46,12 @@ def __init__(self, *args, **kwargs):
4146
super().__init__(*args, **kwargs)
4247
self.fields["expiration_date"].help_text = "By default, announcements expire after two weeks. To change this, click in the box above."
4348

44-
self.fields["notify_post_resend"].help_text = "If this box is checked, students who have signed up for notifications will receive an email."
49+
self.fields["notify_post_resend"].help_text = (
50+
"If this box is checked, students who have signed up for email "
51+
"notifications will receive an email "
52+
"and those who have signed up for push notifications will "
53+
"receive a push notification."
54+
)
4555

4656
self.fields["notify_email_all_resend"].help_text = (
4757
"This will resend an email notification to all of the users who can see this post. This option "
@@ -105,7 +115,12 @@ class AnnouncementAdminForm(forms.Form):
105115

106116
def __init__(self, *args, **kwargs):
107117
super().__init__(*args, **kwargs)
108-
self.fields["notify_post"].help_text = "If this box is checked, students who have signed up for notifications will receive an email."
118+
self.fields["notify_post"].help_text = (
119+
"If this box is checked, students who have signed up for email "
120+
"notifications will receive an email "
121+
"and those who have signed up for push notifications will receive a "
122+
"push notification."
123+
)
109124
self.fields["notify_email_all"].help_text = (
110125
"This will send an email notification to all of the users who can see this post. This option "
111126
"does NOT take users' email notification preferences into account, so please use with care."

intranet/apps/announcements/notifications.py

+32-2
Original file line numberDiff line numberDiff line change
@@ -7,12 +7,18 @@
77
from django.contrib import messages
88
from django.contrib.auth import get_user_model
99
from django.core import exceptions
10+
from django.db.models import Q
1011
from django.urls import reverse
12+
from django.utils.html import strip_tags
13+
from push_notifications.models import WebPushDevice
1114
from requests_oauthlib import OAuth1
1215
from sentry_sdk import capture_exception
1316

1417
from ...utils.date import get_senior_graduation_year
15-
from ..notifications.tasks import email_send_task
18+
from ..notifications.tasks import email_send_task, send_bulk_notification
19+
from ..notifications.utils import truncate_content, truncate_title
20+
from ..users.models import User
21+
from .models import Announcement
1622

1723
logger = logging.getLogger(__name__)
1824

@@ -135,7 +141,7 @@ def announcement_posted_email(request, obj, send_all=False):
135141
emails.append(u.notification_email)
136142
users_send.append(u)
137143

138-
if not settings.PRODUCTION and len(emails) > 3:
144+
if not settings.PRODUCTION and len(emails) > 3 and not settings.FORCE_EMAIL_SEND:
139145
raise exceptions.PermissionDenied("You're about to email a lot of people, and you aren't in production!")
140146

141147
base_url = request.build_absolute_uri(reverse("index"))
@@ -200,3 +206,27 @@ def notify_twitter(status):
200206
req = requests.post(url, data=data, auth=auth, timeout=15)
201207

202208
return req.text
209+
210+
211+
def announcement_posted_push_notification(obj: Announcement) -> None:
212+
"""Send a (Web)push notification to users when an announcement is posted.
213+
214+
obj: The announcement object
215+
216+
"""
217+
218+
if not obj.groups.all():
219+
users = User.objects.filter(push_notification_preferences__announcement_notifications=True)
220+
devices = WebPushDevice.objects.filter(user__in=users)
221+
else:
222+
users = User.objects.filter(Q(groups__in=obj.groups.all()) & Q(push_notification_preferences__announcement_notifications=True))
223+
devices = WebPushDevice.objects.filter(user__in=users)
224+
225+
send_bulk_notification.delay(
226+
filtered_objects=devices,
227+
title=f"Announcement: {truncate_title(obj.title)} ({obj.get_author()})",
228+
body=truncate_content(strip_tags(obj.content_no_links)),
229+
data={
230+
"url": settings.PUSH_NOTIFICATIONS_BASE_URL + reverse("view_announcement", args=[obj.id]),
231+
},
232+
)

intranet/apps/announcements/views.py

+2
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
admin_request_announcement_email,
2020
announcement_approved_email,
2121
announcement_posted_email,
22+
announcement_posted_push_notification,
2223
announcement_posted_twitter,
2324
request_announcement_email,
2425
)
@@ -48,6 +49,7 @@ def announcement_posted_hook(request, obj):
4849
"""
4950
if obj.notify_post:
5051
announcement_posted_twitter(request, obj)
52+
announcement_posted_push_notification(obj)
5153
try:
5254
notify_all = obj.notify_email_all
5355
except AttributeError:

intranet/apps/api/urls.py

+12
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
from ..bus import api as bus_api
55
from ..eighth.views import api as eighth_api
66
from ..emerg import api as emerg_api
7+
from ..notifications import api as notification_api
78
from ..schedule import api as schedule_api
89
from ..users import api as users_api
910
from .views import api_root
@@ -43,4 +44,15 @@
4344
re_path(r"^/emerg$", emerg_api.emerg_status, name="api_emerg_status"),
4445
re_path(r"^/bus$", bus_api.RouteList.as_view(), name="api_bus_list"),
4546
re_path(r"^/bus/(?P<pk>\d+)$", bus_api.RouteDetail.as_view(), name="api_bus_detail"),
47+
re_path(
48+
r"^/notifications/webpush/application_key$", notification_api.GetApplicationServerKey.as_view(), name="api_get_vapid_application_server_key"
49+
),
50+
re_path(r"^/notifications/webpush/subscribe$", notification_api.WebpushSubscribeDevice.as_view(), name="api_webpush_subscribe"),
51+
re_path(r"^/notifications/webpush/unsubscribe$", notification_api.WebpushUnsubscribeDevice.as_view(), name="api_webpush_unsubscribe"),
52+
re_path(r"^/notifications/webpush/update_subscription$", notification_api.WebpushUpdateDevice.as_view(), name="api_webpush_update_subscription"),
53+
re_path(
54+
r"^/notifications/webpush/subscription_status$",
55+
notification_api.GetWebpushSubscriptionStatus.as_view(),
56+
name="api_webpush_subscription_status",
57+
),
4658
]

intranet/apps/bus/tasks.py

+57
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,12 @@
11
from celery import shared_task
22
from celery.utils.log import get_task_logger
3+
from django.urls import reverse
4+
from django.utils import timezone
35

6+
from ... import settings
7+
from ..notifications.tasks import send_notification_to_user
8+
from ..schedule.models import Day
9+
from ..users.models import User
410
from .models import Route
511

612
logger = get_task_logger(__name__)
@@ -12,3 +18,54 @@ def reset_routes() -> None:
1218

1319
for route in Route.objects.all():
1420
route.reset_status()
21+
22+
23+
@shared_task
24+
def push_bus_notifications(schedule: bool = False) -> None:
25+
if schedule:
26+
today = Day.objects.today()
27+
28+
if today is not None:
29+
dismissal = today.end_time
30+
31+
if dismissal is not None:
32+
block_datetime = dismissal.date_obj(timezone.now())
33+
block_datetime = timezone.make_aware(block_datetime, timezone.get_current_timezone())
34+
35+
push_bus_notifications.apply_async(eta=block_datetime)
36+
logger.info("Push bus notifications scheduled at %s (bus info)", str(block_datetime))
37+
38+
else:
39+
route_translations = {key: convert_dataset(value) for key, value in settings.PUSH_ROUTE_TRANSLATIONS.items()}
40+
41+
users = User.objects.filter(push_notification_preferences__bus_notifications=True)
42+
43+
for user in users:
44+
if user.bus_route.status == "d":
45+
send_notification_to_user(
46+
user=user,
47+
title="Bus Delayed",
48+
body=f"Sorry, your bus ({user.bus_route.bus_number}) has been delayed.",
49+
data={
50+
"url": settings.PUSH_NOTIFICATIONS_BASE_URL + reverse("bus"),
51+
},
52+
)
53+
else:
54+
space = user.bus_route.space
55+
if space is not None:
56+
for key, value in route_translations.items():
57+
if space in value:
58+
send_notification_to_user(
59+
user=user,
60+
title="Bus Location",
61+
body=f"Your bus is at the {key} of the parking lot.",
62+
data={
63+
"url": settings.PUSH_NOTIFICATIONS_BASE_URL + reverse("bus"),
64+
},
65+
)
66+
67+
68+
def convert_dataset(dataset):
69+
# Convert each number to the format "_number" and return as a set
70+
# because that's how the ID spots are named
71+
return {"_" + str(number) for number in dataset}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
from django.core.management.base import BaseCommand
2+
3+
from intranet.apps.eighth.models import EighthSignup
4+
from intranet.apps.eighth.notifications import absence_notification
5+
6+
7+
class Command(BaseCommand):
8+
help = "Push notify users who have an Eighth Period absence (via Webpush.)"
9+
10+
def add_arguments(self, parser):
11+
parser.add_argument("--silent", action="store_true", dest="silent", default=False, help="Be silent.")
12+
13+
parser.add_argument("--pretend", action="store_true", dest="pretend", default=False, help="Pretend, and don't actually do anything.")
14+
15+
def handle(self, *args, **options):
16+
log = not options["silent"]
17+
18+
absences = EighthSignup.objects.get_absences().filter(absence_notified=False)
19+
20+
for signup in absences:
21+
if log:
22+
self.stdout.write(str(signup))
23+
if not options["pretend"]:
24+
absence_notification(signup)
25+
signup.absence_notified = True
26+
signup.save()
27+
28+
if log:
29+
self.stdout.write("Done.")

0 commit comments

Comments
 (0)