You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
recyclarr/trash.py

280 lines
12 KiB

import requests
import re
import json
from collections import defaultdict
import argparse
# from dataclasses import dataclass
from enum import Enum
from packaging import version # pip install packaging
# Argparse setup
argparser = argparse.ArgumentParser(description='Automatically mirror TRaSH guides to your *darr instance.')
argparser.add_argument('--preview', help='Only display the processed markdown results and nothing else.',
action='store_true')
argparser.add_argument('--debug', help='Display additional logs useful for development/debug purposes',
action='store_true')
argparser.add_argument('base_uri', help='The base URL for your Sonarr instance, for example `http://localhost:8989`.')
argparser.add_argument('api_key', help='Your API key.')
args = argparser.parse_args()
Filter = Enum('FilterType', 'Preferred Required Ignored')
header_regex = re.compile(r'^(#+)\s([\w\s\d]+)\s*$')
score_regex = re.compile(r'score.*?\[([-\d]+)\]')
# included_preferred_regex = re.compile(r'include preferred', re.IGNORECASE)
# not_regex = re.compile(r'not', re.IGNORECASE)
filter_regexes = (
(Filter.Ignored, re.compile(r'must not contain', re.IGNORECASE)),
(Filter.Preferred, re.compile(r'preferred', re.IGNORECASE)),
)
# SONARR API STUFFS
base_uri = f'{args.base_uri}/api/v3'
key = f'?apikey={args.api_key}'
# --------------------------------------------------------------------------------------------------
def get_trash_anime_markdown():
trash_anime_markdown_url = 'https://raw.githubusercontent.com/TRaSH-/Guides/master/docs/Sonarr/V3/Sonarr-Release-Profile-RegEx-Anime.md'
response = requests.get(trash_anime_markdown_url)
return response.content.decode('utf8')
# --------------------------------------------------------------------------------------------------
def parse_filter(line):
for rx in filter_regexes:
if rx[1].search(line):
return rx[0]
return None
# --------------------------------------------------------------------------------------------------
def debug_print(text):
if args.debug:
print(text)
# --------------------------------------------------------------------------------------------------
class ProfileData:
def __init__(self):
self.preferred = defaultdict(list)
self.required = []
self.ignored = []
# We use 'none' here to represent no explicit mention of the "include preferred" string
# found in the markdown. We use this to control whether or not the corresponding profile
# section gets printed in the first place.
self.include_preferred_when_renaming = None
def parse_markdown(markdown_content):
class state:
results = defaultdict(ProfileData)
profile_name = None
score = None
filter = None
bracket_depth = 0
for line in markdown_content.splitlines():
# Header processing
if match := header_regex.search(line):
header_depth = len(match.group(1))
header_text = match.group(2)
# Profile name (always reset previous state here)
if header_depth == 3:
state.score = None
state.filter = Filter.Preferred
state.profile_name = header_text
debug_print(f'New Profile: {header_text}')
# Filter type for provided regexes
elif header_depth == 4:
state.filter = parse_filter(header_text)
if state.filter:
debug_print(f' Filter Set: {state.filter}')
# Lines we always look for
elif line.startswith('```'):
state.bracket_depth = 1 - state.bracket_depth
# Filter-based line processing
elif state.profile_name:
profile = state.results[state.profile_name]
lower_line = line.lower()
if 'include preferred' in lower_line:
profile.include_preferred_when_renaming = 'not' not in lower_line
debug_print(f' Include preferred found: {profile.include_preferred_when_renaming}, {lower_line}')
elif state.filter == Filter.Preferred:
if match := score_regex.search(line):
# bracket_depth = 0
state.score = int(match.group(1))
elif state.bracket_depth:
if state.score is not None:
debug_print(f' [Preferred] Score: {state.score}, Term: {line}')
profile.preferred[state.score].append(line)
elif state.filter == Filter.Ignored:
if state.bracket_depth:
# Sometimes a comma is present at the end of these regexes, because when it's
# pasted into Sonarr it acts as a delimiter. However, when using them with the
# API we do not need them.
profile.ignored.append(line.rstrip(','))
debug_print('\n\n')
return state.results
# --------------------------------------------------------------------------------------------------
def filter_profiles(profiles):
for name in list(profiles.keys()):
profile = profiles[name]
if not len(profile.required) and not len(profile.ignored) and not len(profile.preferred):
del profiles[name]
# --------------------------------------------------------------------------------------------------
def print_terms_and_scores(profiles):
for name, profile in profiles.items():
print(name)
if profile.include_preferred_when_renaming is not None:
print(' Include Preferred when Renaming?')
print(' ' + ('CHECKED' if profile.include_preferred_when_renaming else 'NOT CHECKED'))
print('')
if len(profile.required):
print(' Must Contain:')
for term in profile.required:
print(f' {term}')
print('')
if len(profile.ignored):
print(' Must Not Contain:')
for term in profile.ignored:
print(f' {term}')
print('')
if len(profile.preferred):
print(' Preferred:')
for score, terms in profile.preferred.items():
for term in terms:
print(f' {score:<10} {term}')
print('')
# --------------------------------------------------------------------------------------------------
def success_status_code(response):
return response.status_code >= 200 and response.status_code < 300
# --------------------------------------------------------------------------------------------------
def get_error_message(response: requests.Response):
response_body = json.dumps(response.content.decode('utf8'))
if type(response_body) is list and len(response_body) > 0:
return response[0]['errorMessage']
return None
# --------------------------------------------------------------------------------------------------
def sonarr_request(method, endpoint, data=None):
dispatch = {
'put': requests.put,
'get': requests.get,
'post': requests.post,
}
complete_uri = base_uri + endpoint + key
r = dispatch.get(method)(complete_uri, json.dumps(data))
r.raise_for_status()
return json.loads(r.content)
# --------------------------------------------------------------------------------------------------
def sonarr_get_version():
body = sonarr_request('get', '/system/status')
return version.parse(body['version'])
# --------------------------------------------------------------------------------------------------
def sonarr_create_release_profile(profile_name: str, profile: ProfileData):
json_preferred = []
for score, terms in profile.preferred.items():
for term in terms:
json_preferred.append({"key": term, "value": score})
data = {
'name': profile_name,
'enabled': True,
'required': ','.join(profile.required),
'ignored': ','.join(profile.ignored),
'preferred': json_preferred,
'includePreferredWhenRenaming': profile.include_preferred_when_renaming,
'tags': [],
'indexerId': 0
}
sonarr_request('post', '/releaseprofile', data)
# --------------------------------------------------------------------------------------------------
def sonarr_get_release_profiles():
return sonarr_request('get', '/releaseprofile')
# --------------------------------------------------------------------------------------------------
def find_existing_profile(profile_name, existing_profiles):
for p in existing_profiles:
if p.get('name') == new_profile_name:
return p
return None
# --------------------------------------------------------------------------------------------------
def sonarr_update_existing_profile(existing_profile, profile):
profile_id = existing_profile['id']
debug_print(f'update existing profile with id {profile_id}')
# Create the release profile
json_preferred = []
for score, terms in profile.preferred.items():
for term in terms:
json_preferred.append({"key": term, "value": score})
existing_profile['required'] = ','.join(profile.required)
existing_profile['ignored'] = ','.join(profile.ignored)
existing_profile['preferred'] = json_preferred
existing_profile['includePreferredWhenRenaming'] = profile.include_preferred_when_renaming
sonarr_request('put', f'/releaseprofile/{profile_id}', existing_profile)
####################################################################################################
try:
profiles = parse_markdown(get_trash_anime_markdown())
# A few false-positive profiles are added sometimes. We filter these out by checking if they
# actually have meaningful data attached to them, such as preferred terms. If they are mostly empty,
# we remove them here.
filter_profiles(profiles)
if args.preview:
print_terms_and_scores(profiles)
exit(0)
# Since this script requires a specific version of v3 Sonarr that implements name support for
# release profiles, we perform that version check here and bail out if it does not meet a minimum
# required version.
minimum_version = version.parse('3.0.4.1098')
version = sonarr_get_version()
if version < minimum_version:
print(f'ERROR: Your Sonarr version ({version}) does not meet the minimum required version of {minimum_version} to use this script.')
exit(1)
# Obtain all of the existing release profiles first. If any were previously created by our script
# here, we favor replacing those instead of creating new ones, which would just be mostly duplicates
# (but with some differences, since there have likely been updates since the last run).
existing_profiles = sonarr_get_release_profiles()
for name, profile in profiles.items():
new_profile_name = f'[Trash] Anime - {name}'
profile_to_update = find_existing_profile(new_profile_name, existing_profiles)
if profile_to_update:
print(f'Updating existing profile: {new_profile_name}')
sonarr_update_existing_profile(profile_to_update, profile)
else:
print(f'Creating new profile: {new_profile_name}')
sonarr_create_release_profile(new_profile_name, profile)
except requests.exceptions.HTTPError as e:
print(e)
if error_msg := get_error_message(e.response):
print(f'Response Message: {error_msg}')
exit(1)