2018-08-23 10:38:59 +00:00
|
|
|
#!/usr/bin/env python
|
2019-06-21 06:55:11 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
"""Unshare or Remove users who have been inactive for X days. Prints out last seen for all users.
|
2017-12-17 21:49:47 +00:00
|
|
|
|
2019-05-01 21:09:50 +00:00
|
|
|
Just run.
|
2017-12-17 21:49:47 +00:00
|
|
|
|
|
|
|
Comment out `remove_friend(username)` and `unshare(username)` to test.
|
|
|
|
"""
|
2020-07-04 20:08:59 +00:00
|
|
|
from __future__ import print_function
|
2019-05-01 21:09:50 +00:00
|
|
|
from sys import exit
|
|
|
|
from requests import Session
|
|
|
|
from datetime import datetime
|
2018-08-23 10:41:49 +00:00
|
|
|
from plexapi.server import PlexServer, CONFIG
|
2017-12-17 21:49:47 +00:00
|
|
|
|
|
|
|
|
2019-05-01 21:09:50 +00:00
|
|
|
# EDIT THESE SETTINGS #
|
2018-08-23 10:41:49 +00:00
|
|
|
PLEX_URL = ''
|
|
|
|
PLEX_TOKEN = ''
|
2018-12-07 14:42:23 +00:00
|
|
|
TAUTULLI_URL = ''
|
|
|
|
TAUTULLI_APIKEY = ''
|
2017-12-17 21:49:47 +00:00
|
|
|
|
2019-05-01 21:09:50 +00:00
|
|
|
REMOVE_LIMIT = 30 # Days
|
|
|
|
UNSHARE_LIMIT = 15 # Days
|
2017-12-17 21:49:47 +00:00
|
|
|
|
2019-05-01 21:09:50 +00:00
|
|
|
USERNAME_IGNORE = ['user1', 'username2']
|
|
|
|
IGNORE_NEVER_SEEN = True
|
|
|
|
DRY_RUN = True
|
|
|
|
# EDIT THESE SETTINGS #
|
2018-12-07 14:42:23 +00:00
|
|
|
|
2019-05-01 21:09:50 +00:00
|
|
|
# CODE BELOW #
|
2018-12-07 14:42:23 +00:00
|
|
|
|
2019-05-01 21:09:50 +00:00
|
|
|
PLEX_URL = PLEX_URL or CONFIG.data['auth'].get('server_baseurl')
|
|
|
|
PLEX_TOKEN = PLEX_TOKEN or CONFIG.data['auth'].get('server_token')
|
|
|
|
TAUTULLI_URL = TAUTULLI_URL or CONFIG.data['auth'].get('tautulli_baseurl')
|
|
|
|
TAUTULLI_APIKEY = TAUTULLI_APIKEY or CONFIG.data['auth'].get('tautulli_apikey')
|
|
|
|
USERNAME_IGNORE = [username.lower() for username in USERNAME_IGNORE]
|
|
|
|
SESSION = Session()
|
2018-08-23 10:41:49 +00:00
|
|
|
# Ignore verifying the SSL certificate
|
2019-05-01 21:09:50 +00:00
|
|
|
SESSION.verify = False # '/path/to/certfile'
|
2018-08-23 10:41:49 +00:00
|
|
|
# If verify is set to a path to a directory,
|
2019-05-01 21:09:50 +00:00
|
|
|
# the directory must have been processed using the c_rehash utility supplied with OpenSSL.
|
|
|
|
if not SESSION.verify:
|
2018-08-23 10:41:49 +00:00
|
|
|
# Disable the warning that the request is insecure, we know that...
|
2019-05-01 21:09:50 +00:00
|
|
|
from urllib3 import disable_warnings
|
|
|
|
from urllib3.exceptions import InsecureRequestWarning
|
|
|
|
disable_warnings(InsecureRequestWarning)
|
|
|
|
|
|
|
|
SERVER = PlexServer(baseurl=PLEX_URL, token=PLEX_TOKEN, session=SESSION)
|
|
|
|
ACCOUNT = SERVER.myPlexAccount()
|
|
|
|
SECTIONS = [section.title for section in SERVER.library.sections()]
|
|
|
|
PLEX_USERS = {user.id: user.title for user in ACCOUNT.users()}
|
|
|
|
PLEX_USERS.update({int(ACCOUNT.id): ACCOUNT.title})
|
|
|
|
IGNORED_UIDS = [uid for uid, username in PLEX_USERS.items() if username.lower() in USERNAME_IGNORE]
|
|
|
|
IGNORED_UIDS.extend((int(ACCOUNT.id), 0))
|
|
|
|
# Get the Tautulli history.
|
|
|
|
PARAMS = {
|
|
|
|
'cmd': 'get_users_table',
|
|
|
|
'order_column': 'last_seen',
|
|
|
|
'order_dir': 'asc',
|
|
|
|
'length': 200,
|
|
|
|
'apikey': TAUTULLI_APIKEY
|
|
|
|
}
|
|
|
|
TAUTULLI_USERS = []
|
|
|
|
try:
|
|
|
|
GET = SESSION.get(TAUTULLI_URL.rstrip('/') + '/api/v2', params=PARAMS).json()['response']['data']['data']
|
|
|
|
for user in GET:
|
|
|
|
if user['user_id'] in IGNORED_UIDS:
|
|
|
|
continue
|
|
|
|
elif IGNORE_NEVER_SEEN and not user['last_seen']:
|
|
|
|
continue
|
|
|
|
TAUTULLI_USERS.append(user)
|
|
|
|
except Exception as e:
|
|
|
|
exit("Tautulli API 'get_users_table' request failed. Error: {}.".format(e))
|
|
|
|
|
|
|
|
|
|
|
|
def time_format(total_seconds):
|
|
|
|
# Display user's last history entry
|
|
|
|
days = total_seconds // 86400
|
|
|
|
hours = (total_seconds - days * 86400) // 3600
|
|
|
|
minutes = (total_seconds - days * 86400 - hours * 3600) // 60
|
|
|
|
seconds = total_seconds - days * 86400 - hours * 3600 - minutes * 60
|
|
|
|
result = ("{} day{}, ".format(days, "s" if days != 1 else "") if days else "") + \
|
|
|
|
("{} hour{}, ".format(hours, "s" if hours != 1 else "") if hours else "") + \
|
|
|
|
("{} minute{}, ".format(minutes, "s" if minutes != 1 else "") if minutes else "") + \
|
|
|
|
("{} second{}, ".format(seconds, "s" if seconds != 1 else "") if seconds else "")
|
|
|
|
return result.strip().rstrip(',')
|
|
|
|
|
|
|
|
|
|
|
|
NOW = datetime.today()
|
|
|
|
for user in TAUTULLI_USERS:
|
|
|
|
OUTPUT = []
|
|
|
|
USERNAME = user['friendly_name']
|
|
|
|
UID = user['user_id']
|
|
|
|
if not user['last_seen']:
|
|
|
|
TOTAL_SECONDS = None
|
|
|
|
OUTPUT = '{} has never used the server'.format(USERNAME)
|
|
|
|
else:
|
|
|
|
TOTAL_SECONDS = int((NOW - datetime.fromtimestamp(user['last_seen'])).total_seconds())
|
|
|
|
OUTPUT = '{} was last seen {} ago'.format(USERNAME, time_format(TOTAL_SECONDS))
|
2017-12-17 21:49:47 +00:00
|
|
|
|
2019-05-01 21:09:50 +00:00
|
|
|
if UID not in PLEX_USERS.keys():
|
|
|
|
print('{}, and exists in Tautulli but does not exist in Plex. Skipping.'.format(OUTPUT))
|
|
|
|
continue
|
2017-12-17 21:49:47 +00:00
|
|
|
|
2019-05-01 21:09:50 +00:00
|
|
|
TOTAL_SECONDS = TOTAL_SECONDS or 86400 * UNSHARE_LIMIT
|
|
|
|
if TOTAL_SECONDS >= (REMOVE_LIMIT * 86400):
|
|
|
|
if DRY_RUN:
|
|
|
|
print('{}, and would be removed.'.format(OUTPUT))
|
|
|
|
else:
|
|
|
|
print('{}, and has reached their shareless threshold. Removing.'.format(OUTPUT))
|
|
|
|
ACCOUNT.removeFriend(PLEX_USERS[UID])
|
|
|
|
elif TOTAL_SECONDS >= (UNSHARE_LIMIT * 86400):
|
|
|
|
if DRY_RUN:
|
|
|
|
print('{}, and would unshare libraries.'.format(OUTPUT))
|
|
|
|
else:
|
2019-10-02 18:42:33 +00:00
|
|
|
|
|
|
|
for server in ACCOUNT.user(PLEX_USERS[UID]).servers:
|
|
|
|
if server.machineIdentifier == SERVER.machineIdentifier and server.sections():
|
|
|
|
print('{}, and has reached their inactivity limit. Unsharing.'.format(OUTPUT))
|
|
|
|
ACCOUNT.updateFriend(PLEX_USERS[UID], SERVER, SECTIONS, removeSections=True)
|
|
|
|
else:
|
|
|
|
print("{}, has already been unshared, but has not reached their shareless threshold."
|
|
|
|
"Skipping.".format(OUTPUT))
|