2018-03-29 13:53:34 +00:00
|
|
|
"""
|
|
|
|
Description: Use conditions to kill a stream
|
2018-12-20 01:08:27 +00:00
|
|
|
Author: Blacktwin, Arcanemagus, Samwiseg0, JonnyWong16
|
2018-03-29 13:53:34 +00:00
|
|
|
|
2018-06-17 07:27:13 +00:00
|
|
|
Adding the script to Tautulli:
|
2018-12-20 01:08:27 +00:00
|
|
|
Tautulli > Settings > Notification Agents > Add a new notification agent >
|
2018-06-17 07:27:13 +00:00
|
|
|
Script
|
2018-03-29 13:53:34 +00:00
|
|
|
|
|
|
|
Configuration:
|
2018-12-20 01:08:27 +00:00
|
|
|
Tautulli > Settings > Notification Agents > New Script > Configuration:
|
2018-03-29 13:53:34 +00:00
|
|
|
|
2018-06-17 07:27:13 +00:00
|
|
|
Script Folder: /path/to/your/scripts
|
|
|
|
Script File: ./kill_stream.py (Should be selectable in a dropdown list)
|
|
|
|
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:
|
2018-12-20 01:08:27 +00:00
|
|
|
Tautulli > Settings > Notification Agents > New Script > Triggers:
|
2018-03-29 13:53:34 +00:00
|
|
|
|
2018-06-17 07:27:13 +00:00
|
|
|
Check: Playback Start and/or Playback Pause
|
2018-03-29 13:53:34 +00:00
|
|
|
Save
|
|
|
|
|
|
|
|
Conditions:
|
2018-12-20 01:08:27 +00:00
|
|
|
Tautulli > Settings > Notification Agents > New Script > Conditions:
|
2018-03-29 13:53:34 +00:00
|
|
|
|
|
|
|
Set Conditions: [{condition} | {operator} | {value} ]
|
|
|
|
Save
|
|
|
|
|
|
|
|
Script Arguments:
|
2018-12-20 01:08:27 +00:00
|
|
|
Tautulli > Settings > Notification Agents > New Script > Script Arguments:
|
2018-03-29 13:53:34 +00:00
|
|
|
|
|
|
|
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 06:36:27 +00:00
|
|
|
--interval 30 --limit 1200
|
2018-09-09 00:31:06 +00:00
|
|
|
--richMessage RICH_TYPE --serverName {server_name}
|
|
|
|
--plexUrl {plex_url} --posterUrl {poster_url}
|
|
|
|
--richColor '#E5A00D'
|
2018-07-10 12:17:29 +00:00
|
|
|
--killMessage 'Your message here.'
|
2018-03-29 13:53:34 +00:00
|
|
|
|
|
|
|
Save
|
|
|
|
Close
|
|
|
|
"""
|
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
|
2018-03-29 13:53:34 +00:00
|
|
|
import os
|
2018-09-09 00:31:06 +00:00
|
|
|
import sys
|
|
|
|
import json
|
|
|
|
import time
|
2018-12-20 01:57:30 +00:00
|
|
|
import argparse
|
2018-06-17 06:36:27 +00:00
|
|
|
from datetime import datetime
|
2018-12-20 01:57:30 +00:00
|
|
|
from requests import Session
|
|
|
|
from requests.adapters import HTTPAdapter
|
|
|
|
from requests.exceptions import RequestException
|
2018-09-09 00:31:06 +00:00
|
|
|
|
2018-03-29 13:53:34 +00:00
|
|
|
|
2018-06-12 19:51:08 +00:00
|
|
|
TAUTULLI_URL = ''
|
|
|
|
TAUTULLI_APIKEY = ''
|
2018-09-09 00:31:06 +00:00
|
|
|
TAUTULLI_PUBLIC_URL = ''
|
2018-06-12 19:51:08 +00:00
|
|
|
TAUTULLI_URL = os.getenv('TAUTULLI_URL', TAUTULLI_URL)
|
2018-09-09 00:31:06 +00:00
|
|
|
TAUTULLI_PUBLIC_URL = os.getenv('TAUTULLI_PUBLIC_URL', TAUTULLI_PUBLIC_URL)
|
2018-06-12 19:51:08 +00:00
|
|
|
TAUTULLI_APIKEY = os.getenv('TAUTULLI_APIKEY', TAUTULLI_APIKEY)
|
2018-06-29 20:14:13 +00:00
|
|
|
TAUTULLI_ENCODING = os.getenv('TAUTULLI_ENCODING', 'UTF-8')
|
2018-09-09 00:31:06 +00:00
|
|
|
VERIFY_SSL = False
|
|
|
|
|
|
|
|
if TAUTULLI_PUBLIC_URL != '/':
|
|
|
|
# Check to see if there is a public URL set in Tautulli
|
|
|
|
TAUTULLI_LINK = TAUTULLI_PUBLIC_URL
|
|
|
|
else:
|
|
|
|
TAUTULLI_LINK = TAUTULLI_URL
|
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
|
|
|
|
2018-12-20 03:15:45 +00:00
|
|
|
SELECTOR = ['stream', 'allStreams', 'multiIpStreams', 'paused']
|
2018-06-11 19:57:29 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
RICH_TYPE = ['discord', 'slack']
|
2018-06-11 19:57:29 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
TAUTULLI_ICON = 'https://github.com/Tautulli/Tautulli/raw/master/data/interfaces/default/images/logo-circle.png'
|
2018-06-17 03:59:30 +00:00
|
|
|
|
2018-06-11 19:57:29 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
def utc_now_iso():
|
|
|
|
"""Get current time in ISO format"""
|
|
|
|
utcnow = datetime.utcnow()
|
2018-06-11 19:57:29 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
return utcnow.isoformat()
|
2018-06-11 19:57:29 +00:00
|
|
|
|
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
def hex_to_int(value):
|
|
|
|
"""Convert hex value to integer"""
|
|
|
|
try:
|
|
|
|
return int(value, 16)
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
return 0
|
2018-06-17 04:01:21 +00:00
|
|
|
|
2018-06-11 19:57:29 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
def arg_decoding(arg):
|
|
|
|
"""Decode args, encode UTF-8"""
|
|
|
|
return arg.decode(TAUTULLI_ENCODING).encode('UTF-8')
|
2018-06-11 19:57:29 +00:00
|
|
|
|
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
def debug_dump_vars():
|
|
|
|
"""Dump parameters for debug"""
|
|
|
|
print('Tautulli URL - ' + TAUTULLI_URL)
|
|
|
|
print('Tautulli Public URL - ' + TAUTULLI_PUBLIC_URL)
|
|
|
|
print('Verify SSL - ' + str(VERIFY_SSL))
|
|
|
|
print('Tautulli API key - ' + TAUTULLI_APIKEY[-4:]
|
|
|
|
.rjust(len(TAUTULLI_APIKEY), "x"))
|
2018-06-11 19:57:29 +00:00
|
|
|
|
|
|
|
|
2018-12-20 01:57:30 +00:00
|
|
|
def get_all_streams(tautulli, user_id=None):
|
2018-09-09 00:31:06 +00:00
|
|
|
"""Get a list of all current streams.
|
2018-06-17 04:01:21 +00:00
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
user_id : int
|
|
|
|
The ID of the user to grab sessions for.
|
2018-12-20 01:57:30 +00:00
|
|
|
tautulli : obj
|
|
|
|
Tautulli object.
|
2018-06-17 04:01:21 +00:00
|
|
|
Returns
|
|
|
|
-------
|
2018-09-09 00:31:06 +00:00
|
|
|
objects
|
|
|
|
The of stream objects.
|
2018-06-17 04:01:21 +00:00
|
|
|
"""
|
2018-09-09 00:31:06 +00:00
|
|
|
sessions = tautulli.get_activity()['sessions']
|
2018-06-17 04:01:21 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
if user_id:
|
|
|
|
streams = [Stream(session=s) for s in sessions if s['user_id'] == user_id]
|
|
|
|
else:
|
|
|
|
streams = [Stream(session=s) for s in sessions]
|
2018-06-17 04:01:21 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
return streams
|
2018-06-17 06:19:30 +00:00
|
|
|
|
2018-06-11 19:57:29 +00:00
|
|
|
|
2018-12-20 01:57:30 +00:00
|
|
|
def notify(all_opts, message, kill_type=None, stream=None, tautulli=None):
|
2018-09-09 00:31:06 +00:00
|
|
|
"""Decides which notifier type to use"""
|
2018-12-20 01:57:30 +00:00
|
|
|
if all_opts.notify and all_opts.richMessage:
|
|
|
|
rich_notify(all_opts.notify, all_opts.richMessage, all_opts.richColor, kill_type,
|
|
|
|
all_opts.serverName, all_opts.plexUrl, all_opts.posterUrl, message, stream, tautulli)
|
|
|
|
elif all_opts.notify:
|
|
|
|
basic_notify(all_opts.notify, all_opts.sessionId, all_opts.username, message, stream, tautulli)
|
2018-06-11 19:57:29 +00:00
|
|
|
|
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
def rich_notify(notifier_id, rich_type, color=None, kill_type=None, server_name=None,
|
2018-12-20 01:57:30 +00:00
|
|
|
plex_url=None, poster_url=None, message=None, stream=None, tautulli=None):
|
2018-09-09 00:31:06 +00:00
|
|
|
"""Decides which rich notifier type to use. Set default values for empty variables
|
2018-06-17 06:36:27 +00:00
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
2018-09-09 00:31:06 +00:00
|
|
|
notifier_id : int
|
|
|
|
The ID of the user to grab sessions for.
|
|
|
|
rich_type : str
|
|
|
|
Contains 'discord' or 'slack'.
|
2018-12-20 01:57:30 +00:00
|
|
|
color : Union[int, str]
|
|
|
|
Hex string or integer representation of color.
|
2018-09-09 00:31:06 +00:00
|
|
|
kill_type : str
|
|
|
|
The kill type used.
|
|
|
|
server_name : str
|
|
|
|
The name of the plex server.
|
|
|
|
plex_url : str
|
|
|
|
Plex media URL.
|
|
|
|
poster_url : str
|
|
|
|
The media poster URL.
|
2018-06-17 06:36:27 +00:00
|
|
|
message : str
|
2018-09-09 00:31:06 +00:00
|
|
|
Message sent to the client.
|
|
|
|
stream : obj
|
|
|
|
Stream object.
|
2018-12-20 01:57:30 +00:00
|
|
|
tautulli : obj
|
|
|
|
Tautulli object.
|
2018-06-17 06:36:27 +00:00
|
|
|
"""
|
2018-09-09 00:31:06 +00:00
|
|
|
notification = Notification(notifier_id, SUBJECT_TEXT, BODY_TEXT, tautulli, stream)
|
2018-12-20 01:57:30 +00:00
|
|
|
# Initialize Variables
|
|
|
|
title = ''
|
|
|
|
footer = ''
|
2018-09-09 00:31:06 +00:00
|
|
|
# Set a default server_name if none is provided
|
|
|
|
if server_name is None:
|
|
|
|
server_name = 'Plex Server'
|
|
|
|
|
2018-12-20 01:08:27 +00:00
|
|
|
# Set a default color if none is provided
|
2018-09-09 00:31:06 +00:00
|
|
|
if color is None:
|
|
|
|
color = '#E5A00D'
|
|
|
|
|
2018-12-20 01:08:27 +00:00
|
|
|
# Set a default plexUrl if none is provided
|
2018-09-09 00:31:06 +00:00
|
|
|
if plex_url is None:
|
|
|
|
plex_url = 'https://app.plex.tv'
|
|
|
|
|
2018-12-20 01:08:27 +00:00
|
|
|
# Set a default posterUrl if none is provided
|
2018-09-09 00:31:06 +00:00
|
|
|
if poster_url is None:
|
|
|
|
poster_url = TAUTULLI_ICON
|
|
|
|
|
|
|
|
# Set a default message if none is provided
|
|
|
|
if message is None:
|
|
|
|
message = 'The server owner has ended the stream.'
|
|
|
|
|
|
|
|
if kill_type == 'Stream':
|
|
|
|
title = "Killed {}'s stream.".format(stream.friendly_name)
|
|
|
|
footer = '{} | Kill {}'.format(server_name, kill_type)
|
|
|
|
|
|
|
|
elif kill_type == 'Paused':
|
|
|
|
title = "Killed {}'s paused stream.".format(stream.friendly_name)
|
|
|
|
footer = '{} | Kill {}'.format(server_name, kill_type)
|
|
|
|
|
|
|
|
elif kill_type == 'All Streams':
|
|
|
|
title = "Killed {}'s stream.".format(stream.friendly_name)
|
|
|
|
footer = '{} | Kill {}'.format(server_name, kill_type)
|
|
|
|
poster_url = TAUTULLI_ICON
|
|
|
|
plex_url = 'https://app.plex.tv'
|
|
|
|
|
|
|
|
if rich_type == 'discord':
|
|
|
|
color = hex_to_int(color.lstrip('#'))
|
|
|
|
notification.send_discord(title, color, poster_url, plex_url, message, footer)
|
|
|
|
|
|
|
|
elif rich_type == 'slack':
|
|
|
|
notification.send_slack(title, color, poster_url, plex_url, message, footer)
|
|
|
|
|
|
|
|
|
2018-12-20 01:57:30 +00:00
|
|
|
def basic_notify(notifier_id, session_id, username=None, message=None, stream=None, tautulli=None):
|
2018-09-09 00:31:06 +00:00
|
|
|
"""Basic notifier"""
|
|
|
|
notification = Notification(notifier_id, SUBJECT_TEXT, BODY_TEXT, tautulli, stream)
|
|
|
|
|
|
|
|
if username:
|
|
|
|
body = BODY_TEXT_USER.format(user=username,
|
|
|
|
message=message)
|
|
|
|
else:
|
|
|
|
body = BODY_TEXT.format(id=session_id, message=message)
|
|
|
|
notification.send(SUBJECT_TEXT, body)
|
|
|
|
|
|
|
|
|
|
|
|
class Tautulli:
|
|
|
|
def __init__(self, url, apikey, verify_ssl=False, debug=None):
|
|
|
|
self.url = url
|
|
|
|
self.apikey = apikey
|
|
|
|
self.debug = debug
|
|
|
|
|
2018-12-20 01:57:30 +00:00
|
|
|
self.session = Session()
|
|
|
|
self.adapters = HTTPAdapter(max_retries=3,
|
|
|
|
pool_connections=1,
|
|
|
|
pool_maxsize=1,
|
|
|
|
pool_block=True)
|
2018-09-09 00:31:06 +00:00
|
|
|
self.session.mount('http://', self.adapters)
|
|
|
|
self.session.mount('https://', self.adapters)
|
|
|
|
|
|
|
|
# Ignore verifying the SSL certificate
|
|
|
|
if verify_ssl is False:
|
|
|
|
self.session.verify = False
|
|
|
|
# Disable the warning that the request is insecure, we know that...
|
|
|
|
import urllib3
|
|
|
|
urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning)
|
|
|
|
|
2018-12-20 01:57:30 +00:00
|
|
|
def _call_api(self, cmd, payload, method='GET'):
|
2018-09-09 00:31:06 +00:00
|
|
|
payload['cmd'] = cmd
|
|
|
|
payload['apikey'] = self.apikey
|
|
|
|
|
|
|
|
try:
|
|
|
|
response = self.session.request(method, self.url + '/api/v2', params=payload)
|
2018-12-20 01:57:30 +00:00
|
|
|
except RequestException as e:
|
|
|
|
print("Tautulli request failed for cmd '{}'. Invalid Tautulli URL? Error: {}".format(cmd, e))
|
2018-09-09 00:31:06 +00:00
|
|
|
if self.debug:
|
|
|
|
traceback.print_exc()
|
2018-06-18 19:38:10 +00:00
|
|
|
return
|
2018-06-17 06:36:27 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
try:
|
|
|
|
response_json = response.json()
|
|
|
|
except ValueError:
|
|
|
|
print("Failed to parse json response for Tautulli API cmd '{}'".format(cmd))
|
|
|
|
return
|
2018-06-17 06:36:27 +00:00
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
if response_json['response']['result'] == 'success':
|
|
|
|
if self.debug:
|
|
|
|
print("Successfully called Tautulli API cmd '{}'".format(cmd))
|
|
|
|
return response_json['response']['data']
|
|
|
|
else:
|
|
|
|
error_msg = response_json['response']['message']
|
|
|
|
print("Tautulli API cmd '{}' failed: {}".format(cmd, error_msg))
|
|
|
|
return
|
|
|
|
|
|
|
|
def get_activity(self, session_key=None, session_id=None):
|
|
|
|
"""Call Tautulli's get_activity api endpoint"""
|
|
|
|
payload = {}
|
|
|
|
|
|
|
|
if session_key:
|
|
|
|
payload['session_key'] = session_key
|
|
|
|
elif session_id:
|
|
|
|
payload['session_id'] = session_id
|
|
|
|
|
|
|
|
return self._call_api('get_activity', payload)
|
|
|
|
|
|
|
|
def notify(self, notifier_id, subject, body):
|
|
|
|
"""Call Tautulli's notify api endpoint"""
|
|
|
|
payload = {'notifier_id': notifier_id,
|
|
|
|
'subject': subject,
|
|
|
|
'body': body}
|
|
|
|
|
|
|
|
return self._call_api('notify', payload)
|
|
|
|
|
|
|
|
def terminate_session(self, session_key=None, session_id=None, message=''):
|
|
|
|
"""Call Tautulli's terminate_session api endpoint"""
|
|
|
|
payload = {}
|
|
|
|
|
|
|
|
if session_key:
|
|
|
|
payload['session_key'] = session_key
|
|
|
|
elif session_id:
|
|
|
|
payload['session_id'] = session_id
|
|
|
|
|
|
|
|
if message:
|
|
|
|
payload['message'] = message
|
|
|
|
|
|
|
|
return self._call_api('terminate_session', payload)
|
|
|
|
|
|
|
|
|
|
|
|
class Stream:
|
|
|
|
def __init__(self, session_id=None, user_id=None, username=None, tautulli=None, session=None):
|
2018-12-20 01:57:30 +00:00
|
|
|
self.state = None
|
2018-12-20 03:18:34 +00:00
|
|
|
self.ip_address = None
|
2018-09-09 00:31:06 +00:00
|
|
|
self.session_id = session_id
|
|
|
|
self.user_id = user_id
|
|
|
|
self.username = username
|
|
|
|
self.session_exists = False
|
|
|
|
self.tautulli = tautulli
|
|
|
|
|
|
|
|
if session is not None:
|
|
|
|
self._set_stream_attributes(session)
|
|
|
|
|
|
|
|
def _set_stream_attributes(self, session):
|
|
|
|
for k, v in session.items():
|
|
|
|
setattr(self, k, v)
|
|
|
|
|
|
|
|
def get_all_stream_info(self):
|
|
|
|
"""Get all stream info from Tautulli."""
|
|
|
|
session = self.tautulli.get_activity(session_id=self.session_id)
|
|
|
|
if session:
|
|
|
|
self._set_stream_attributes(session)
|
|
|
|
self.session_exists = True
|
|
|
|
else:
|
|
|
|
self.session_exists = False
|
|
|
|
|
|
|
|
def terminate(self, message=''):
|
|
|
|
"""Calls Tautulli to terminate the session.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
message : str
|
|
|
|
The message to use if the stream is terminated.
|
|
|
|
"""
|
|
|
|
self.tautulli.terminate_session(session_id=self.session_id, message=message)
|
|
|
|
|
|
|
|
def terminate_long_pause(self, message, limit, interval):
|
|
|
|
"""Kills the session if it is paused for longer than <limit> seconds.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
message : str
|
|
|
|
The message to use if the stream is terminated.
|
|
|
|
limit : int
|
|
|
|
The number of seconds the session is allowed to remain paused before it
|
|
|
|
is terminated.
|
|
|
|
interval : int
|
|
|
|
The amount of time to wait between checks of the session state.
|
|
|
|
"""
|
|
|
|
start = datetime.now()
|
|
|
|
checked_time = 0
|
|
|
|
# Continue checking 2 intervals past the allowed limit in order to
|
|
|
|
# account for system variances.
|
|
|
|
check_limit = limit + (interval * 2)
|
|
|
|
|
|
|
|
while checked_time < check_limit:
|
|
|
|
self.get_all_stream_info()
|
|
|
|
|
|
|
|
if self.session_exists is False:
|
|
|
|
sys.stdout.write(
|
|
|
|
"Session '{}' from user '{}' is no longer active "
|
|
|
|
.format(self.session_id, self.username)
|
|
|
|
+ "on the server, stopping monitoring.\n")
|
|
|
|
return False
|
|
|
|
|
|
|
|
now = datetime.now()
|
|
|
|
checked_time = (now - start).total_seconds()
|
|
|
|
|
|
|
|
if self.state == 'paused':
|
|
|
|
if checked_time >= limit:
|
|
|
|
self.terminate(message)
|
|
|
|
sys.stdout.write(
|
|
|
|
"Session '{}' from user '{}' has been killed.\n"
|
|
|
|
.format(self.session_id, self.username))
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
time.sleep(interval)
|
|
|
|
|
|
|
|
elif self.state == 'playing' or self.state == 'buffering':
|
|
|
|
sys.stdout.write(
|
|
|
|
"Session '{}' from user '{}' has been resumed, "
|
|
|
|
.format(self.session_id, self.username)
|
|
|
|
+ "stopping monitoring.\n")
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
class Notification:
|
|
|
|
def __init__(self, notifier_id, subject, body, tautulli, stream):
|
|
|
|
self.notifier_id = notifier_id
|
|
|
|
self.subject = subject
|
|
|
|
self.body = body
|
|
|
|
|
|
|
|
self.tautulli = tautulli
|
|
|
|
self.stream = stream
|
|
|
|
|
|
|
|
def send(self, subject='', body=''):
|
|
|
|
"""Send to Tautulli notifier.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
subject : str
|
|
|
|
Subject of the message.
|
|
|
|
body : str
|
|
|
|
Body of the message.
|
|
|
|
"""
|
|
|
|
subject = subject or self.subject
|
|
|
|
body = body or self.body
|
|
|
|
self.tautulli.notify(notifier_id=self.notifier_id, subject=subject, body=body)
|
|
|
|
|
|
|
|
def send_discord(self, title, color, poster_url, plex_url, message, footer):
|
|
|
|
"""Build the Discord message.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
title : str
|
|
|
|
The title of the message.
|
|
|
|
color : int
|
|
|
|
The color of the message
|
|
|
|
poster_url : str
|
|
|
|
The media poster URL.
|
|
|
|
plex_url : str
|
|
|
|
Plex media URL.
|
|
|
|
message : str
|
|
|
|
Message sent to the player.
|
|
|
|
footer : str
|
|
|
|
Footer of the message.
|
|
|
|
"""
|
|
|
|
discord_message = {
|
|
|
|
"embeds": [
|
|
|
|
{
|
|
|
|
"author": {
|
|
|
|
"icon_url": TAUTULLI_ICON,
|
|
|
|
"name": "Tautulli",
|
|
|
|
"url": TAUTULLI_LINK.rstrip('/')
|
|
|
|
},
|
|
|
|
"color": color,
|
|
|
|
"fields": [
|
|
|
|
{
|
|
|
|
"inline": True,
|
|
|
|
"name": "User",
|
|
|
|
"value": self.stream.friendly_name
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"inline": True,
|
|
|
|
"name": "Session Key",
|
|
|
|
"value": self.stream.session_key
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"inline": True,
|
|
|
|
"name": "Watching",
|
|
|
|
"value": self.stream.full_title
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"inline": False,
|
|
|
|
"name": "Message Sent",
|
|
|
|
"value": message
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"thumbnail": {
|
|
|
|
"url": poster_url
|
|
|
|
},
|
|
|
|
"title": title,
|
|
|
|
"timestamp": utc_now_iso(),
|
|
|
|
"url": plex_url,
|
|
|
|
"footer": {
|
|
|
|
"text": footer
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
],
|
|
|
|
}
|
|
|
|
|
|
|
|
discord_message = json.dumps(discord_message, sort_keys=True,
|
|
|
|
separators=(',', ': '))
|
|
|
|
self.send(body=discord_message)
|
|
|
|
|
|
|
|
def send_slack(self, title, color, poster_url, plex_url, message, footer):
|
|
|
|
"""Build the Slack message.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
title : str
|
|
|
|
The title of the message.
|
|
|
|
color : int
|
|
|
|
The color of the message
|
|
|
|
poster_url : str
|
|
|
|
The media poster URL.
|
|
|
|
plex_url : str
|
|
|
|
Plex media URL.
|
|
|
|
message : str
|
|
|
|
Message sent to the player.
|
|
|
|
footer : str
|
|
|
|
Footer of the message.
|
|
|
|
"""
|
|
|
|
slack_message = {
|
|
|
|
"attachments": [
|
|
|
|
{
|
|
|
|
"title": title,
|
|
|
|
"title_link": plex_url,
|
|
|
|
"author_icon": TAUTULLI_ICON,
|
|
|
|
"author_name": "Tautulli",
|
|
|
|
"author_link": TAUTULLI_LINK.rstrip('/'),
|
|
|
|
"color": color,
|
|
|
|
"fields": [
|
|
|
|
{
|
|
|
|
"title": "User",
|
|
|
|
"value": self.stream.friendly_name,
|
|
|
|
"short": True
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"title": "Session Key",
|
|
|
|
"value": self.stream.session_key,
|
|
|
|
"short": True
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"title": "Watching",
|
|
|
|
"value": self.stream.full_title,
|
|
|
|
"short": True
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"title": "Message Sent",
|
|
|
|
"value": message,
|
|
|
|
"short": False
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"thumb_url": poster_url,
|
|
|
|
"footer": footer,
|
|
|
|
"ts": time.time()
|
|
|
|
}
|
|
|
|
|
|
|
|
],
|
|
|
|
}
|
|
|
|
|
|
|
|
slack_message = json.dumps(slack_message, sort_keys=True,
|
|
|
|
separators=(',', ': '))
|
|
|
|
self.send(body=slack_message)
|
2018-06-29 20:14:13 +00:00
|
|
|
|
|
|
|
|
2018-06-11 19:57:29 +00:00
|
|
|
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.')
|
2018-06-29 20:14:13 +00:00
|
|
|
parser.add_argument('--username', type=arg_decoding,
|
2018-06-14 12:56:20 +00:00
|
|
|
help='The username of the person streaming.')
|
2018-08-13 11:01:05 +00:00
|
|
|
parser.add_argument('--sessionId',
|
2018-06-14 12:56:20 +00:00
|
|
|
help='The unique identifier for the stream.')
|
|
|
|
parser.add_argument('--notify', type=int,
|
2018-09-09 00:31:06 +00:00
|
|
|
help='Notification Agent ID number to Agent to send '
|
|
|
|
+ 'notification.')
|
2018-06-17 06:36:27 +00:00
|
|
|
parser.add_argument('--limit', type=int, default=(20 * 60), # 20 minutes
|
|
|
|
help='The time session is allowed to remain paused.')
|
|
|
|
parser.add_argument('--interval', type=int, default=30,
|
|
|
|
help='The seconds between paused session checks.')
|
2018-06-29 20:14:13 +00:00
|
|
|
parser.add_argument('--killMessage', nargs='+', type=arg_decoding,
|
2018-06-17 05:26:58 +00:00
|
|
|
help='Message to send to user whose stream is killed.')
|
2018-09-09 00:31:06 +00:00
|
|
|
parser.add_argument('--richMessage', type=arg_decoding, choices=RICH_TYPE,
|
|
|
|
help='Rich message type selector.\nChoices: (%(choices)s)')
|
|
|
|
parser.add_argument('--serverName', type=arg_decoding,
|
|
|
|
help='Plex Server Name')
|
|
|
|
parser.add_argument('--plexUrl', type=arg_decoding,
|
|
|
|
help='URL to plex media')
|
|
|
|
parser.add_argument('--posterUrl', type=arg_decoding,
|
|
|
|
help='Poster URL of the media')
|
|
|
|
parser.add_argument('--richColor', type=arg_decoding,
|
|
|
|
help='Color of the rich message')
|
|
|
|
parser.add_argument("--debug", action='store_true',
|
|
|
|
help='Enable debug messages.')
|
2018-06-11 19:57:29 +00:00
|
|
|
|
|
|
|
opts = parser.parse_args()
|
|
|
|
|
2018-08-13 11:03:37 +00:00
|
|
|
if not opts.sessionId and opts.jbop != 'allStreams':
|
2018-06-20 17:45:00 +00:00
|
|
|
sys.stderr.write("No sessionId provided! Is this synced content?\n")
|
|
|
|
sys.exit(1)
|
|
|
|
|
2018-09-09 00:31:06 +00:00
|
|
|
if opts.debug:
|
2018-12-20 01:08:27 +00:00
|
|
|
# Import traceback to get more detailed information
|
2018-09-09 00:31:06 +00:00
|
|
|
import traceback
|
2018-12-20 01:08:27 +00:00
|
|
|
# Dump the ENVs passed from tautulli
|
2018-09-09 00:31:06 +00:00
|
|
|
debug_dump_vars()
|
|
|
|
|
|
|
|
# Create a Tautulli instance
|
2018-12-20 01:57:30 +00:00
|
|
|
tautulli_server = Tautulli(TAUTULLI_URL.rstrip('/'), TAUTULLI_APIKEY, VERIFY_SSL, opts.debug)
|
2018-09-09 00:31:06 +00:00
|
|
|
|
|
|
|
# Create initial Stream object with basic info
|
2018-12-20 01:57:30 +00:00
|
|
|
tautulli_stream = Stream(opts.sessionId, opts.userId, opts.username, tautulli_server)
|
2018-09-09 00:31:06 +00:00
|
|
|
|
|
|
|
# Only pull all stream info if using richMessage
|
|
|
|
if opts.notify and opts.richMessage:
|
2018-12-20 01:57:30 +00:00
|
|
|
tautulli_stream.get_all_stream_info()
|
2018-09-09 00:31:06 +00:00
|
|
|
|
2018-12-20 01:08:27 +00:00
|
|
|
# Set a default message if none is provided
|
2018-06-14 12:56:20 +00:00
|
|
|
if opts.killMessage:
|
2018-12-20 01:57:30 +00:00
|
|
|
kill_message = ' '.join(opts.killMessage)
|
2018-06-14 12:56:20 +00:00
|
|
|
else:
|
2018-12-20 01:57:30 +00:00
|
|
|
kill_message = 'The server owner has ended the stream.'
|
2018-06-14 12:56:20 +00:00
|
|
|
|
2018-06-11 19:57:29 +00:00
|
|
|
if opts.jbop == 'stream':
|
2018-12-20 01:57:30 +00:00
|
|
|
tautulli_stream.terminate(kill_message)
|
|
|
|
notify(opts, kill_message, 'Stream', tautulli_stream, tautulli_server)
|
2018-09-09 00:31:06 +00:00
|
|
|
|
2018-06-11 19:57:29 +00:00
|
|
|
elif opts.jbop == 'allStreams':
|
2018-12-20 01:57:30 +00:00
|
|
|
all_streams = get_all_streams(tautulli_server, opts.userId)
|
|
|
|
for a_stream in all_streams:
|
|
|
|
tautulli_server.terminate_session(session_id=a_stream.session_id, message=kill_message)
|
|
|
|
notify(opts, kill_message, 'All Streams', a_stream, tautulli_server)
|
2018-09-09 00:31:06 +00:00
|
|
|
|
2018-12-20 03:15:45 +00:00
|
|
|
elif opts.jbop == 'multiIpStreams':
|
2018-12-20 05:26:35 +00:00
|
|
|
if not opts.richMessage:
|
2018-12-20 03:15:45 +00:00
|
|
|
tautulli_stream.get_all_stream_info()
|
|
|
|
user_streams = get_all_streams(tautulli_server, opts.userId)
|
|
|
|
for a_stream in user_streams:
|
|
|
|
if a_stream.ip_address != tautulli_stream.ip_address:
|
|
|
|
tautulli_stream.terminate(kill_message)
|
|
|
|
notify(opts, kill_message, 'Multiple IP Streams', tautulli_stream, tautulli_server)
|
2018-12-20 03:17:44 +00:00
|
|
|
|
2018-06-17 06:36:27 +00:00
|
|
|
elif opts.jbop == 'paused':
|
2018-12-20 01:57:30 +00:00
|
|
|
killed_stream = tautulli_stream.terminate_long_pause(kill_message, opts.limit, opts.interval)
|
2018-09-09 00:31:06 +00:00
|
|
|
if killed_stream:
|
2018-12-20 01:57:30 +00:00
|
|
|
notify(opts, kill_message, 'Paused', tautulli_stream, tautulli_server)
|