2018-07-25 11:45:34 +00:00
|
|
|
#!/usr/bin/env python
|
2018-07-23 14:51:24 +00:00
|
|
|
"""
|
|
|
|
Description: Comparing content between two or more Plex servers.
|
|
|
|
Creates .json file in script directory of server compared.
|
|
|
|
.json file contents include list items by media types (movie, show)
|
|
|
|
type combined between server 1 and server 2
|
|
|
|
type missing (_mine) from s2 but found in s1
|
|
|
|
type missing (_missing) from s1 but found in s2
|
|
|
|
type shared (_shared) between s1 and s2
|
|
|
|
Author: Blacktwin
|
|
|
|
Requires: requests, plexapi
|
|
|
|
|
|
|
|
Example:
|
|
|
|
python find_diff_other_servers.py --server "My Plex Server" --server PlexServer2
|
|
|
|
python find_diff_other_servers.py --server "My Plex Server" --server PlexServer2 --server "Steven Plex"
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
import argparse
|
|
|
|
import requests
|
|
|
|
import json
|
|
|
|
import sys
|
|
|
|
from plexapi.server import PlexServer, CONFIG
|
|
|
|
|
|
|
|
TAUTULLI_URL = ''
|
|
|
|
TAUTULLI_APIKEY = ''
|
|
|
|
TAUTULLI_URL = CONFIG.data['auth'].get('tautulli_baseurl', TAUTULLI_URL)
|
|
|
|
TAUTULLI_APIKEY = CONFIG.data['auth'].get('tautulli_apikey', TAUTULLI_APIKEY)
|
|
|
|
|
|
|
|
PLEX_URL = ''
|
|
|
|
PLEX_TOKEN = ''
|
|
|
|
PLEX_URL = CONFIG.data['auth'].get('server_baseurl', PLEX_URL)
|
|
|
|
PLEX_TOKEN = CONFIG.data['auth'].get('server_token', PLEX_TOKEN)
|
|
|
|
|
|
|
|
# Sections to ignore from comparision.
|
|
|
|
IGNORE_LST = ['Library name']
|
|
|
|
|
|
|
|
sess = requests.Session()
|
|
|
|
# Ignore verifying the SSL certificate
|
|
|
|
sess.verify = False # '/path/to/certfile'
|
|
|
|
# If verify is set to a path to a directory,
|
|
|
|
# the directory must have been processed using the c_rehash utility supplied
|
|
|
|
# with OpenSSL.
|
|
|
|
if sess.verify is False:
|
|
|
|
# Disable the warning that the request is insecure, we know that...
|
|
|
|
import urllib3
|
|
|
|
|
|
|
|
urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning)
|
|
|
|
|
|
|
|
plex = PlexServer(PLEX_URL, PLEX_TOKEN, session=sess)
|
|
|
|
|
|
|
|
SERVER_DICT = {server.name: server for server in
|
|
|
|
plex.myPlexAccount().resources()
|
|
|
|
if 'server' in server.provides.split(',')}
|
|
|
|
|
|
|
|
shared_lst = []
|
2018-07-25 11:22:19 +00:00
|
|
|
server_lst = []
|
2018-07-23 14:51:24 +00:00
|
|
|
|
2018-07-26 19:05:35 +00:00
|
|
|
def find_things(server, media_type):
|
|
|
|
dict_tt = {}
|
2018-07-23 14:51:24 +00:00
|
|
|
print('Finding items from {}.'.format(server.friendlyName))
|
|
|
|
for section in server.library.sections():
|
2018-07-26 19:05:35 +00:00
|
|
|
if section.title not in IGNORE_LST and section.type in media_type:
|
|
|
|
dict_tt[section.type] = []
|
2018-07-23 14:51:24 +00:00
|
|
|
for item in server.library.section(section.title).all():
|
2018-07-24 13:00:18 +00:00
|
|
|
dict_tt[section.type].append(server.fetchItem(item.ratingKey))
|
2018-07-23 14:51:24 +00:00
|
|
|
|
|
|
|
return dict_tt
|
|
|
|
|
|
|
|
|
2018-07-27 15:54:17 +00:00
|
|
|
def get_meta(main, friend, item, media_type):
|
|
|
|
meta = main.get(item)
|
|
|
|
if not meta:
|
|
|
|
meta = friend.get(item)
|
|
|
|
|
|
|
|
if media_type == 'movie':
|
|
|
|
meta_dict = {'title': item,
|
|
|
|
'rating': meta.rating,
|
|
|
|
'bitrate': meta.bitrate,
|
|
|
|
'genres': [x.tag for x in meta.genres]
|
|
|
|
}
|
|
|
|
else:
|
|
|
|
meta_dict = {'title': item,
|
|
|
|
'rating': meta.rating,
|
|
|
|
'genres': [x.tag for x in meta.genres]
|
|
|
|
}
|
|
|
|
|
|
|
|
return meta_dict
|
|
|
|
|
|
|
|
|
2018-07-26 19:05:35 +00:00
|
|
|
def org_diff(main, friend, key):
|
|
|
|
diff_dict = {}
|
2018-07-27 15:54:17 +00:00
|
|
|
meta_lst = []
|
|
|
|
mtitles = main.keys()
|
|
|
|
ftitles = friend.keys()
|
|
|
|
shared = set(mtitles + ftitles)
|
2018-07-26 19:05:35 +00:00
|
|
|
print('... combining {}s'.format(key))
|
|
|
|
|
2018-07-27 15:54:17 +00:00
|
|
|
mine = list(set(mtitles) - set(ftitles))
|
|
|
|
missing = list(set(ftitles) - set(mtitles))
|
|
|
|
combined = list(set(ftitles + mtitles))
|
|
|
|
|
|
|
|
for item in combined:
|
|
|
|
meta_lst.append(get_meta(main, friend, item, key))
|
|
|
|
|
|
|
|
diff_dict['{}_combined'.format(key)] = {'list': meta_lst,
|
2018-07-26 19:05:35 +00:00
|
|
|
'total': len(combined)}
|
|
|
|
|
|
|
|
print('... comparing {}s'.format(key))
|
|
|
|
print('... finding what is mine')
|
|
|
|
diff_dict['{}_mine'.format(key)] = {'list': mine,
|
|
|
|
'total': len(mine)}
|
|
|
|
print('... finding what is missing')
|
|
|
|
diff_dict['{}_missing'.format(key)] = {'list': missing,
|
|
|
|
'total': len(missing)}
|
|
|
|
print('... finding what is shared')
|
|
|
|
ddiff = set(mine + missing)
|
|
|
|
shared_lst = list(shared.union(ddiff) - shared.intersection(ddiff))
|
|
|
|
diff_dict['{}_shared'.format(key)] = {'list': shared_lst,
|
|
|
|
'total': len(shared_lst)}
|
2018-07-24 13:00:18 +00:00
|
|
|
|
2018-07-26 19:05:35 +00:00
|
|
|
return diff_dict
|
2018-07-23 14:51:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
def diff_things(main_dict, friend_dict):
|
2018-07-25 20:09:00 +00:00
|
|
|
diff_dict = {}
|
2018-07-23 14:51:24 +00:00
|
|
|
for key in main_dict.keys():
|
2018-07-27 15:54:17 +00:00
|
|
|
main_titles = {x.title: x for x in main_dict[key]}
|
|
|
|
friend_titles = {x.title: x for x in friend_dict[key]}
|
2018-07-26 19:05:35 +00:00
|
|
|
diff_dict[key] = org_diff(main_titles, friend_titles, key)
|
2018-07-25 11:45:34 +00:00
|
|
|
# todo-me guid double check?
|
2018-07-23 14:51:24 +00:00
|
|
|
|
2018-07-25 20:09:00 +00:00
|
|
|
# todo-me check back to obj in main/friend for rating and bitrate weights
|
2018-07-23 14:51:24 +00:00
|
|
|
|
2018-07-25 20:09:00 +00:00
|
|
|
return diff_dict
|
2018-07-23 14:51:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(
|
2018-07-26 19:05:35 +00:00
|
|
|
description="Comparing content between two or more Plex servers.",
|
|
|
|
formatter_class = argparse.RawTextHelpFormatter)
|
2018-07-23 14:51:24 +00:00
|
|
|
parser.add_argument('--server', required=True, choices=SERVER_DICT.keys(),
|
|
|
|
action='append', nargs='?', metavar='',
|
2018-07-26 19:05:35 +00:00
|
|
|
help='Choose servers to connect to and compare.\n'
|
|
|
|
'Choices: (%(choices)s)')
|
|
|
|
parser.add_argument('--media_type', required=True, choices=['movie', 'show', 'artist'],
|
|
|
|
nargs='+', metavar='', default=['movie', 'show'],
|
|
|
|
help='Choose media type(s) to compare.'
|
|
|
|
'\nDefault: (%(default)s)'
|
2018-07-23 14:51:24 +00:00
|
|
|
'\nChoices: (%(choices)s)')
|
2018-07-26 19:05:35 +00:00
|
|
|
# todo-me add media_type [x], library_ignore[], media filters (genre, etc.) []
|
2018-07-23 14:51:24 +00:00
|
|
|
|
|
|
|
opts = parser.parse_args()
|
|
|
|
|
|
|
|
if len(opts.server) < 2:
|
|
|
|
sys.stderr.write("Need more than one server to compare.\n")
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
server_compare = SERVER_DICT[opts.server[0]]
|
|
|
|
main_server = server_compare.connect()
|
|
|
|
print('Connected to {} server.'.format(main_server.friendlyName))
|
|
|
|
|
|
|
|
for server in opts.server[1:]:
|
|
|
|
other_server = SERVER_DICT[server]
|
|
|
|
try:
|
|
|
|
server_connected = other_server.connect()
|
|
|
|
print('Connected to {} server.'.format(
|
|
|
|
server_connected.friendlyName))
|
2018-07-25 11:22:19 +00:00
|
|
|
server_lst.append(server_connected)
|
2018-07-23 14:51:24 +00:00
|
|
|
except Exception as e:
|
2018-07-25 11:22:19 +00:00
|
|
|
sys.stderr.write("Error: {}.\nSkipping...\n".format(e))
|
|
|
|
pass
|
|
|
|
|
|
|
|
if len(server_lst) == 0:
|
|
|
|
sys.stderr.write("Need more than one server to compare.\n")
|
|
|
|
sys.exit(1)
|
|
|
|
|
2018-07-26 19:05:35 +00:00
|
|
|
main_section_dict = find_things(main_server, opts.media_type)
|
2018-07-23 14:51:24 +00:00
|
|
|
|
2018-07-25 11:22:19 +00:00
|
|
|
for connection in server_lst:
|
2018-07-26 19:05:35 +00:00
|
|
|
their_section_dict = find_things(connection, opts.media_type)
|
2018-07-23 14:51:24 +00:00
|
|
|
print('Comparing findings from {} and {}'.format(
|
2018-07-25 11:22:19 +00:00
|
|
|
main_server.friendlyName, connection.friendlyName))
|
2018-07-23 14:51:24 +00:00
|
|
|
main_dict = diff_things(main_section_dict, their_section_dict)
|
2018-07-25 20:09:00 +00:00
|
|
|
filename = 'diff_{}_{}_servers.json'.format(opts.server[0],
|
|
|
|
connection.friendlyName)
|
2018-07-24 13:00:18 +00:00
|
|
|
|
2018-07-23 14:51:24 +00:00
|
|
|
with open(filename, 'w') as fp:
|
|
|
|
json.dump(main_dict, fp, indent=4, sort_keys=True)
|