-
-
Notifications
You must be signed in to change notification settings - Fork 15
/
sabnzbd-notify.py
executable file
·254 lines (226 loc) · 7.11 KB
/
sabnzbd-notify.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
#!/usr/bin/env python
# -*- encoding: utf-8 -*-
#
# Notifification wrapper for SABnzbd
#
# Copyright (C) 2014-2022 Chris Caron <[email protected]>
#
# This file adds support for SABnzbd v1.1.0 or higher
#
# NZB-Notify is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 2 of the License, or
# (at your option) any later version.
#
# NZB-Notify is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with NZB-Notify. If not, see <http://www.gnu.org/licenses/>.
import subprocess
import sys
import os
try:
from signal import SIGKILL
except ImportError:
# Windows 10 with 2.7.10 gave this error to a user for some reason
# hopefully this will handle this situation
SIGKILL = 9
from os import getpid
from os import kill
from os.path import abspath
from os.path import dirname
from os.path import join
from time import sleep
from datetime import datetime
from datetime import timedelta
import logging
import logging.handlers
# Logging Support
logger = logging.getLogger("sabnzbd-notify - %s" % getpid())
logging.raiseExceptions = 0
logger.setLevel(logging.INFO)
h1 = logging.StreamHandler(sys.stdout)
h1.setFormatter(logging. \
Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s"))
logger.addHandler(h1)
# Path to notify script
NOTIFY_SCRIPT = join(abspath(dirname(__file__)), 'Notify.py')
# Just some timeout value
NOTIFY_MAX_WAIT_TIME_SEC = 300
# This will be what I'll need you users to toggle if you thing you've
# found a bug you need me to solve. Send your logs to [email protected]
# or i'll really have no idea what's going on.
DEBUG_MODE = False
# A mapping of possible notifications to their respected image. This
# is only referenced by notifications that support this feature.
SABNZBD_NOTIFICATION_MAP = {
# Startup/Shutdown
'startup': (
'Startup/Shutdown',
'info',
),
# Pause/Resume
'pause_resume': (
'Pause/Resume',
'info',
),
# Added NZB
'download': (
'Added NZB',
'info',
),
# Post-processing started
'pp': (
'Post-Processing Started',
'info',
),
# Job finished
'complete': (
'Job Finished',
'success',
),
# Job failed
'failed': (
'Job Failed',
'failure',
),
# Warning
'warning': (
'Warning',
'warning',
),
# Error
'error': (
'Error',
'failure',
),
# Disk full
'disk_full': (
'Disk Full',
'warning',
),
# Queue finished
'queue_done': (
'Queue Finished',
'info',
),
# User logged in
'new_login': (
'User Logged In',
'info',
),
# Other Messages
'other': (
'Other Messages',
'info',
),
}
def syntax():
"""
A simple script to print the syntax to the end user
"""
return 'Syntax: sabnzbd-notify.py <Type> <Title> <Message> '+\
'url1[,url2[,urlN]]' +\
os.linesep + '* The <Type> can be one of the following:' +\
os.linesep + os.linesep.join(['\t%s: %s' % (k, v[0]) \
for (k,v) in SABNZBD_NOTIFICATION_MAP.items()]) +\
os.linesep + "* The <Title> and <Message> are self explanitory. "+\
"If the <Title> is however left" + os.linesep +"\tblank, then the "+\
"description of the <Type> is used instead." + os.linesep +\
"* All remaining arguments are treated as URLs. You can also " +\
"delimit multiple" + os.linesep + "\tURLs in a single " +\
"string/argument with the use of a comma (,)." + os.linesep +\
"\tURLs can also be provided as the SAB_NOTIFICATION_PARAMETERS " +\
"environment variable."
def notify(ntype, title, body, urls, debug=None):
"""
A callable function so SABnzbd can import this file and just
call the notifications through here if they wish.
"""
if debug is None:
debug = DEBUG_MODE
# We use the Python interpreter that was also used by
# SABnzbd when it executed this script
cmd = [
sys.executable if sys.executable else "python",
NOTIFY_SCRIPT,
'-t', title,
'-b', body,
'-T', 'sabnzbd',
'-n', SABNZBD_NOTIFICATION_MAP[ntype][1],
'-s', urls,
]
if debug:
# Debug Mode Enabled
cmd.append('-D')
# Execute our Process
p1 = subprocess.Popen(cmd)
## Calculate Wait Time
max_wait_time = datetime.utcnow() + \
timedelta(seconds=NOTIFY_MAX_WAIT_TIME_SEC)
while p1.poll() == None:
if datetime.utcnow() >= max_wait_time:
logger.error("Process aborted (took too long)")
try:
kill(p1.pid, SIGKILL)
except:
pass
## CPU Throttle
sleep(0.8)
if p1.poll() == None:
## Safety
try:
kill(p1.pid, SIGKILL)
except:
pass
## Ensure execution leaves memory
p1.wait()
if p1.returncode not in (0, 93):
# 93 is a return value recognized by NZBGet as 'good'
# all systems okay; we want to translate this back to
# standard shell scripting responses if we get anything
# outside of what is identified above
return False
return True
if __name__ == "__main__":
# Simple parsing of the command line and env variable
notify_urls_string = os.environ.get("SAB_NOTIFICATION_PARAMETERS", None)
if len(sys.argv) <= 3 and not notify_urls_string:
logger.error('Not enough arguments specified.')
print(syntax())
exit(1)
# Make sure our Notify.py script is available to us
if not (os.path.isfile(NOTIFY_SCRIPT) and \
os.access(NOTIFY_SCRIPT, os.X_OK)):
logger.error('The engine %s was not found!' % NOTIFY_SCRIPT)
exit(1)
# Parse our arguments to make sure they're valid
notify_type = sys.argv[1].strip().lower()
if notify_type not in SABNZBD_NOTIFICATION_MAP.keys():
logger.error('Invalid <Type> specified (argument 1)')
print(syntax())
exit(1)
# Take on specified title
notify_title = sys.argv[2].strip()
if not notify_title:
notify_title = SABNZBD_NOTIFICATION_MAP[notify_type][0]
# Store body (empty or not)
notify_body = sys.argv[3].strip()
# Environment variable takes precedence over the command line parameter
# The URLs are complex and very depending on what we're notifying
# so we'll let Notify.py take care of them at this point.
if notify_urls_string:
notify_urls = notify_urls_string.split()
else:
notify_urls = sys.argv[4:]
notify_urls = ','.join([v.strip() for v in notify_urls])
# Perform Notification
exit(int(not notify(
ntype=notify_type,
title=notify_title,
body=notify_body,
urls=notify_urls,
)))