2018-03-29 13:53:34 +00:00
|
|
|
"""
|
|
|
|
Description: Use conditions to kill a stream
|
|
|
|
Author: Blacktwin
|
|
|
|
Requires: requests
|
|
|
|
|
|
|
|
Enabling Scripts in Tautulli:
|
|
|
|
Taultulli > Settings > Notification Agents > Add a Notification Agent > Script
|
|
|
|
|
|
|
|
Configuration:
|
|
|
|
Taultulli > Settings > Notification Agents > New Script > Configuration:
|
|
|
|
|
|
|
|
Script Name: kill_stream
|
|
|
|
Set Script Timeout: {timeout}
|
2018-06-11 19:57:29 +00:00
|
|
|
Description: Kill stream(s)
|
2018-03-29 13:53:34 +00:00
|
|
|
Save
|
|
|
|
|
|
|
|
Triggers:
|
|
|
|
Taultulli > Settings > Notification Agents > New Script > Triggers:
|
|
|
|
|
|
|
|
Check: {trigger}
|
|
|
|
Save
|
|
|
|
|
|
|
|
Conditions:
|
|
|
|
Taultulli > Settings > Notification Agents > New Script > Conditions:
|
|
|
|
|
|
|
|
Set Conditions: [{condition} | {operator} | {value} ]
|
|
|
|
Save
|
|
|
|
|
|
|
|
Script Arguments:
|
|
|
|
Taultulli > Settings > Notification Agents > New Script > Script Arguments:
|
|
|
|
|
|
|
|
Select: Playback Start, Playback Pause
|
2018-06-15 12:32:28 +00:00
|
|
|
Arguments: --jbop SELECTOR --userId {user_id} --username {username}
|
2018-06-17 05:26:58 +00:00
|
|
|
--sessionId {session_id} --notify notifierID
|
2018-06-17 03:13:25 +00:00
|
|
|
--killMessage Your message here. No quotes.
|
2018-03-29 13:53:34 +00:00
|
|
|
|
|
|
|
Save
|
|
|
|
Close
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
import requests
|
2018-06-11 19:57:29 +00:00
|
|
|
import argparse
|
2018-03-29 13:53:34 +00:00
|
|
|
import sys
|
|
|
|
import os
|
|
|
|
|
2018-06-12 19:51:08 +00:00
|
|
|
TAUTULLI_URL = ''
|
|
|
|
TAUTULLI_APIKEY = ''
|
|
|
|
TAUTULLI_URL = os.getenv('TAUTULLI_URL', TAUTULLI_URL)
|
|
|
|
TAUTULLI_APIKEY = os.getenv('TAUTULLI_APIKEY', TAUTULLI_APIKEY)
|
2018-03-29 13:53:34 +00:00
|
|
|
|
2018-06-11 19:57:29 +00:00
|
|
|
SUBJECT_TEXT = "Tautulli has killed a stream."
|
2018-06-17 06:15:59 +00:00
|
|
|
BODY_TEXT = "Killed session ID '{id}'. Reason: {message}"
|
|
|
|
BODY_TEXT_USER = "Killed {user}'s stream. Reason: {message}."
|
2018-06-11 19:57:29 +00:00
|
|
|
|
2018-05-09 18:43:02 +00:00
|
|
|
sess = requests.Session()
|
|
|
|
# Ignore verifying the SSL certificate
|
2018-06-17 03:13:25 +00:00
|
|
|
sess.verify = False # '/path/to/certfile'
|
2018-05-09 18:43:02 +00:00
|
|
|
# If verify is set to a path to a directory,
|
2018-06-17 03:13:25 +00:00
|
|
|
# the directory must have been processed using the c_rehash utility supplied
|
|
|
|
# with OpenSSL.
|
2018-06-17 05:23:32 +00:00
|
|
|
if sess.verify is False:
|
|
|
|
# Disable the warning that the request is insecure, we know that...
|
|
|
|
import urllib3
|
|
|
|
urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning)
|
2018-05-09 18:43:02 +00:00
|
|
|
|
2018-06-14 04:47:21 +00:00
|
|
|
SELECTOR = ['stream', 'allStreams']
|
2018-06-11 19:57:29 +00:00
|
|
|
|
|
|
|
|
|
|
|
def send_notification(subject_text, body_text, notifier_id):
|
2018-06-17 03:59:30 +00:00
|
|
|
"""Send a notification through Tautulli
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
subject_text : str
|
|
|
|
The text to use for the subject line of the message.
|
|
|
|
body_text : str
|
|
|
|
The text to use for the body of the notification.
|
|
|
|
notifier_id : int
|
|
|
|
Tautulli Notification Agent ID to send the notification to.
|
|
|
|
"""
|
2018-06-11 19:57:29 +00:00
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
|
|
|
'cmd': 'notify',
|
|
|
|
'notifier_id': notifier_id,
|
|
|
|
'subject': subject_text,
|
|
|
|
'body': body_text}
|
|
|
|
|
|
|
|
try:
|
2018-06-17 05:24:48 +00:00
|
|
|
r = sess.post(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
2018-06-11 19:57:29 +00:00
|
|
|
response = r.json()
|
|
|
|
|
|
|
|
if response['response']['result'] == 'success':
|
|
|
|
sys.stdout.write("Successfully sent Tautulli notification.")
|
|
|
|
else:
|
|
|
|
raise Exception(response['response']['message'])
|
|
|
|
except Exception as e:
|
2018-06-17 03:13:25 +00:00
|
|
|
sys.stderr.write(
|
|
|
|
"Tautulli API 'notify' request failed: {0}.".format(e))
|
2018-06-11 19:57:29 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
|
2018-06-17 04:01:21 +00:00
|
|
|
def get_activity():
|
|
|
|
"""Get the current activity on the PMS.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
list
|
|
|
|
The current active sessions on the Plex server.
|
|
|
|
"""
|
2018-06-11 19:57:29 +00:00
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
|
|
|
'cmd': 'get_activity'}
|
|
|
|
|
|
|
|
try:
|
2018-06-17 05:24:48 +00:00
|
|
|
req = sess.get(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
2018-06-11 19:57:29 +00:00
|
|
|
response = req.json()
|
|
|
|
|
|
|
|
res_data = response['response']['data']['sessions']
|
2018-06-17 04:01:21 +00:00
|
|
|
return res_data
|
2018-06-11 19:57:29 +00:00
|
|
|
|
|
|
|
except Exception as e:
|
2018-06-17 03:13:25 +00:00
|
|
|
sys.stderr.write(
|
|
|
|
"Tautulli API 'get_activity' request failed: {0}.".format(e))
|
2018-06-11 19:57:29 +00:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
2018-06-17 04:04:55 +00:00
|
|
|
def get_user_session_ids(user_id):
|
|
|
|
"""Get current session IDs for a specific user.
|
2018-06-17 04:01:21 +00:00
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
user_id : int
|
|
|
|
The ID of the user to grab sessions for.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
list
|
2018-06-17 04:04:55 +00:00
|
|
|
The active session IDs for the specific user ID.
|
2018-06-17 04:01:21 +00:00
|
|
|
|
|
|
|
"""
|
|
|
|
sessions = get_activity()
|
|
|
|
user_streams = [s['session_id']
|
|
|
|
for s in sessions if s['user_id'] == user_id]
|
|
|
|
return user_streams
|
|
|
|
|
|
|
|
|
2018-06-17 06:15:59 +00:00
|
|
|
def terminate_session(session_id, message, notifier=None, username=None):
|
2018-06-11 19:57:29 +00:00
|
|
|
# Stop a streaming session.
|
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
|
|
|
'cmd': 'terminate_session',
|
|
|
|
'session_id': session_id,
|
|
|
|
'message': message}
|
|
|
|
|
|
|
|
try:
|
|
|
|
req = sess.post(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
|
|
|
response = req.json()
|
|
|
|
|
|
|
|
if response['response']['result'] == 'success':
|
2018-06-17 03:13:25 +00:00
|
|
|
sys.stdout.write(
|
|
|
|
"Successfully killed Plex session: {0}.".format(session_id))
|
2018-06-17 06:15:59 +00:00
|
|
|
if notifier:
|
|
|
|
if username:
|
|
|
|
body = BODY_TEXT.format(user=username, message=message)
|
|
|
|
else:
|
|
|
|
body = BODY_TEXT.format(id=session_id, message=message)
|
|
|
|
send_notification(SUBJECT_TEXT, body, notifier)
|
2018-06-11 19:57:29 +00:00
|
|
|
else:
|
|
|
|
raise Exception(response['response']['message'])
|
|
|
|
except Exception as e:
|
2018-06-17 03:13:25 +00:00
|
|
|
sys.stderr.write(
|
|
|
|
"Tautulli API 'terminate_session' request failed: {0}.".format(e))
|
2018-06-11 19:57:29 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2018-06-17 03:13:25 +00:00
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
description="Killing Plex streams from Tautulli.")
|
2018-06-14 12:56:20 +00:00
|
|
|
parser.add_argument('--jbop', required=True, choices=SELECTOR,
|
|
|
|
help='Kill selector.\nChoices: (%(choices)s)')
|
|
|
|
parser.add_argument('--userId', type=int,
|
|
|
|
help='The unique identifier for the user.')
|
|
|
|
parser.add_argument('--username',
|
|
|
|
help='The username of the person streaming.')
|
|
|
|
parser.add_argument('--sessionId', required=True,
|
|
|
|
help='The unique identifier for the stream.')
|
|
|
|
parser.add_argument('--notify', type=int,
|
2018-06-17 03:13:25 +00:00
|
|
|
help='Notification Agent ID number to Agent to send ' +
|
|
|
|
'notification.')
|
2018-06-17 05:26:58 +00:00
|
|
|
parser.add_argument('--killMessage', nargs='+',
|
|
|
|
help='Message to send to user whose stream is killed.')
|
2018-06-11 19:57:29 +00:00
|
|
|
|
|
|
|
opts = parser.parse_args()
|
|
|
|
|
2018-06-14 12:56:20 +00:00
|
|
|
if opts.killMessage:
|
|
|
|
message = ' '.join(opts.killMessage)
|
|
|
|
else:
|
|
|
|
message = ''
|
|
|
|
|
2018-06-11 19:57:29 +00:00
|
|
|
if opts.jbop == 'stream':
|
2018-06-17 06:15:59 +00:00
|
|
|
terminate_session(opts.sessionId, message, opts.notify, opts.username)
|
2018-06-11 19:57:29 +00:00
|
|
|
elif opts.jbop == 'allStreams':
|
2018-06-17 04:04:55 +00:00
|
|
|
streams = get_user_session_ids(opts.userId)
|
2018-06-11 19:57:29 +00:00
|
|
|
for session_id in streams:
|
2018-06-17 06:15:59 +00:00
|
|
|
terminate_session(session_id, message, opts.notify, opts.username)
|