Skip to content

Commit eb15c93

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 bda0c50 commit eb15c93

File tree

8 files changed

+158
-0
lines changed

8 files changed

+158
-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
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
ALSA_DEVICE: alsa_output.pci-0000_00_1f.3-platform-skl_hda_dsp_generic.HiFi__hw_sofhdadsp__sink
2+
NOTIFICATION_SOUND: /usr/share/sounds/sound-icons/trumpet-12.wav
3+
NOTIFICATION_SOUND_VOLUME: 35536
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,87 @@
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+
31+
class NotifyServer(BaseHTTPRequestHandler):
32+
def create_notification(self, title, body, link):
33+
notification = Notify.Notification.new(title, body)
34+
notification.connect("closed", on_notification_closed)
35+
notification.add_action(
36+
"action_click",
37+
"View in browser",
38+
on_link_click,
39+
link
40+
)
41+
notification.show()
42+
43+
44+
def notification_sound(self, sound):
45+
# Use Popen to launch a non-blocking background process
46+
subprocess.Popen(["paplay", "--volume", str(NOTIFICATION_SOUND_VOLUME), "--device", ALSA_DEVICE, sound])
47+
48+
49+
def do_POST(self):
50+
length = int(self.headers.get('Content-Length'))
51+
body = self.rfile.read(length)
52+
content = json.loads(body)
53+
print(json.dumps(content, indent=2))
54+
55+
att = content['attachments'][0]
56+
title = att['title']
57+
link = att['title_link']
58+
body = att['text']
59+
60+
if filter_notification(title, body, link):
61+
try:
62+
self.create_notification(title, body, link)
63+
except Exception:
64+
print(traceback.format_exc())
65+
try:
66+
self.notification_sound(NOTIFICATION_SOUND)
67+
except Exception:
68+
print(traceback.format_exc())
69+
70+
self.send_response(200)
71+
self.send_header("Content-Type", "text/plain")
72+
self.end_headers()
73+
74+
self.wfile.write(bytes("ok", "utf-8"))
75+
76+
77+
Notify.init("DOMjudge notifications")
78+
server = HTTPServer((HOSTNAME, PORT), NotifyServer)
79+
80+
try:
81+
server.serve_forever()
82+
except KeyboardInterrupt:
83+
pass
84+
85+
# Clean up
86+
server.server_close()
87+
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,17 @@
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
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)