Skip to content

Commit 0d605b8

Browse files
committed
Install dj_notify service on admin machines
The dj_notify python scripts acts as a Slack webhook receiver, which forwards the notifications to the desktop notifications and plays an alert sound to notify everyone nearby.
1 parent fae3976 commit 0d605b8

File tree

8 files changed

+165
-0
lines changed

8 files changed

+165
-0
lines changed

provision-contest/ansible/admin.yml

+2
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,8 @@
4040
tags: clusterssh
4141
- role: phpstorm
4242
tags: phpstorm
43+
- role: dj_notify
44+
tags: dj_notify
4345
- role: prometheus_target_all
4446
tags: prometheus_target_all
4547
when: GRAFANA_MONITORING

provision-contest/ansible/group_vars/admin.yml

+4
Original file line numberDiff line numberDiff line change
@@ -8,3 +8,7 @@ PHP_FPM_MAX_CHILDREN: 5
88

99
DOMSERVER_SSL_CERT: /etc/ssl/certs/localhost.crt
1010
DOMSERVER_SSL_KEY: /etc/ssl/private/localhost.key
11+
12+
ALSA_DEVICE: alsa_output.pci-0000_00_1f.3-platform-skl_hda_dsp_generic.HiFi__hw_sofhdadsp__sink
13+
NOTIFICATION_SOUND: /usr/share/sounds/sound-icons/trumpet-12.wav
14+
NOTIFICATION_SOUND_VOLUME: 35536
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
from http.server import BaseHTTPRequestHandler, HTTPServer
2+
import json
3+
import subprocess
4+
import gi
5+
import os
6+
import webbrowser
7+
import subprocess
8+
import traceback
9+
gi.require_version('Notify', '0.7')
10+
from gi.repository import Notify
11+
12+
HOSTNAME = "0.0.0.0"
13+
PORT = 9999
14+
ALSA_DEVICE = os.environ['ALSA_DEVICE']
15+
NOTIFICATION_SOUND = os.environ['NOTIFICATION_SOUND']
16+
NOTIFICATION_SOUND_VOLUME = int(os.environ['NOTIFICATION_SOUND_VOLUME'])
17+
18+
19+
def on_notification_closed(notification):
20+
print(f"Notification {notification.id} closed.")
21+
22+
23+
def on_link_click(notification, action, link):
24+
webbrowser.open(link)
25+
26+
27+
def filter_notification(title, body, link):
28+
return not title.startswith("Symfony\\Component\\HttpKernel\\Exception\\NotFoundHttpException")
29+
30+
class NotifyServer(BaseHTTPRequestHandler):
31+
def create_notification(self, title, body, link):
32+
notification = Notify.Notification.new(title, body)
33+
notification.connect("closed", on_notification_closed)
34+
notification.add_action(
35+
"action_click",
36+
"View in browser",
37+
on_link_click,
38+
link
39+
)
40+
notification.show()
41+
42+
43+
def notification_sound(self, sound):
44+
# Use Popen to launch a non-blocking background process
45+
subprocess.Popen(["paplay", "--volume", str(NOTIFICATION_SOUND_VOLUME), "--device", ALSA_DEVICE, sound])
46+
47+
48+
def do_POST(self):
49+
length = int(self.headers.get('Content-Length'))
50+
body = self.rfile.read(length)
51+
content = json.loads(body)
52+
print(json.dumps(content, indent=2))
53+
54+
att = content['attachments'][0]
55+
title = att['title']
56+
link = att['title_link']
57+
body = att['text']
58+
59+
if filter_notification(title, body, link):
60+
try:
61+
self.create_notification(title, body, link)
62+
except Exception:
63+
print(traceback.format_exc())
64+
try:
65+
self.notification_sound(NOTIFICATION_SOUND)
66+
except Exception:
67+
print(traceback.format_exc())
68+
69+
self.send_response(200)
70+
self.send_header("Content-Type", "text/plain")
71+
self.end_headers()
72+
73+
self.wfile.write(bytes("ok", "utf-8"))
74+
75+
76+
Notify.init("DOMjudge notifications")
77+
server = HTTPServer((HOSTNAME, PORT), NotifyServer)
78+
79+
try:
80+
server.serve_forever()
81+
except KeyboardInterrupt:
82+
pass
83+
84+
# Clean up
85+
server.server_close()
86+
Notify.uninit()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
- name: Restart dj_notify
3+
systemd:
4+
name: dj_notify
5+
enabled: true
6+
state: restarted
7+
daemon_reload: true
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
---
2+
# These tasks install the DOMjudge Notify script
3+
4+
- name: Install dj_notify
5+
copy:
6+
src: "dj_notify.py"
7+
dest: "/home/domjudge/bin/dj_notify.py"
8+
owner: domjudge
9+
group: domjudge
10+
mode: 0755
11+
notify: Restart dj_notify
12+
13+
- name: Copy dj_notify systemd unit file
14+
template:
15+
src: "dj_notify.service.j2"
16+
dest: "/etc/systemd/system/dj_notify.service"
17+
notify: Restart dj_notify
18+
19+
- name: Enable and start dj_notify
20+
systemd:
21+
name: dj_notify
22+
enabled: true
23+
state: started
24+
daemon_reload: true
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
[Unit]
2+
Description="DOMjudge Notify"
3+
After=network.target
4+
5+
[Service]
6+
Type=simple
7+
8+
Environment=ALSA_DEVICE={{ ALSA_DEVICE }}
9+
Environment=NOTIFICATION_SOUND={{ NOTIFICATION_SOUND }}
10+
Environment=DISPLAY=:0
11+
Environment=DBUS_SESSION_BUS_ADDRESS=unix:path=/run/user/1001/bus
12+
Environment=PULSE_SERVER=/run/user/1001/pulse/native
13+
WorkingDirectory=/home/domjudge
14+
ExecStart=/usr/bin/python3 -u /home/domjudge/bin/dj_notify.py
15+
User=domjudge
16+
17+
Restart=always
18+
RestartSec=3
19+
20+
[Install]
21+
WantedBy=graphical.target
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,3 @@
11
GLITCHTIP_PORT: 8000
22
GLITCHTIP_TOKEN:
3+
GLITCHTIP_WEBHOOK_DOMAIN: domjudge-ccsadmin2

provision-contest/ansible/roles/glitchtip/tasks/main.yml

+20
Original file line numberDiff line numberDiff line change
@@ -150,6 +150,26 @@
150150
body_format: json
151151
loop: "{{ ['setup-phase'] + groups['domserver'] }}"
152152

153+
- name: Create project dj_notify webhook
154+
when: glitchtip_proj.json | community.general.json_query("[?name=='{{ item }}']") == []
155+
ansible.builtin.uri:
156+
method: POST
157+
return_content: yes
158+
url: "http://localhost:{{ GLITCHTIP_PORT }}/api/0/projects/domjudge/{{ item }}/alerts/"
159+
status_code: 201
160+
headers:
161+
Authorization: "Bearer {{ GLITCHTIP_TOKEN }}"
162+
body:
163+
name: "dj_notify"
164+
alertRecipients:
165+
- recipientType: "webhook"
166+
url: "http://{{ GLITCHTIP_WEBHOOK_DOMAIN }}:9999/"
167+
timespanMinutes: 1
168+
quantity: 1
169+
uptime: true
170+
body_format: json
171+
loop: "{{ ['setup-phase'] + groups['domserver'] }}"
172+
153173
- name: Check for existing monitors
154174
ansible.builtin.uri:
155175
method: GET

0 commit comments

Comments
 (0)