2018-03-19 15:35:48 +00:00
|
|
|
"""
|
2017-09-10 20:15:54 +00:00
|
|
|
Delay Notification Agent message for concurrent streams
|
|
|
|
|
2018-03-19 16:33:44 +00:00
|
|
|
Arguments passed from Tautulli
|
2017-09-10 20:15:54 +00:00
|
|
|
-u {user} -srv {server_name}
|
|
|
|
You can add more arguments if you want more details in the email body
|
|
|
|
|
2018-03-19 16:33:44 +00:00
|
|
|
Adding to Tautulli
|
|
|
|
Tautulli > Settings > Notification Agents > Scripts > Bell icon:
|
2017-09-10 20:15:54 +00:00
|
|
|
[X] Notify on concurrent streams
|
2018-03-19 16:33:44 +00:00
|
|
|
Tautulli > Settings > Notification Agents > Scripts > Gear icon:
|
2017-09-10 20:15:54 +00:00
|
|
|
User Concurrent Streams: notify_delay.py
|
|
|
|
|
2018-03-19 16:33:44 +00:00
|
|
|
Tautulli Settings > Notification Agents > Scripts (Gear) > Script Timeout: 0 to disable or set to > 180
|
2018-03-19 15:35:48 +00:00
|
|
|
"""
|
2017-09-10 20:15:54 +00:00
|
|
|
|
|
|
|
import requests
|
|
|
|
import sys
|
|
|
|
import argparse
|
|
|
|
from time import sleep
|
|
|
|
|
|
|
|
## EDIT THESE SETTINGS ##
|
2018-03-19 16:33:44 +00:00
|
|
|
TAUTULLI_APIKEY = '' # Your Tautulli API key
|
|
|
|
TAUTULLI_URL = 'http://localhost:8181/' # Your Tautulli URL
|
2017-09-10 20:15:54 +00:00
|
|
|
CONCURRENT_TOTAL = 2
|
|
|
|
TIMEOUT = 180
|
|
|
|
INTERVAL = 20
|
|
|
|
|
2018-03-19 16:33:44 +00:00
|
|
|
NOTIFIER_ID = 10 # Notification notifier ID for Tautulli
|
2017-09-10 20:15:54 +00:00
|
|
|
# Find Notification agent ID here:
|
|
|
|
# https://github.com/JonnyWong16/plexpy/blob/master/API.md#notify
|
|
|
|
|
|
|
|
SUBJECT_TEXT = 'Concurrent Streams from {p.user} on {p.plex_server}'
|
|
|
|
BODY_TEXT = """\
|
|
|
|
<html>
|
|
|
|
<head></head>
|
|
|
|
<body>
|
|
|
|
<p>Hi!<br>
|
|
|
|
{p.user} has had {total} concurrent streams for longer than {time} minutes.
|
|
|
|
</p>
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
2018-03-19 16:33:44 +00:00
|
|
|
def get_activity():
|
2017-09-10 20:17:20 +00:00
|
|
|
# Get the current activity on the PMS.
|
2018-03-19 16:33:44 +00:00
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
2017-09-10 20:15:54 +00:00
|
|
|
'cmd': 'get_activity'}
|
|
|
|
|
|
|
|
try:
|
2018-03-19 16:33:44 +00:00
|
|
|
r = requests.get(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
2017-09-10 20:15:54 +00:00
|
|
|
response = r.json()
|
|
|
|
res_data = response['response']['data']['sessions']
|
|
|
|
return [d['user'] for d in res_data]
|
|
|
|
|
|
|
|
except Exception as e:
|
2018-03-19 16:33:44 +00:00
|
|
|
sys.stderr.write("Tautulli API 'get_activity' request failed: {0}.".format(e))
|
2017-09-10 20:15:54 +00:00
|
|
|
pass
|
|
|
|
|
2018-03-19 15:35:48 +00:00
|
|
|
|
|
|
|
def send_notification(subject_text, body_text):
|
2017-09-10 20:15:54 +00:00
|
|
|
# Format notification text
|
|
|
|
try:
|
2018-03-19 15:35:48 +00:00
|
|
|
subject = subject_text.format(p=p, total=cc_total)
|
|
|
|
body = body_text.format(p=p, total=cc_total, time=TIMEOUT / 60)
|
2017-09-10 20:15:54 +00:00
|
|
|
|
|
|
|
except LookupError as e:
|
|
|
|
sys.stderr.write("Unable to substitute '{0}' in the notification subject or body".format(e))
|
|
|
|
return None
|
2018-03-19 16:33:44 +00:00
|
|
|
# Send the notification through Tautulli
|
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
2017-09-10 20:15:54 +00:00
|
|
|
'cmd': 'notify',
|
2018-03-19 15:35:48 +00:00
|
|
|
'notifier_id': NOTIFIER_ID,
|
2017-09-10 20:15:54 +00:00
|
|
|
'subject': subject,
|
|
|
|
'body': body}
|
|
|
|
|
|
|
|
try:
|
2018-03-19 16:33:44 +00:00
|
|
|
r = requests.post(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
2017-09-10 20:15:54 +00:00
|
|
|
response = r.json()
|
|
|
|
|
|
|
|
if response['response']['result'] == 'success':
|
2018-03-19 16:33:44 +00:00
|
|
|
sys.stdout.write("Successfully sent Tautulli notification.")
|
2017-09-10 20:15:54 +00:00
|
|
|
else:
|
|
|
|
raise Exception(response['response']['message'])
|
|
|
|
except Exception as e:
|
2018-03-19 16:33:44 +00:00
|
|
|
sys.stderr.write("Tautulli API 'notify' request failed: {0}.".format(e))
|
2017-09-10 20:15:54 +00:00
|
|
|
return None
|
|
|
|
|
2018-03-19 15:35:48 +00:00
|
|
|
|
2017-09-10 20:15:54 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
|
|
|
|
parser.add_argument('-u', '--user', action='store', default='',
|
|
|
|
help='Username of the person watching the stream')
|
|
|
|
parser.add_argument('-srv', '--plex_server', action='store', default='',
|
|
|
|
help='The name of the Plex server')
|
|
|
|
|
|
|
|
p = parser.parse_args()
|
|
|
|
|
|
|
|
x = 0
|
|
|
|
while x < TIMEOUT and x is not None:
|
|
|
|
# check if user still has concurrent streams
|
|
|
|
print('Checking concurrent stream count.')
|
2018-03-19 16:33:44 +00:00
|
|
|
cc_total = get_activity().count(p.user)
|
2017-09-10 20:15:54 +00:00
|
|
|
if cc_total >= CONCURRENT_TOTAL:
|
|
|
|
print('{p.user} still has {total} concurrent streams.'.format(p=p, total=cc_total))
|
|
|
|
sleep(INTERVAL)
|
|
|
|
x += INTERVAL
|
|
|
|
else:
|
|
|
|
print('Exiting, user no longer has concurrent streams.')
|
|
|
|
exit()
|
|
|
|
|
|
|
|
print('Concurrent stream monitoring timeout limit has been reached. Sending notification.')
|
|
|
|
send_notification(SUBJECT_TEXT, BODY_TEXT)
|