337 lines
14 KiB
Python
337 lines
14 KiB
Python
#!/usr/bin/env python
|
|
'''
|
|
Share or unshare libraries.
|
|
|
|
optional arguments:
|
|
-h, --help show this help message and exit
|
|
--share To share libraries.
|
|
--shared Display user's share settings.
|
|
--unshare To unshare all libraries.
|
|
--kill Kill user's current stream(s). Include message to override default message
|
|
--add Add additional libraries.
|
|
--remove Remove existing libraries.
|
|
--user [ ...] Space separated list of case sensitive names to process. Allowed names are:
|
|
(choices: All users names)
|
|
--allUsers Select all users.
|
|
--libraries [ ...]
|
|
Space separated list of case sensitive names to process. Allowed names are:
|
|
(choices: All library names)
|
|
(default: All Libraries)
|
|
--allLibraries Select all libraries.
|
|
--sync Allow user to sync content
|
|
--camera Allow user to upload photos
|
|
--channel Allow user to utilize installed channels
|
|
--movieRatings Add rating restrictions to movie library types
|
|
--movieLabels Add label restrictions to movie library types
|
|
--tvRatings Add rating restrictions to show library types
|
|
--tvLabels Add label restrictions to show library types
|
|
--musicLabels Add label restrictions to music library types
|
|
--backup Backup share settings from json file
|
|
--restore Restore share settings from json file
|
|
Filename of json file to use.
|
|
(choices: %(json files found in cwd)s)
|
|
|
|
Usage:
|
|
|
|
plex_api_share.py --user USER --shared
|
|
- Current shares for USER: ['Movies', 'Music']
|
|
|
|
plex_api_share.py --share --user USER --libraries Movies
|
|
- Shared libraries: ['Movies'] with USER
|
|
|
|
plex_api_share.py --share --allUsers --libraries Movies
|
|
- Shared libraries: ['Movies'] with USER
|
|
- Shared libraries: ['Movies'] with USER1 ...
|
|
|
|
plex_api_share.py --share --user USER --libraries Movies "TV Shows"
|
|
- Shared libraries: ['Movies', 'TV Shows'] with USER
|
|
* Double Quote libraries with spaces
|
|
|
|
plex_api_share.py --share --user USER --allLibraries
|
|
- Shared all libraries with USER.
|
|
|
|
plex_api_share.py --user USER --add --libraries Movies
|
|
- Adds Movies library share to USER
|
|
|
|
plex_api_share.py --allUsers --remove --libraries Movies
|
|
- Removes Movies library share from all Users
|
|
|
|
plex_api_share.py --unshare --user USER
|
|
- Unshared all libraries with USER.
|
|
- USER is still exists as a Friend or Home User
|
|
|
|
Excluding;
|
|
|
|
--user becomes excluded if --allUsers is set
|
|
plex_api_share.py --share --allUsers --user USER --libraries Movies
|
|
- Shared libraries: ['Movies' ]with USER1.
|
|
- Shared libraries: ['Movies'] with USER2 ... all users but USER
|
|
|
|
--libraries becomes excluded if --allLibraries is set
|
|
plex_api_share.py --share -u USER --allLibraries --libraries Movies
|
|
- Shared [all libraries but Movies] with USER.
|
|
|
|
'''
|
|
|
|
from plexapi.server import PlexServer, CONFIG
|
|
import time
|
|
import argparse
|
|
import requests
|
|
import os
|
|
import json
|
|
|
|
PLEX_URL = ''
|
|
PLEX_TOKEN = ''
|
|
PLEX_URL = CONFIG.data['auth'].get('server_baseurl', PLEX_URL)
|
|
PLEX_TOKEN = CONFIG.data['auth'].get('server_token', PLEX_TOKEN)
|
|
|
|
DEFAULT_MESSAGE = "Steam is being killed by admin."
|
|
|
|
json_check = sorted([f for f in os.listdir('.') if os.path.isfile(f) and
|
|
f.endswith(".json")], key=os.path.getmtime)
|
|
|
|
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)
|
|
|
|
user_lst = [x.title for x in plex.myPlexAccount().users()]
|
|
sections_lst = [x.title for x in plex.library.sections()]
|
|
movies_keys = [x.key for x in plex.library.sections() if x.type == 'movie']
|
|
show_keys = [x.key for x in plex.library.sections() if x.type == 'show']
|
|
|
|
my_server_names = []
|
|
# Find all owners server names. For owners with multiple servers.
|
|
for res in plex.myPlexAccount().resources():
|
|
if res.provides == 'server' and res.owned == True:
|
|
my_server_names.append(res.name)
|
|
|
|
|
|
def get_ratings_lst(section_id):
|
|
headers = {'Accept': 'application/json'}
|
|
params = {'X-Plex-Token': PLEX_TOKEN}
|
|
content = requests.get("{}/library/sections/{}/contentRating".format(PLEX_URL, section_id),
|
|
headers=headers, params=params)
|
|
|
|
ratings_keys = content.json()['MediaContainer']['Directory']
|
|
ratings_lst = [x['title'] for x in ratings_keys]
|
|
return ratings_lst
|
|
|
|
|
|
def filter_clean(filter_type):
|
|
clean = ''
|
|
try:
|
|
clean = dict(item.split("=") for item in filter_type.split("|"))
|
|
for k, v in clean.items():
|
|
labels = v.replace('%20', ' ')
|
|
labels = labels.split('%2C')
|
|
clean[k] = labels
|
|
except Exception as e:
|
|
pass
|
|
return clean
|
|
|
|
|
|
def find_shares(user):
|
|
account = plex.myPlexAccount()
|
|
user_acct = account.user(user)
|
|
|
|
user_backup = {
|
|
'title': user_acct.title,
|
|
'username': user_acct.username,
|
|
'email': user_acct.email,
|
|
'userID': user_acct.id,
|
|
'allowSync': user_acct.allowSync,
|
|
'camera': user_acct.allowCameraUpload,
|
|
'channels': user_acct.allowChannels,
|
|
'filterMovies': filter_clean(user_acct.filterMovies),
|
|
'filterTelevision': filter_clean(user_acct.filterTelevision),
|
|
'filterMusic': filter_clean(user_acct.filterMusic),
|
|
'servers': []}
|
|
|
|
for server in user_acct.servers:
|
|
if server.name in my_server_names:
|
|
sections = []
|
|
for section in server.sections():
|
|
if section.shared == True:
|
|
sections.append(section.title)
|
|
user_backup['servers'].append({'serverName': server.name,
|
|
'sections': sections})
|
|
|
|
return user_backup
|
|
|
|
|
|
def kill_session(user, message):
|
|
reason = DEFAULT_MESSAGE
|
|
for session in plex.sessions():
|
|
# Check for users stream
|
|
if session.usernames[0] in user:
|
|
title = (session.grandparentTitle + ' - ' if session.type == 'episode' else '') + session.title
|
|
print('{user} was watching {title}. Killing stream and unsharing.'.format(
|
|
user=user, title=title))
|
|
if message:
|
|
reason = message
|
|
session.stop(reason=reason)
|
|
|
|
|
|
def share(user, sections, allowSync, camera, channels, filterMovies, filterTelevision, filterMusic):
|
|
plex.myPlexAccount().updateFriend(user=user, server=plex, sections=sections, allowSync=allowSync,
|
|
allowCameraUpload=camera, allowChannels=channels, filterMovies=filterMovies,
|
|
filterTelevision=filterTelevision, filterMusic=filterMusic)
|
|
print('Shared libraries: {sections} with {user}.'.format(sections=sections, user=user))
|
|
|
|
|
|
def unshare(user, sections):
|
|
plex.myPlexAccount().updateFriend(user=user, server=plex, removeSections=True, sections=sections)
|
|
print('Unuser_shares all libraries from {user}.'.format(user=user))
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
movie_ratings = []
|
|
show_ratings = []
|
|
for movie in movies_keys:
|
|
movie_ratings += get_ratings_lst(movie)
|
|
for show in show_keys:
|
|
show_ratings += get_ratings_lst(show)
|
|
|
|
timestr = time.strftime("%Y%m%d-%H%M%S")
|
|
|
|
parser = argparse.ArgumentParser(description="Share or unshare libraries.",
|
|
formatter_class=argparse.RawTextHelpFormatter)
|
|
parser.add_argument('--share', default=False, action='store_true',
|
|
help='To share libraries.')
|
|
parser.add_argument('--shared', default=False, action='store_true',
|
|
help='Display user\'s shared libraries.')
|
|
parser.add_argument('--unshare', default=False, action='store_true',
|
|
help='To unshare all libraries.')
|
|
parser.add_argument('--kill', default=False, nargs='?',
|
|
help='Kill user\'s current stream(s). Include message to override default message.')
|
|
parser.add_argument('--add', default=False, action='store_true',
|
|
help='Add additional libraries.')
|
|
parser.add_argument('--remove', default=False, action='store_true',
|
|
help='Remove existing libraries.')
|
|
parser.add_argument('--user', nargs='+', choices=user_lst, metavar='',
|
|
help='Space separated list of case sensitive names to process. Allowed names are: \n'
|
|
'(choices: %(choices)s)')
|
|
parser.add_argument('--allUsers', default=False, action='store_true',
|
|
help='Select all users.')
|
|
parser.add_argument('--libraries', nargs='+', default=False, choices=sections_lst, metavar='',
|
|
help='Space separated list of case sensitive names to process. Allowed names are: \n'
|
|
'(choices: %(choices)s')
|
|
parser.add_argument('--allLibraries', default=False, action='store_true',
|
|
help='Select all libraries.')
|
|
parser.add_argument('--sync', default=False, action='store_true',
|
|
help='Use to allow user to sync content.')
|
|
parser.add_argument('--camera', default=False, action='store_true',
|
|
help='Use to allow user to upload photos.')
|
|
parser.add_argument('--channels', default=False, action='store_true',
|
|
help='Use to allow user to utilize installed channels.')
|
|
parser.add_argument('--movieRatings', nargs='+', choices=list(set(movie_ratings)), metavar='',
|
|
help='Use to add rating restrictions to movie library types.\n'
|
|
'Space separated list of case sensitive names to process. Allowed names are: \n'
|
|
'(choices: %(choices)s')
|
|
parser.add_argument('--movieLabels', nargs='+', metavar='',
|
|
help='Use to add label restrictions for movie library types.')
|
|
parser.add_argument('--tvRatings', nargs='+', choices=list(set(show_ratings)), metavar='',
|
|
help='Use to add rating restrictions for show library types.\n'
|
|
'Space separated list of case sensitive names to process. Allowed names are: \n'
|
|
'(choices: %(choices)s')
|
|
parser.add_argument('--tvLabels', nargs='+', metavar='',
|
|
help='Use to add label restrictions for show library types.')
|
|
parser.add_argument('--musicLabels', nargs='+', metavar='',
|
|
help='Use to add label restrictions for music library types.')
|
|
|
|
parser.add_argument('--backup', default=False, action='store_true',
|
|
help='Backup share settings from json file.')
|
|
parser.add_argument('--restore', type = str, choices = json_check,
|
|
help='Restore share settings from json file.\n'
|
|
'Filename of json file to use.\n'
|
|
'(choices: %(choices)s)')
|
|
|
|
opts = parser.parse_args()
|
|
users = ''
|
|
libraries = ''
|
|
filterMovies = {}
|
|
filterTelevision = {}
|
|
filterMusic = {}
|
|
|
|
# Setting additional share options
|
|
if opts.movieLabels:
|
|
filterMovies['label'] = opts.movieLabels
|
|
if opts.movieRatings:
|
|
filterMovies['contentRating'] = opts.movieRatings
|
|
if opts.tvLabels:
|
|
filterTelevision['label'] = opts.tvLabels
|
|
if opts.tvRatings:
|
|
filterTelevision['contentRating'] = opts.tvRatings
|
|
if opts.musicLabels:
|
|
filterMusic['label'] = opts.musicLabels
|
|
|
|
# Defining users
|
|
if opts.allUsers and not opts.user:
|
|
users = user_lst
|
|
elif not opts.allUsers and opts.user:
|
|
users = opts.user
|
|
elif opts.allUsers and opts.user:
|
|
# If allUsers is used then any users listed will be excluded
|
|
for user in opts.user:
|
|
user_lst.remove(user)
|
|
users = user_lst
|
|
|
|
# Defining libraries
|
|
if opts.allLibraries and not opts.libraries:
|
|
libraries = sections_lst
|
|
elif not opts.allLibraries and opts.libraries:
|
|
libraries = opts.libraries
|
|
elif opts.allLibraries and opts.libraries:
|
|
# If allLibraries is used then any libraries listed will be excluded
|
|
for library in opts.libraries:
|
|
sections_lst.remove(library)
|
|
libraries = sections_lst
|
|
|
|
# Share, Unshare, Kill, Add, or Remove
|
|
for user in users:
|
|
user_shares = find_shares(user)
|
|
if libraries:
|
|
if opts.share:
|
|
share(user, libraries, opts.sync, opts.camera, opts.channels, filterMovies, filterTelevision,
|
|
filterMusic)
|
|
if opts.add and user_shares['sections']:
|
|
libraries = libraries + user_shares['sections']
|
|
libraries = list(set(libraries))
|
|
share(user, libraries, opts.sync, opts.camera, opts.channels, filterMovies, filterTelevision,
|
|
filterMusic)
|
|
if opts.remove and user_shares['sections']:
|
|
libraries = [sect for sect in user_shares['sections'] if sect not in libraries]
|
|
share(user, libraries, opts.sync, opts.camera, opts.channels, filterMovies, filterTelevision,
|
|
filterMusic)
|
|
if opts.shared:
|
|
user_json = json.dumps(user_shares, indent=4, sort_keys=True)
|
|
print('Current share settings for {}: {}'.format(user, user_json))
|
|
if opts.unshare and opts.kill:
|
|
kill_session(user, opts.kill)
|
|
time.sleep(3)
|
|
unshare(user, sections_lst)
|
|
elif opts.unshare:
|
|
unshare(user, sections_lst)
|
|
elif opts.kill:
|
|
kill_session(user, opts.kill)
|
|
|
|
if opts.backup:
|
|
print('Backing up share information...')
|
|
users_shares = []
|
|
for user in user_lst:
|
|
# print('...Found {}'.format(user))
|
|
users_shares.append(find_shares(user))
|
|
json_file = 'Plex_share_backup_{}.json'.format(timestr)
|
|
with open(json_file, 'w') as fp:
|
|
json.dump(users_shares, fp, indent=4, sort_keys=True) |