2019-06-21 06:55:11 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
"""
|
2017-07-12 18:06:35 +00:00
|
|
|
If server admin stream is experiencing buffering and there are concurrent transcode streams from
|
|
|
|
another user, kill concurrent transcode stream that has the lowest percent complete. Message in
|
|
|
|
kill stream will list why it was killed ('Server Admin's stream take priority and this user has X
|
|
|
|
concurrent streams'). Message will also include an approximation of when the other concurrent stream
|
|
|
|
will finish, stream that is closest to finish will be used.
|
|
|
|
|
2018-03-19 16:33:44 +00:00
|
|
|
Tautulli > Settings > Notification Agents > Scripts > Bell icon:
|
2017-07-12 18:06:35 +00:00
|
|
|
[X] Notify on buffer warning
|
|
|
|
|
2018-03-19 16:33:44 +00:00
|
|
|
Tautulli > Settings > Notification Agents > Scripts > Gear icon:
|
2017-07-12 18:06:35 +00:00
|
|
|
Buffer Warnings: kill_else_if_buffering.py
|
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
"""
|
2017-07-12 18:06:35 +00:00
|
|
|
|
|
|
|
import requests
|
|
|
|
from operator import itemgetter
|
2017-07-14 12:38:56 +00:00
|
|
|
import unicodedata
|
2017-10-19 19:32:10 +00:00
|
|
|
from plexapi.server import PlexServer
|
|
|
|
|
2017-07-12 18:06:35 +00:00
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
# ## EDIT THESE SETTINGS ##
|
2017-10-19 19:32:10 +00:00
|
|
|
PLEX_TOKEN = 'xxxx'
|
|
|
|
PLEX_URL = 'http://localhost:32400'
|
|
|
|
|
2017-07-12 18:06:35 +00:00
|
|
|
DEFAULT_REASON = 'Server Admin\'s stream takes priority and {user}(you) has {x} concurrent streams.' \
|
|
|
|
' {user}\'s stream of {video} is {time}% complete. Should be finished in {comp} minutes. ' \
|
|
|
|
'Try again then.'
|
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
ADMIN_USER = ('Admin') # Additional usernames can be added ('Admin', 'user2')
|
|
|
|
# ##
|
2017-07-12 18:06:35 +00:00
|
|
|
|
2017-10-19 19:32:10 +00:00
|
|
|
sess = requests.Session()
|
|
|
|
sess.verify = False
|
|
|
|
plex = PlexServer(PLEX_URL, PLEX_TOKEN, session=sess)
|
|
|
|
|
|
|
|
|
|
|
|
def kill_session(sess_key, message):
|
|
|
|
for session in plex.sessions():
|
|
|
|
# Check for users stream
|
|
|
|
username = session.usernames[0]
|
|
|
|
if session.sessionKey == sess_key:
|
2018-11-20 04:41:58 +00:00
|
|
|
title = unicode(session.grandparentTitle + ' - ' if session.type == 'episode' else '') + session.title
|
|
|
|
title = unicodedata.normalize('NFKD', title).encode('ascii', 'ignore').translate(None, "'")
|
2017-10-19 19:32:10 +00:00
|
|
|
session.stop(reason=message)
|
2018-11-20 04:41:58 +00:00
|
|
|
print('Terminated {user}\'s stream of {title} to prioritize admin stream.'.format(user=username,
|
|
|
|
title=title))
|
2017-10-19 19:32:10 +00:00
|
|
|
|
2017-07-12 18:06:35 +00:00
|
|
|
|
|
|
|
def add_to_dictlist(d, key, val):
|
|
|
|
if key not in d:
|
|
|
|
d[key] = [val]
|
|
|
|
else:
|
|
|
|
d[key].append(val)
|
|
|
|
|
|
|
|
|
2017-10-19 19:32:10 +00:00
|
|
|
def main():
|
2017-07-12 18:06:35 +00:00
|
|
|
user_dict = {}
|
2017-10-19 19:32:10 +00:00
|
|
|
|
|
|
|
for session in plex.sessions():
|
2018-11-20 04:41:58 +00:00
|
|
|
if session.transcodeSessions:
|
|
|
|
trans_dec = session.transcodeSessions[0].videoDecision
|
|
|
|
username = session.usernames[0]
|
|
|
|
if trans_dec == 'transcode' and username not in ADMIN_USER:
|
|
|
|
sess_key = session.sessionKey
|
|
|
|
percent_comp = int((float(session.viewOffset) / float(session.duration)) * 100)
|
|
|
|
time_to_comp = int(int(session.duration) - int(session.viewOffset)) / 1000 / 60
|
|
|
|
title = unicode(session.grandparentTitle + ' - ' if session.type == 'episode' else '') + session.title
|
|
|
|
title = unicodedata.normalize('NFKD', title).encode('ascii', 'ignore').translate(None, "'")
|
|
|
|
add_to_dictlist(user_dict, username, [sess_key, percent_comp, title, username, time_to_comp])
|
2017-07-12 18:06:35 +00:00
|
|
|
|
|
|
|
# Remove users with only 1 stream. Targeting users with multiple concurrent streams
|
|
|
|
filtered_dict = {key: value for key, value in user_dict.items()
|
2019-06-21 06:55:11 +00:00
|
|
|
if len(value) != 1}
|
2017-07-12 18:06:35 +00:00
|
|
|
|
|
|
|
# Find who to kill and who will be finishing first.
|
2018-11-20 04:41:58 +00:00
|
|
|
if filtered_dict:
|
|
|
|
for users in filtered_dict.values():
|
|
|
|
to_kill = min(users, key=itemgetter(1))
|
|
|
|
to_finish = max(users, key=itemgetter(1))
|
2019-06-21 06:55:11 +00:00
|
|
|
|
2018-11-20 04:41:58 +00:00
|
|
|
MESSAGE = DEFAULT_REASON.format(user=to_finish[3], x=len(filtered_dict.values()[0]),
|
|
|
|
video=to_finish[2], time=to_finish[1], comp=to_finish[4])
|
2019-06-21 06:55:11 +00:00
|
|
|
|
2018-11-20 04:41:58 +00:00
|
|
|
print(MESSAGE)
|
|
|
|
kill_session(to_kill[0], MESSAGE)
|
2017-10-19 19:32:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|