plex-prerolls/schedule_preroll.py

405 lines
15 KiB
Python
Raw Normal View History

2021-01-01 16:12:22 +00:00
#!/usr/bin/python
"""Schedule Plex server related Pre-roll Intro videos
A helper script to automate management of Plex pre-rolls.
Define when you want different pre-rolls to play throughout the year.
Ideas include:
- Holiday pre-roll rotations
- Special occasions
- Summer/Winter rotations
- Breaking up the monoteny
- Keeping your family/friends on their toes!
Set it and forget it!
Optional Arguments:
-h, --help show this help message and exit
-c [], --config-path [] path/to/config.ini file to use
(Optional: uses local/plexapi default path if omitted)
-s [],--schedule-path [] path/to/preroll_schedules.yaml file to use
(Optional: uses local path if omitted)
-l [], --logconfig-path [] path/to/logging.conf logging configuration file
(Optional: uses local path if omitted)
Requirements:
- See Requirements.txt for Python modules
Scheduling:
Add to system scheduler such as:
> crontab -e
> 0 0 * * * python path/to/schedule_preroll.py >/dev/null 2>&1
Raises:
FileNotFoundError: [description]
KeyError: [description]
ConfigError: [description]
FileNotFoundError: [description]
"""
import os
import sys
import logging
import logging.config
import requests
import datetime
import yaml
from argparse import ArgumentParser
from configparser import ConfigParser
from configparser import Error as ConfigError
from plexapi.server import PlexServer, CONFIG
# import local modules
import plexutil
logger = logging.getLogger(__name__)
filename = os.path.basename(sys.argv[0])
SCRIPT_NAME = os.path.splitext(filename)[0]
def getArguments():
"""Return command line arguments
See https://docs.python.org/3/howto/argparse.html
Returns:
argparse.Namespace: Namespace object
"""
description = 'Automate scheduling of pre-roll intros for Plex'
2021-01-01 18:31:15 +00:00
version = '0.7.2'
2021-01-01 16:12:22 +00:00
config_default = './config.ini'
2021-01-01 18:31:15 +00:00
log_config_default = './logging.conf'
2021-01-01 16:12:22 +00:00
schedule_default = './preroll_schedules.yaml'
parser = ArgumentParser(description='{}'.format(description))
parser.add_argument('-v', '--version', action='version', version='%(prog)s {}'.format(version), help='show the version number and exit')
2021-01-01 18:31:15 +00:00
parser.add_argument('-l', '--logconfig-path', dest='log_config_file', default=log_config_default, action='store', help='Path to logging config file. [Default: {}]'.format(log_config_default))
parser.add_argument('-c', '--config-path', dest='config_file', action='store', help='Path to Config.ini to use for Plex Server info. [Default: {}]'.format(config_default))
parser.add_argument('-s', '--schedule-path', dest='schedule_file', action='store', help='Path to pre-roll schedule file (YAML) to be use. [Default: {}]'.format(schedule_default))
2021-01-01 16:12:22 +00:00
args = parser.parse_args()
return args
def getYAMLSchema():
"""Return the main schema layout of the preroll_schedules.yaml file
Returns:
dict: Dict of main schema items
"""
schema = {'default': None, 'monthly': None, 'weekly': None, 'date_range': None, 'misc': None}
return schema
def getWeekRange(year, weeknum):
"""Return the starting/ending date range of a given year/week
Args:
2021-01-01 18:31:15 +00:00
year (int): Year to calc range for
weeknum (int): Month of the year (1-12)
2021-01-01 16:12:22 +00:00
Returns:
Date: Start date of the Year/Month
Date: End date of the Year/Month
"""
start = datetime.datetime.strptime('{}-W{}-0'.format(year, int(weeknum)-1), "%Y-W%W-%w").date()
end = start + datetime.timedelta(days=6)
return start, end
def getMonthRange(year, monthnum):
"""Return the starting/ending date range of a given year/month
Args:
2021-01-01 18:31:15 +00:00
year (int): Year to calc range for
2021-01-01 16:12:22 +00:00
monthnum (int): Month of the year (1-12)
Returns:
Date: Start date of the Year/Month
Date: End date of the Year/Month
"""
start = datetime.date(year, monthnum, 1)
next_month = start.replace(day=28) + datetime.timedelta(days=4)
end = next_month - datetime.timedelta(days=next_month.day)
return start, end
2021-01-01 18:31:15 +00:00
def getPrerollSchedule(schedule_file=None):
2021-01-01 16:12:22 +00:00
"""Return a listing of defined preroll schedules for searching/use
2021-01-01 18:31:15 +00:00
Args:
schedule_file (string): path/to/schedule_preroll.yaml style config file (YAML Format)
2021-01-01 16:12:22 +00:00
Raises:
FileNotFoundError: If no schedule config file exists
Returns:
list: list of schedules (Dict: {Type, StartDate, EndDate, Path})
"""
default_files = ['preroll_schedules.yaml', 'preroll_schedules.yml']
2021-01-01 18:31:15 +00:00
filename = None
if schedule_file:
if os.path.exists(schedule_file):
filename = schedule_file
2021-01-01 16:12:22 +00:00
else:
2021-01-01 18:31:15 +00:00
raise FileNotFoundError('Preroll Schedule file -s "{}" not found'.format(schedule_file))
2021-01-01 16:12:22 +00:00
else:
for f in default_files:
if os.path.exists(f):
2021-01-01 18:31:15 +00:00
filename = f
2021-01-01 16:12:22 +00:00
break
# if we still cant find a schedule file, we hae to abort
2021-01-01 18:31:15 +00:00
if not filename:
2021-01-01 16:12:22 +00:00
msg = 'No {} Found'.format(' / '.join(default_files))
logger.critical(msg)
raise FileNotFoundError(msg)
2021-01-01 18:31:15 +00:00
with open(filename, 'r') as file:
2021-01-01 16:12:22 +00:00
#contents = yaml.load(file, Loader=yaml.SafeLoader)
contents = yaml.load(file, Loader=yaml.FullLoader)
today = datetime.date.today()
schedule = []
for schedule_type in getYAMLSchema():
if schedule_type == 'weekly':
try:
use = contents[schedule_type]['enabled']
if use:
for i in range(1,53):
try:
path = contents[schedule_type][i]
if path:
entry = {}
start, end = getWeekRange(today.year, i)
entry['Type'] = schedule_type
entry['StartDate'] = start
entry['EndDate'] = end
entry['Path'] = path
schedule.append(entry)
except KeyError as e:
# skip KeyError for missing Weeks
logger.debug('Key Value not found: "{}"->"{}", skipping week'.format(schedule_type, i))
continue
except KeyError as e:
logger.error('Key Value not found in "{}" section'.format(schedule_type), exc_info=e)
raise e
elif schedule_type == 'monthly':
try:
use = contents[schedule_type]['enabled']
if use:
for i in range(1,13):
month_abrev = datetime.date(today.year, i, 1).strftime('%b').lower()
try:
path = contents[schedule_type][month_abrev]
if path:
entry = {}
start, end = getMonthRange(today.year, i)
entry['Type'] = schedule_type
entry['StartDate'] = start
entry['EndDate'] = end
entry['Path'] = path
schedule.append(entry)
except KeyError as e:
# skip KeyError for missing Months
logger.warning('Key Value not found: "{}"->"{}", skipping month'.format(schedule_type, month_abrev))
continue
except KeyError as e:
logger.error('Key Value not found in "{}" section'.format(schedule_type), exc_info=e)
raise e
elif schedule_type == 'date_range':
try:
use = contents[schedule_type]['enabled']
if use:
for r in contents[schedule_type]['ranges']:
try:
path = r['path']
if path:
entry = {}
entry['Type'] = schedule_type
entry['StartDate'] = r['start_date']
entry['EndDate'] = r['end_date']
entry['Path'] = path
schedule.append(entry)
except KeyError as e:
#logger.error('Key Value not found: "{}"'.format(schedule_type), exc_info=e)
raise e
except KeyError as e:
logger.error('Key Value not found in "{}" section'.format(schedule_type), exc_info=e)
raise e
elif schedule_type == 'misc':
try:
use = contents[schedule_type]['enabled']
if use:
try:
path = contents[schedule_type]['always_use']
if path:
entry = {}
entry['Type'] = schedule_type
entry['StartDate'] = datetime.date(today.year, today.month, today.day)
entry['EndDate'] = datetime.date(today.year, today.month, today.day)
entry['Path'] = path
schedule.append(entry)
except KeyError as e:
#logger.error('Key Value not found: "{}"'.format(schedule_type), exc_info=e)
raise e
except KeyError as e:
logger.error('Key Value not found in "{}" section'.format(schedule_type), exc_info=e)
raise e
elif schedule_type == 'default':
try:
use = contents[schedule_type]['enabled']
if use:
try:
path = contents[schedule_type]['path']
if path:
entry = {}
entry['Type'] = schedule_type
entry['StartDate'] = datetime.date(today.year, today.month, today.day)
entry['EndDate'] = datetime.date(today.year, today.month, today.day)
entry['Path'] = path
schedule.append(entry)
except KeyError as e:
#logger.error('Key Value not found: "{}"'.format(schedule_type), exc_info=e)
raise e
except KeyError as e:
logger.error('Key Value not found in "{}" section'.format(schedule_type), exc_info=e)
raise e
else:
continue
# Sort list so most recent Ranges appear first
schedule.sort(reverse=True, key=lambda x:x['StartDate'])
return schedule
def buildListingString(items, play_all=False):
"""Build the Plex formatted string of preroll paths
Args:
2021-01-01 18:31:15 +00:00
items (list): List of preroll video paths to place into a string listing
play_all (bool, optional): Play all videos. [Default: False (Random choice)]
2021-01-01 16:12:22 +00:00
Returns:
string: CSV Listing (, or ;) based on play_all param of preroll video paths
"""
if play_all:
# use , to play all entries
listing = ','.join(items)
else:
pass
#use ; to play random selection
listing = ';'.join(items)
return listing
2021-01-01 18:31:15 +00:00
def getPrerollListingString(schedule, for_date=None):
2021-01-01 16:12:22 +00:00
"""Return listing of preroll videos to be used by Plex
2021-01-01 18:31:15 +00:00
Args:
schedule (list): List of schedule entries (See: getPrerollSchedule)
for_Date (date, optional): Date to process pre-roll string for [Default: Today]
Useful if wanting to test what different schedules produce
2021-01-01 16:12:22 +00:00
Returns:
2021-01-01 18:31:15 +00:00
string: listing of preroll video paths to be used for Extras. CSV style: (;|,)
2021-01-01 16:12:22 +00:00
"""
listing = ''
entries = dict(getYAMLSchema())
2021-01-01 18:31:15 +00:00
# prep the storage lists
2021-01-01 16:12:22 +00:00
for e in getYAMLSchema():
entries[e] = []
2021-01-01 18:31:15 +00:00
# determine which date to build the listing for
if for_date:
check_date = for_date
else:
check_date = datetime.date.today()
# process the schedule for the given date
2021-01-01 16:12:22 +00:00
for entry in schedule:
2021-01-01 18:31:15 +00:00
try:
if entry['StartDate'] <= check_date <= entry['EndDate']:
2021-01-01 16:12:22 +00:00
entry_type = entry['Type']
2021-01-01 18:31:15 +00:00
path = entry['Path']
2021-01-01 16:12:22 +00:00
2021-01-01 18:31:15 +00:00
if path:
entries[entry_type].append(path)
except KeyError as ke:
logger.warning('KeyError with entry "{}"'.format(entry), exc_info=ke)
continue
2021-01-01 16:12:22 +00:00
# Build the merged output based or order of Priority
merged_list = []
if entries['misc']:
merged_list.extend(entries['misc'])
if entries['date_range']:
merged_list.extend(entries['date_range'])
if entries['weekly'] and not entries['date_range']:
merged_list.extend(entries['weekly'])
if entries['monthly'] \
and not entries['weekly'] and not entries['date_range']:
merged_list.extend(entries['monthly'])
if entries['default'] \
and not entries['monthly'] and not entries['weekly'] and not entries['date_range']:
merged_list.extend(entries['default'])
listing = buildListingString(merged_list)
return listing
2021-01-01 18:31:15 +00:00
def savePrerollList(plex, preroll_listing):
2021-01-01 16:12:22 +00:00
"""Save Plex Preroll info to PlexServer settings
Args:
plex (PlexServer): Plex server to update
preroll_listing (string, list): csv listing or List of preroll paths to save
"""
# if happend to send in an Iterable List, merge to a string
if type(preroll_listing) is list:
preroll_listing = buildListingString(preroll_listing)
plex.settings.get('cinemaTrailersPrerollID').set(preroll_listing)
plex.settings.save()
if __name__ == '__main__':
args = getArguments()
2021-01-01 18:31:15 +00:00
plexutil.setupLogger(args.log_config_file)
2021-01-01 16:12:22 +00:00
cfg = plexutil.getPlexConfig(args.config_file)
# Initialize Session information
sess = requests.Session()
# Ignore verifying the SSL certificate
sess.verify = False # '/path/to/certfile'
# If verify is set to a path of a directory (not a cert file),
# the directory needs to be processed with the c_rehash utility
# from 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)
try:
plex = PlexServer(cfg['PLEX_URL'], cfg['PLEX_TOKEN'], session=sess)
except Exception as e:
logger.error('Error Connecting to Plex', exc_info=e)
raise e
2021-01-01 18:31:15 +00:00
schedule = getPrerollSchedule(args.schedule_file)
prerolls = getPrerollListingString(schedule)
2021-01-01 16:12:22 +00:00
logger.info('Saving Preroll List: "{}"'.format(prerolls))
2021-01-01 18:31:15 +00:00
savePrerollList(plex, prerolls)