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.
1670 lines
71 KiB
1670 lines
71 KiB
bazarr_version = '0.6.5'
|
|
|
|
import gc
|
|
gc.enable()
|
|
|
|
from get_argv import config_dir, no_update
|
|
|
|
import os
|
|
import sys
|
|
reload(sys)
|
|
sys.setdefaultencoding('utf8')
|
|
sys.path.insert(0, os.path.join(os.path.dirname(__file__), 'libs/'))
|
|
|
|
import sqlite3
|
|
from init import *
|
|
from update_db import *
|
|
|
|
|
|
from get_settings import get_general_settings, get_proxy_settings
|
|
import logging
|
|
from logging.handlers import TimedRotatingFileHandler
|
|
|
|
logger = logging.getLogger('waitress')
|
|
log_level = get_general_settings()[4]
|
|
if log_level is None:
|
|
log_level = "INFO"
|
|
|
|
class OneLineExceptionFormatter(logging.Formatter):
|
|
def formatException(self, exc_info):
|
|
"""
|
|
Format an exception so that it prints on a single line.
|
|
"""
|
|
result = super(OneLineExceptionFormatter, self).formatException(exc_info)
|
|
return repr(result) # or format into one line however you want to
|
|
|
|
def format(self, record):
|
|
s = super(OneLineExceptionFormatter, self).format(record)
|
|
if record.exc_text:
|
|
s = s.replace('\n', '') + '|'
|
|
return s
|
|
|
|
def configure_logging():
|
|
global fh
|
|
fh = TimedRotatingFileHandler(os.path.join(config_dir, 'log/bazarr.log'), when="midnight", interval=1, backupCount=7)
|
|
f = OneLineExceptionFormatter('%(asctime)s|%(levelname)s|%(message)s|',
|
|
'%d/%m/%Y %H:%M:%S')
|
|
fh.setFormatter(f)
|
|
logging.getLogger("enzyme").setLevel(logging.CRITICAL)
|
|
logging.getLogger("apscheduler").setLevel(logging.WARNING)
|
|
logging.getLogger("subliminal").setLevel(logging.CRITICAL)
|
|
logging.getLogger("stevedore.extension").setLevel(logging.CRITICAL)
|
|
root = logging.getLogger()
|
|
root.setLevel(log_level)
|
|
root.addHandler(fh)
|
|
|
|
configure_logging()
|
|
|
|
import requests
|
|
if get_proxy_settings()[0] != 'None':
|
|
if get_proxy_settings()[3] != '' and get_proxy_settings()[4] != '':
|
|
proxy = get_proxy_settings()[0] + '://' + get_proxy_settings()[3] + ':' + get_proxy_settings()[4] + '@' + get_proxy_settings()[1] + ':' + get_proxy_settings()[2]
|
|
else:
|
|
proxy = get_proxy_settings()[0] + '://' + get_proxy_settings()[1] + ':' + get_proxy_settings()[2]
|
|
os.environ['HTTP_PROXY'] = str(proxy)
|
|
os.environ['HTTPS_PROXY'] = str(proxy)
|
|
os.environ['NO_PROXY'] = str(get_proxy_settings()[5])
|
|
|
|
from bottle import route, run, template, static_file, request, redirect, response, HTTPError, app
|
|
import bottle
|
|
bottle.TEMPLATE_PATH.insert(0, os.path.join(os.path.dirname(__file__), 'views/'))
|
|
bottle.debug(True)
|
|
bottle.TEMPLATES.clear()
|
|
|
|
from beaker.middleware import SessionMiddleware
|
|
from cork import Cork
|
|
from json import dumps
|
|
import itertools
|
|
import operator
|
|
import pretty
|
|
from datetime import datetime, timedelta
|
|
from io import BytesIO
|
|
import math
|
|
import ast
|
|
import hashlib
|
|
import time
|
|
|
|
from get_languages import load_language_in_db, language_from_alpha3
|
|
from get_providers import load_providers, get_providers, get_providers_auth
|
|
load_providers()
|
|
|
|
from get_series import *
|
|
from get_episodes import *
|
|
from get_settings import base_url, ip, port, path_replace, path_replace_movie
|
|
if no_update is False:
|
|
from check_update import check_and_apply_update
|
|
from list_subtitles import store_subtitles, store_subtitles_movie, series_scan_subtitles, movies_scan_subtitles, list_missing_subtitles, list_missing_subtitles_movies
|
|
from get_subtitle import download_subtitle, series_download_subtitles, movies_download_subtitles, wanted_download_subtitles, wanted_search_missing_subtitles, manual_search, manual_download_subtitle
|
|
from utils import history_log, history_log_movie
|
|
from scheduler import *
|
|
from notifier import send_notifications, send_notifications_movie
|
|
|
|
# Reset restart required warning on start
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = conn.cursor()
|
|
c.execute("UPDATE system SET configured = 0, updated = 0")
|
|
conn.commit()
|
|
c.close()
|
|
|
|
# Load languages in database
|
|
load_language_in_db()
|
|
|
|
from get_settings import get_auth_settings
|
|
|
|
aaa = Cork(os.path.normpath(os.path.join(config_dir, 'config')))
|
|
|
|
app = app()
|
|
session_opts = {
|
|
'session.cookie_expires': True,
|
|
'session.key': 'Bazarr',
|
|
'session.httponly': True,
|
|
'session.timeout': 3600 * 24, # 1 day
|
|
'session.type': 'cookie',
|
|
'session.validate_key': True
|
|
}
|
|
app = SessionMiddleware(app, session_opts)
|
|
login_auth = get_auth_settings()[0]
|
|
|
|
|
|
def custom_auth_basic(check):
|
|
def decorator(func):
|
|
def wrapper(*a, **ka):
|
|
if get_auth_settings()[0] == 'basic':
|
|
user, password = request.auth or (None, None)
|
|
if user is None or not check(user, password):
|
|
err = HTTPError(401, "Access denied")
|
|
err.add_header('WWW-Authenticate', 'Basic realm="Bazarr"')
|
|
return err
|
|
return func(*a, **ka)
|
|
else:
|
|
return func(*a, **ka)
|
|
|
|
return wrapper
|
|
return decorator
|
|
|
|
def check_credentials(user, pw):
|
|
from get_settings import get_auth_settings
|
|
|
|
username = get_auth_settings()[1]
|
|
password = get_auth_settings()[2]
|
|
if hashlib.md5(pw).hexdigest() == password and user == username:
|
|
return True
|
|
return False
|
|
|
|
|
|
def authorize():
|
|
if login_auth == 'form':
|
|
aaa = Cork(os.path.normpath(os.path.join(config_dir, 'config')))
|
|
aaa.require(fail_redirect=(base_url + 'login'))
|
|
|
|
|
|
def post_get(name, default=''):
|
|
return request.POST.get(name, default).strip()
|
|
|
|
|
|
@route(base_url + 'login')
|
|
def login_form():
|
|
msg = bottle.request.query.get('msg', '')
|
|
return template('login', base_url=base_url, msg=msg)
|
|
|
|
|
|
@route(base_url + 'login', method='POST')
|
|
def login():
|
|
aaa = Cork(os.path.normpath(os.path.join(config_dir, 'config')))
|
|
username = post_get('username')
|
|
password = post_get('password')
|
|
aaa.login(username, password, success_redirect=base_url, fail_redirect=(base_url + 'login?msg=fail'))
|
|
|
|
|
|
@route(base_url + 'logout')
|
|
def logout():
|
|
aaa.logout(success_redirect=(base_url + 'login'))
|
|
|
|
|
|
@route('/')
|
|
@custom_auth_basic(check_credentials)
|
|
def redirect_root():
|
|
authorize()
|
|
redirect (base_url)
|
|
|
|
@route(base_url + 'static/:path#.+#', name='static')
|
|
@custom_auth_basic(check_credentials)
|
|
def static(path):
|
|
return static_file(path, root=os.path.join(os.path.dirname(__file__), 'static'))
|
|
|
|
@route(base_url + 'emptylog')
|
|
@custom_auth_basic(check_credentials)
|
|
def emptylog():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
fh.doRollover()
|
|
logging.info('Log file emptied')
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'bazarr.log')
|
|
@custom_auth_basic(check_credentials)
|
|
def download_log():
|
|
authorize()
|
|
return static_file('bazarr.log', root=os.path.join(config_dir, 'log/'), download='bazarr.log')
|
|
|
|
@route(base_url + 'image_proxy/<url:path>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def image_proxy(url):
|
|
authorize()
|
|
url_sonarr = get_sonarr_settings()[6]
|
|
url_sonarr_short = get_sonarr_settings()[7]
|
|
apikey = get_sonarr_settings()[4]
|
|
url_image = url_sonarr_short + '/' + url + '?apikey=' + apikey
|
|
try:
|
|
image_buffer = BytesIO(requests.get(url_sonarr + '/api' + url_image.split(url_sonarr)[1], timeout=15, verify=False).content)
|
|
except:
|
|
return None
|
|
else:
|
|
image_buffer.seek(0)
|
|
bytes = image_buffer.read()
|
|
response.set_header('Content-type', 'image/jpeg')
|
|
return bytes
|
|
|
|
@route(base_url + 'image_proxy_movies/<url:path>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def image_proxy_movies(url):
|
|
authorize()
|
|
url_radarr = get_radarr_settings()[6]
|
|
url_radarr_short = get_radarr_settings()[7]
|
|
apikey = get_radarr_settings()[4]
|
|
try:
|
|
url_image = (url_radarr_short + '/' + url + '?apikey=' + apikey).replace('/fanart.jpg', '/banner.jpg')
|
|
image_buffer = BytesIO(requests.get(url_radarr + '/api' + url_image.split(url_radarr)[1], timeout=15, verify=False).content)
|
|
except:
|
|
url_image = url_radarr_short + '/' + url + '?apikey=' + apikey
|
|
image_buffer = BytesIO(requests.get(url_radarr + '/api' + url_image.split(url_radarr)[1], timeout=15, verify=False).content)
|
|
else:
|
|
image_buffer.seek(0)
|
|
bytes = image_buffer.read()
|
|
response.set_header('Content-type', 'image/jpeg')
|
|
return bytes
|
|
|
|
|
|
@route(base_url)
|
|
@custom_auth_basic(check_credentials)
|
|
def redirect_root():
|
|
authorize()
|
|
if get_general_settings()[12] is True:
|
|
redirect(base_url + 'series')
|
|
elif get_general_settings()[13] is True:
|
|
redirect(base_url + 'movies')
|
|
else:
|
|
redirect(base_url + 'settings')
|
|
|
|
|
|
@route(base_url + 'series')
|
|
@custom_auth_basic(check_credentials)
|
|
def series():
|
|
authorize()
|
|
single_language = get_general_settings()[7]
|
|
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
db.create_function("path_substitution", 1, path_replace)
|
|
c = db.cursor()
|
|
|
|
c.execute("SELECT COUNT(*) FROM table_shows")
|
|
missing_count = c.fetchone()
|
|
missing_count = missing_count[0]
|
|
page = request.GET.page
|
|
if page == "":
|
|
page = "1"
|
|
page_size = int(get_general_settings()[21])
|
|
offset = (int(page) - 1) * page_size
|
|
max_page = int(math.ceil(missing_count / (page_size + 0.0)))
|
|
|
|
c.execute("SELECT tvdbId, title, path_substitution(path), languages, hearing_impaired, sonarrSeriesId, poster, audio_language FROM table_shows ORDER BY sortTitle ASC LIMIT ? OFFSET ?", (page_size, offset,))
|
|
data = c.fetchall()
|
|
c.execute("SELECT code2, name FROM table_settings_languages WHERE enabled = 1")
|
|
languages = c.fetchall()
|
|
c.execute("SELECT table_shows.sonarrSeriesId, COUNT(table_episodes.missing_subtitles) FROM table_shows LEFT JOIN table_episodes ON table_shows.sonarrSeriesId=table_episodes.sonarrSeriesId WHERE table_shows.languages IS NOT 'None' AND table_episodes.missing_subtitles IS NOT '[]' GROUP BY table_shows.sonarrSeriesId")
|
|
missing_subtitles_list = c.fetchall()
|
|
c.execute("SELECT table_shows.sonarrSeriesId, COUNT(table_episodes.missing_subtitles) FROM table_shows LEFT JOIN table_episodes ON table_shows.sonarrSeriesId=table_episodes.sonarrSeriesId WHERE table_shows.languages IS NOT 'None' GROUP BY table_shows.sonarrSeriesId")
|
|
total_subtitles_list = c.fetchall()
|
|
c.close()
|
|
output = template('series', __file__=__file__, bazarr_version=bazarr_version, rows=data, missing_subtitles_list=missing_subtitles_list, total_subtitles_list=total_subtitles_list, languages=languages, missing_count=missing_count, page=page, max_page=max_page, base_url=base_url, single_language=single_language, page_size=page_size)
|
|
return output
|
|
|
|
@route(base_url + 'serieseditor')
|
|
@custom_auth_basic(check_credentials)
|
|
def serieseditor():
|
|
authorize()
|
|
single_language = get_general_settings()[7]
|
|
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
db.create_function("path_substitution", 1, path_replace)
|
|
c = db.cursor()
|
|
|
|
c.execute("SELECT COUNT(*) FROM table_shows")
|
|
missing_count = c.fetchone()
|
|
missing_count = missing_count[0]
|
|
|
|
c.execute("SELECT tvdbId, title, path_substitution(path), languages, hearing_impaired, sonarrSeriesId, poster, audio_language FROM table_shows ORDER BY title ASC")
|
|
data = c.fetchall()
|
|
c.execute("SELECT code2, name FROM table_settings_languages WHERE enabled = 1")
|
|
languages = c.fetchall()
|
|
c.close()
|
|
output = template('serieseditor', __file__=__file__, bazarr_version=bazarr_version, rows=data, languages=languages, missing_count=missing_count, base_url=base_url, single_language=single_language)
|
|
return output
|
|
|
|
|
|
@route(base_url + 'search_json/<query>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def search_json(query):
|
|
authorize()
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = db.cursor()
|
|
|
|
search_list = []
|
|
if get_general_settings()[12] is True:
|
|
c.execute("SELECT title, sonarrSeriesId FROM table_shows WHERE title LIKE ? ORDER BY title",
|
|
('%' + query + '%',))
|
|
series = c.fetchall()
|
|
for serie in series:
|
|
search_list.append(dict([('name', serie[0]), ('url', base_url + 'episodes/' + str(serie[1]))]))
|
|
|
|
if get_general_settings()[13] is True:
|
|
c.execute("SELECT title, radarrId FROM table_movies WHERE title LIKE ? ORDER BY title", ('%' + query + '%',))
|
|
movies = c.fetchall()
|
|
for movie in movies:
|
|
search_list.append(dict([('name', movie[0]), ('url', base_url + 'movie/' + str(movie[1]))]))
|
|
c.close()
|
|
|
|
response.content_type = 'application/json'
|
|
return dict(items=search_list)
|
|
|
|
|
|
@route(base_url + 'edit_series/<no:int>', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def edit_series(no):
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
lang = request.forms.getall('languages')
|
|
if len(lang) > 0:
|
|
pass
|
|
else:
|
|
lang = 'None'
|
|
|
|
single_language = get_general_settings()[7]
|
|
if single_language is True:
|
|
if str(lang) == "['None']":
|
|
lang = 'None'
|
|
else:
|
|
lang = str(lang)
|
|
else:
|
|
if str(lang) == "['']":
|
|
lang = '[]'
|
|
|
|
hi = request.forms.get('hearing_impaired')
|
|
|
|
if hi == "on":
|
|
hi = "True"
|
|
else:
|
|
hi = "False"
|
|
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = conn.cursor()
|
|
c.execute("UPDATE table_shows SET languages = ?, hearing_impaired = ? WHERE sonarrSeriesId LIKE ?", (str(lang), hi, no))
|
|
conn.commit()
|
|
c.close()
|
|
|
|
list_missing_subtitles(no)
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'edit_serieseditor', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def edit_serieseditor():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
series = request.forms.get('series')
|
|
series = ast.literal_eval(str('[' + series + ']'))
|
|
lang = request.forms.getall('languages')
|
|
hi = request.forms.get('hearing_impaired')
|
|
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = conn.cursor()
|
|
|
|
for serie in series:
|
|
if str(lang) != "[]" and str(lang) != "['']":
|
|
if str(lang) == "['None']":
|
|
lang = 'None'
|
|
else:
|
|
lang = str(lang)
|
|
c.execute("UPDATE table_shows SET languages = ? WHERE sonarrSeriesId LIKE ?", (lang, serie))
|
|
if hi != '':
|
|
c.execute("UPDATE table_shows SET hearing_impaired = ? WHERE sonarrSeriesId LIKE ?", (hi, serie))
|
|
|
|
conn.commit()
|
|
c.close()
|
|
|
|
for serie in series:
|
|
list_missing_subtitles(serie)
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'episodes/<no:int>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def episodes(no):
|
|
authorize()
|
|
# single_language = get_general_settings()[7]
|
|
url_sonarr_short = get_sonarr_settings()[7]
|
|
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
conn.create_function("path_substitution", 1, path_replace)
|
|
c = conn.cursor()
|
|
|
|
series_details = []
|
|
series_details = c.execute("SELECT title, overview, poster, fanart, hearing_impaired, tvdbid, audio_language, languages, path_substitution(path) FROM table_shows WHERE sonarrSeriesId LIKE ?", (str(no),)).fetchone()
|
|
tvdbid = series_details[5]
|
|
|
|
episodes = c.execute("SELECT title, path_substitution(path), season, episode, subtitles, sonarrSeriesId, missing_subtitles, sonarrEpisodeId, scene_name, monitored FROM table_episodes WHERE sonarrSeriesId LIKE ? ORDER BY episode ASC", (str(no),)).fetchall()
|
|
number = len(episodes)
|
|
languages = c.execute("SELECT code2, name FROM table_settings_languages WHERE enabled = 1").fetchall()
|
|
c.close()
|
|
episodes = reversed(sorted(episodes, key=operator.itemgetter(2)))
|
|
seasons_list = []
|
|
for key, season in itertools.groupby(episodes,operator.itemgetter(2)):
|
|
seasons_list.append(list(season))
|
|
|
|
return template('episodes', __file__=__file__, bazarr_version=bazarr_version, no=no, details=series_details, languages=languages, seasons=seasons_list, url_sonarr_short=url_sonarr_short, base_url=base_url, tvdbid=tvdbid, number=number)
|
|
|
|
@route(base_url + 'movies')
|
|
@custom_auth_basic(check_credentials)
|
|
def movies():
|
|
authorize()
|
|
single_language = get_general_settings()[7]
|
|
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
db.create_function("path_substitution", 1, path_replace_movie)
|
|
c = db.cursor()
|
|
|
|
c.execute("SELECT COUNT(*) FROM table_movies")
|
|
missing_count = c.fetchone()
|
|
missing_count = missing_count[0]
|
|
page = request.GET.page
|
|
if page == "":
|
|
page = "1"
|
|
page_size = int(get_general_settings()[21])
|
|
offset = (int(page) - 1) * page_size
|
|
max_page = int(math.ceil(missing_count / (page_size + 0.0)))
|
|
|
|
c.execute("SELECT tmdbId, title, path_substitution(path), languages, hearing_impaired, radarrId, poster, audio_language, monitored FROM table_movies ORDER BY title ASC LIMIT ? OFFSET ?", (page_size, offset,))
|
|
data = c.fetchall()
|
|
c.execute("SELECT code2, name FROM table_settings_languages WHERE enabled = 1")
|
|
languages = c.fetchall()
|
|
c.close()
|
|
output = template('movies', __file__=__file__, bazarr_version=bazarr_version, rows=data, languages=languages, missing_count=missing_count, page=page, max_page=max_page, base_url=base_url, single_language=single_language, page_size=page_size)
|
|
return output
|
|
|
|
@route(base_url + 'movieseditor')
|
|
@custom_auth_basic(check_credentials)
|
|
def movieseditor():
|
|
authorize()
|
|
single_language = get_general_settings()[7]
|
|
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
db.create_function("path_substitution", 1, path_replace_movie)
|
|
c = db.cursor()
|
|
|
|
c.execute("SELECT COUNT(*) FROM table_movies")
|
|
missing_count = c.fetchone()
|
|
missing_count = missing_count[0]
|
|
|
|
c.execute("SELECT tmdbId, title, path_substitution(path), languages, hearing_impaired, radarrId, poster, audio_language FROM table_movies ORDER BY title ASC")
|
|
data = c.fetchall()
|
|
c.execute("SELECT code2, name FROM table_settings_languages WHERE enabled = 1")
|
|
languages = c.fetchall()
|
|
c.close()
|
|
output = template('movieseditor', __file__=__file__, bazarr_version=bazarr_version, rows=data, languages=languages, missing_count=missing_count, base_url=base_url, single_language=single_language)
|
|
return output
|
|
|
|
@route(base_url + 'edit_movieseditor', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def edit_movieseditor():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
movies = request.forms.get('movies')
|
|
movies = ast.literal_eval(str('[' + movies + ']'))
|
|
lang = request.forms.getall('languages')
|
|
hi = request.forms.get('hearing_impaired')
|
|
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = conn.cursor()
|
|
|
|
for movie in movies:
|
|
if str(lang) != "[]" and str(lang) != "['']":
|
|
if str(lang) == "['None']":
|
|
lang = 'None'
|
|
else:
|
|
lang = str(lang)
|
|
c.execute("UPDATE table_movies SET languages = ? WHERE radarrId LIKE ?", (lang, movie))
|
|
if hi != '':
|
|
c.execute("UPDATE table_movies SET hearing_impaired = ? WHERE radarrId LIKE ?", (hi, movie))
|
|
|
|
conn.commit()
|
|
c.close()
|
|
|
|
for movie in movies:
|
|
list_missing_subtitles_movies(movie)
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'edit_movie/<no:int>', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def edit_movie(no):
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
lang = request.forms.getall('languages')
|
|
if len(lang) > 0:
|
|
pass
|
|
else:
|
|
lang = 'None'
|
|
|
|
if str(lang) == "['']":
|
|
lang = '[]'
|
|
|
|
hi = request.forms.get('hearing_impaired')
|
|
|
|
if hi == "on":
|
|
hi = "True"
|
|
else:
|
|
hi = "False"
|
|
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = conn.cursor()
|
|
c.execute("UPDATE table_movies SET languages = ?, hearing_impaired = ? WHERE radarrId LIKE ?", (str(lang), hi, no))
|
|
conn.commit()
|
|
c.close()
|
|
|
|
list_missing_subtitles_movies(no)
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'movie/<no:int>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def movie(no):
|
|
authorize()
|
|
# single_language = get_general_settings()[7]
|
|
url_radarr_short = get_radarr_settings()[7]
|
|
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
conn.create_function("path_substitution", 1, path_replace_movie)
|
|
c = conn.cursor()
|
|
|
|
movies_details = []
|
|
movies_details = c.execute("SELECT title, overview, poster, fanart, hearing_impaired, tmdbid, audio_language, languages, path_substitution(path), subtitles, radarrId, missing_subtitles, sceneName, monitored FROM table_movies WHERE radarrId LIKE ?", (str(no),)).fetchone()
|
|
tmdbid = movies_details[5]
|
|
|
|
languages = c.execute("SELECT code2, name FROM table_settings_languages WHERE enabled = 1").fetchall()
|
|
c.close()
|
|
|
|
return template('movie', __file__=__file__, bazarr_version=bazarr_version, no=no, details=movies_details, languages=languages, url_radarr_short=url_radarr_short, base_url=base_url, tmdbid=tmdbid)
|
|
|
|
@route(base_url + 'scan_disk/<no:int>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def scan_disk(no):
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
series_scan_subtitles(no)
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'scan_disk_movie/<no:int>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def scan_disk_movie(no):
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
movies_scan_subtitles(no)
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'search_missing_subtitles/<no:int>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def search_missing_subtitles(no):
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
series_download_subtitles(no)
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'search_missing_subtitles_movie/<no:int>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def search_missing_subtitles_movie(no):
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
movies_download_subtitles(no)
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'history')
|
|
@custom_auth_basic(check_credentials)
|
|
def history():
|
|
authorize()
|
|
return template('history', __file__=__file__, bazarr_version=bazarr_version, base_url=base_url)
|
|
|
|
@route(base_url + 'historyseries')
|
|
@custom_auth_basic(check_credentials)
|
|
def historyseries():
|
|
authorize()
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = db.cursor()
|
|
|
|
c.execute("SELECT COUNT(*) FROM table_history")
|
|
row_count = c.fetchone()
|
|
row_count = row_count[0]
|
|
page = request.GET.page
|
|
if page == "":
|
|
page = "1"
|
|
page_size = int(get_general_settings()[21])
|
|
offset = (int(page) - 1) * page_size
|
|
max_page = int(math.ceil(row_count / (page_size + 0.0)))
|
|
|
|
now = datetime.now()
|
|
today = []
|
|
thisweek = []
|
|
thisyear = []
|
|
stats = c.execute("SELECT timestamp FROM table_history WHERE action LIKE '1'").fetchall()
|
|
total = len(stats)
|
|
for stat in stats:
|
|
if now - timedelta(hours=24) <= datetime.fromtimestamp(stat[0]) <= now:
|
|
today.append(datetime.fromtimestamp(stat[0]).date())
|
|
if now - timedelta(weeks=1) <= datetime.fromtimestamp(stat[0]) <= now:
|
|
thisweek.append(datetime.fromtimestamp(stat[0]).date())
|
|
if now - timedelta(weeks=52) <= datetime.fromtimestamp(stat[0]) <= now:
|
|
thisyear.append(datetime.fromtimestamp(stat[0]).date())
|
|
stats = [len(today), len(thisweek), len(thisyear), total]
|
|
|
|
c.execute("SELECT table_history.action, table_shows.title, table_episodes.season || 'x' || table_episodes.episode, table_episodes.title, table_history.timestamp, table_history.description, table_history.sonarrSeriesId FROM table_history LEFT JOIN table_shows on table_shows.sonarrSeriesId = table_history.sonarrSeriesId LEFT JOIN table_episodes on table_episodes.sonarrEpisodeId = table_history.sonarrEpisodeId ORDER BY id DESC LIMIT ? OFFSET ?", (page_size, offset,))
|
|
data = c.fetchall()
|
|
c.close()
|
|
data = reversed(sorted(data, key=operator.itemgetter(4)))
|
|
return template('historyseries', __file__=__file__, bazarr_version=bazarr_version, rows=data, row_count=row_count, page=page, max_page=max_page, stats=stats, base_url=base_url, page_size=page_size)
|
|
|
|
@route(base_url + 'historymovies')
|
|
@custom_auth_basic(check_credentials)
|
|
def historymovies():
|
|
authorize()
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = db.cursor()
|
|
|
|
c.execute("SELECT COUNT(*) FROM table_history_movie")
|
|
row_count = c.fetchone()
|
|
row_count = row_count[0]
|
|
page = request.GET.page
|
|
if page == "":
|
|
page = "1"
|
|
page_size = int(get_general_settings()[21])
|
|
offset = (int(page) - 1) * page_size
|
|
max_page = int(math.ceil(row_count / (page_size + 0.0)))
|
|
|
|
now = datetime.now()
|
|
today = []
|
|
thisweek = []
|
|
thisyear = []
|
|
stats = c.execute("SELECT timestamp FROM table_history_movie WHERE action LIKE '1'").fetchall()
|
|
total = len(stats)
|
|
for stat in stats:
|
|
if now - timedelta(hours=24) <= datetime.fromtimestamp(stat[0]) <= now:
|
|
today.append(datetime.fromtimestamp(stat[0]).date())
|
|
if now - timedelta(weeks=1) <= datetime.fromtimestamp(stat[0]) <= now:
|
|
thisweek.append(datetime.fromtimestamp(stat[0]).date())
|
|
if now - timedelta(weeks=52) <= datetime.fromtimestamp(stat[0]) <= now:
|
|
thisyear.append(datetime.fromtimestamp(stat[0]).date())
|
|
stats = [len(today), len(thisweek), len(thisyear), total]
|
|
|
|
c.execute("SELECT table_history_movie.action, table_movies.title, table_history_movie.timestamp, table_history_movie.description, table_history_movie.radarrId FROM table_history_movie LEFT JOIN table_movies on table_movies.radarrId = table_history_movie.radarrId ORDER BY id DESC LIMIT ? OFFSET ?", (page_size, offset,))
|
|
data = c.fetchall()
|
|
c.close()
|
|
data = reversed(sorted(data, key=operator.itemgetter(2)))
|
|
return template('historymovies', __file__=__file__, bazarr_version=bazarr_version, rows=data, row_count=row_count, page=page, max_page=max_page, stats=stats, base_url=base_url, page_size=page_size)
|
|
|
|
@route(base_url + 'wanted')
|
|
@custom_auth_basic(check_credentials)
|
|
def wanted():
|
|
authorize()
|
|
return template('wanted', __file__=__file__, bazarr_version=bazarr_version, base_url=base_url)
|
|
|
|
@route(base_url + 'wantedseries')
|
|
@custom_auth_basic(check_credentials)
|
|
def wantedseries():
|
|
authorize()
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
db.create_function("path_substitution", 1, path_replace)
|
|
c = db.cursor()
|
|
|
|
if get_general_settings()[24] is True:
|
|
monitored_only_query_string = ' AND monitored = "True"'
|
|
else:
|
|
monitored_only_query_string = ""
|
|
|
|
c.execute("SELECT COUNT(*) FROM table_episodes WHERE missing_subtitles != '[]'" + monitored_only_query_string)
|
|
missing_count = c.fetchone()
|
|
missing_count = missing_count[0]
|
|
page = request.GET.page
|
|
if page == "":
|
|
page = "1"
|
|
page_size = int(get_general_settings()[21])
|
|
offset = (int(page) - 1) * page_size
|
|
max_page = int(math.ceil(missing_count / (page_size + 0.0)))
|
|
|
|
c.execute("SELECT table_shows.title, table_episodes.season || 'x' || table_episodes.episode, table_episodes.title, table_episodes.missing_subtitles, table_episodes.sonarrSeriesId, path_substitution(table_episodes.path), table_shows.hearing_impaired, table_episodes.sonarrEpisodeId, table_episodes.scene_name FROM table_episodes INNER JOIN table_shows on table_shows.sonarrSeriesId = table_episodes.sonarrSeriesId WHERE table_episodes.missing_subtitles != '[]'" + monitored_only_query_string + " ORDER BY table_episodes._rowid_ DESC LIMIT ? OFFSET ?", (page_size, offset,))
|
|
data = c.fetchall()
|
|
c.close()
|
|
return template('wantedseries', __file__=__file__, bazarr_version=bazarr_version, rows=data, missing_count=missing_count, page=page, max_page=max_page, base_url=base_url, page_size=page_size)
|
|
|
|
@route(base_url + 'wantedmovies')
|
|
@custom_auth_basic(check_credentials)
|
|
def wantedmovies():
|
|
authorize()
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
db.create_function("path_substitution", 1, path_replace_movie)
|
|
c = db.cursor()
|
|
|
|
if get_general_settings()[24] is True:
|
|
monitored_only_query_string = ' AND monitored = "True"'
|
|
else:
|
|
monitored_only_query_string = ""
|
|
|
|
c.execute("SELECT COUNT(*) FROM table_movies WHERE missing_subtitles != '[]'" + monitored_only_query_string)
|
|
missing_count = c.fetchone()
|
|
missing_count = missing_count[0]
|
|
page = request.GET.page
|
|
if page == "":
|
|
page = "1"
|
|
page_size = int(get_general_settings()[21])
|
|
offset = (int(page) - 1) * page_size
|
|
max_page = int(math.ceil(missing_count / (page_size + 0.0)))
|
|
|
|
c.execute("SELECT title, missing_subtitles, radarrId, path_substitution(path), hearing_impaired, sceneName FROM table_movies WHERE missing_subtitles != '[]'" + monitored_only_query_string + " ORDER BY _rowid_ DESC LIMIT ? OFFSET ?", (page_size, offset,))
|
|
data = c.fetchall()
|
|
c.close()
|
|
return template('wantedmovies', __file__=__file__, bazarr_version=bazarr_version, rows=data, missing_count=missing_count, page=page, max_page=max_page, base_url=base_url, page_size=page_size)
|
|
|
|
@route(base_url + 'wanted_search_missing_subtitles')
|
|
@custom_auth_basic(check_credentials)
|
|
def wanted_search_missing_subtitles_list():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
wanted_search_missing_subtitles()
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'settings')
|
|
@custom_auth_basic(check_credentials)
|
|
def settings():
|
|
authorize()
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = db.cursor()
|
|
c.execute("SELECT * FROM table_settings_languages ORDER BY name")
|
|
settings_languages = c.fetchall()
|
|
c.execute("SELECT * FROM table_settings_providers ORDER BY name")
|
|
settings_providers = c.fetchall()
|
|
c.execute("SELECT * FROM table_settings_notifier ORDER BY name")
|
|
settings_notifier = c.fetchall()
|
|
c.close()
|
|
|
|
from get_settings import get_general_settings, get_proxy_settings, get_auth_settings, get_radarr_settings, get_sonarr_settings
|
|
settings_general = get_general_settings()
|
|
settings_proxy = get_proxy_settings()
|
|
settings_auth = get_auth_settings()
|
|
settings_sonarr = get_sonarr_settings()
|
|
settings_radarr = get_radarr_settings()
|
|
|
|
return template('settings', __file__=__file__, bazarr_version=bazarr_version, settings_general=settings_general, settings_proxy=settings_proxy, settings_auth=settings_auth, settings_languages=settings_languages, settings_providers=settings_providers, settings_sonarr=settings_sonarr, settings_radarr=settings_radarr, settings_notifier=settings_notifier, base_url=base_url)
|
|
|
|
@route(base_url + 'save_settings', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def save_settings():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = conn.cursor()
|
|
|
|
settings_general_ip = request.forms.get('settings_general_ip')
|
|
settings_general_port = request.forms.get('settings_general_port')
|
|
settings_general_baseurl = request.forms.get('settings_general_baseurl')
|
|
if settings_general_baseurl.endswith('/') is False:
|
|
settings_general_baseurl += '/'
|
|
settings_general_loglevel = request.forms.get('settings_general_loglevel')
|
|
settings_general_sourcepath = request.forms.getall('settings_general_sourcepath')
|
|
settings_general_destpath = request.forms.getall('settings_general_destpath')
|
|
settings_general_pathmapping = []
|
|
settings_general_pathmapping.extend([list(a) for a in zip(settings_general_sourcepath, settings_general_destpath)])
|
|
settings_general_sourcepath_movie = request.forms.getall('settings_general_sourcepath_movie')
|
|
settings_general_destpath_movie = request.forms.getall('settings_general_destpath_movie')
|
|
settings_general_pathmapping_movie = []
|
|
settings_general_pathmapping_movie.extend([list(a) for a in zip(settings_general_sourcepath_movie, settings_general_destpath_movie)])
|
|
settings_general_branch = request.forms.get('settings_general_branch')
|
|
settings_general_automatic = request.forms.get('settings_general_automatic')
|
|
if settings_general_automatic is None:
|
|
settings_general_automatic = 'False'
|
|
else:
|
|
settings_general_automatic = 'True'
|
|
settings_general_single_language = request.forms.get('settings_general_single_language')
|
|
if settings_general_single_language is None:
|
|
settings_general_single_language = 'False'
|
|
else:
|
|
settings_general_single_language = 'True'
|
|
settings_general_scenename = request.forms.get('settings_general_scenename')
|
|
if settings_general_scenename is None:
|
|
settings_general_scenename = 'False'
|
|
else:
|
|
settings_general_scenename = 'True'
|
|
settings_general_embedded = request.forms.get('settings_general_embedded')
|
|
if settings_general_embedded is None:
|
|
settings_general_embedded = 'False'
|
|
else:
|
|
settings_general_embedded = 'True'
|
|
settings_general_only_monitored = request.forms.get('settings_general_only_monitored')
|
|
if settings_general_only_monitored is None:
|
|
settings_general_only_monitored = 'False'
|
|
else:
|
|
settings_general_only_monitored = 'True'
|
|
settings_general_adaptive_searching = request.forms.get('settings_general_adaptive_searching')
|
|
if settings_general_adaptive_searching is None:
|
|
settings_general_adaptive_searching = 'False'
|
|
else:
|
|
settings_general_adaptive_searching = 'True'
|
|
settings_general_minimum_score = request.forms.get('settings_general_minimum_score')
|
|
settings_general_minimum_score_movies = request.forms.get('settings_general_minimum_score_movies')
|
|
settings_general_use_postprocessing = request.forms.get('settings_general_use_postprocessing')
|
|
if settings_general_use_postprocessing is None:
|
|
settings_general_use_postprocessing = 'False'
|
|
else:
|
|
settings_general_use_postprocessing = 'True'
|
|
settings_general_postprocessing_cmd = request.forms.get('settings_general_postprocessing_cmd')
|
|
settings_general_use_sonarr = request.forms.get('settings_general_use_sonarr')
|
|
if settings_general_use_sonarr is None:
|
|
settings_general_use_sonarr = 'False'
|
|
else:
|
|
settings_general_use_sonarr = 'True'
|
|
settings_general_use_radarr = request.forms.get('settings_general_use_radarr')
|
|
if settings_general_use_radarr is None:
|
|
settings_general_use_radarr = 'False'
|
|
else:
|
|
settings_general_use_radarr = 'True'
|
|
settings_page_size = request.forms.get('settings_page_size')
|
|
|
|
settings_general = get_general_settings()
|
|
|
|
before = (unicode(settings_general[0]), int(settings_general[1]), unicode(settings_general[2]), unicode(settings_general[4]), unicode(settings_general[3]), unicode(settings_general[12]), unicode(settings_general[13]), unicode(settings_general[14]))
|
|
after = (unicode(settings_general_ip), int(settings_general_port), unicode(settings_general_baseurl), unicode(settings_general_loglevel), unicode(settings_general_pathmapping), unicode(settings_general_use_sonarr), unicode(settings_general_use_radarr), unicode(settings_general_pathmapping_movie))
|
|
from six import text_type
|
|
|
|
cfg = ConfigParser()
|
|
|
|
with open(config_file, 'r') as f:
|
|
cfg.read_file(f)
|
|
|
|
cfg.set('general', 'ip', text_type(settings_general_ip))
|
|
cfg.set('general', 'port', text_type(settings_general_port))
|
|
cfg.set('general', 'base_url', text_type(settings_general_baseurl))
|
|
cfg.set('general', 'path_mappings', text_type(settings_general_pathmapping))
|
|
cfg.set('general', 'log_level', text_type(settings_general_loglevel))
|
|
cfg.set('general', 'branch', text_type(settings_general_branch))
|
|
cfg.set('general', 'auto_update', text_type(settings_general_automatic))
|
|
cfg.set('general', 'single_language', text_type(settings_general_single_language))
|
|
cfg.set('general', 'minimum_score', text_type(settings_general_minimum_score))
|
|
cfg.set('general', 'use_scenename', text_type(settings_general_scenename))
|
|
cfg.set('general', 'use_postprocessing', text_type(settings_general_use_postprocessing))
|
|
cfg.set('general', 'postprocessing_cmd', text_type(settings_general_postprocessing_cmd))
|
|
cfg.set('general', 'use_sonarr', text_type(settings_general_use_sonarr))
|
|
cfg.set('general', 'use_radarr', text_type(settings_general_use_radarr))
|
|
cfg.set('general', 'path_mappings_movie', text_type(settings_general_pathmapping_movie))
|
|
cfg.set('general', 'page_size', text_type(settings_page_size))
|
|
cfg.set('general', 'minimum_score_movie', text_type(settings_general_minimum_score_movies))
|
|
cfg.set('general', 'use_embedded_subs', text_type(settings_general_embedded))
|
|
cfg.set('general', 'only_monitored', text_type(settings_general_only_monitored))
|
|
cfg.set('general', 'adaptive_searching', text_type(settings_general_adaptive_searching))
|
|
|
|
if after != before:
|
|
configured()
|
|
get_general_settings()
|
|
|
|
settings_proxy = get_proxy_settings()
|
|
|
|
if not cfg.has_section('proxy'):
|
|
cfg.add_section('proxy')
|
|
|
|
settings_proxy_type = request.forms.get('settings_proxy_type')
|
|
settings_proxy_url = request.forms.get('settings_proxy_url')
|
|
settings_proxy_port = request.forms.get('settings_proxy_port')
|
|
settings_proxy_username = request.forms.get('settings_proxy_username')
|
|
settings_proxy_password = request.forms.get('settings_proxy_password')
|
|
settings_proxy_exclude = request.forms.get('settings_proxy_exclude')
|
|
|
|
before_proxy_password = (unicode(settings_proxy[0]), unicode(settings_proxy[5]))
|
|
if before_proxy_password[0] != settings_proxy_type:
|
|
configured()
|
|
if before_proxy_password[1] == settings_proxy_password:
|
|
cfg.set('proxy', 'type', text_type(settings_proxy_type))
|
|
cfg.set('proxy', 'url', text_type(settings_proxy_url))
|
|
cfg.set('proxy', 'port', text_type(settings_proxy_port))
|
|
cfg.set('proxy', 'username', text_type(settings_proxy_username))
|
|
cfg.set('proxy', 'exclude', text_type(settings_proxy_exclude))
|
|
else:
|
|
cfg.set('proxy', 'type', text_type(settings_proxy_type))
|
|
cfg.set('proxy', 'url', text_type(settings_proxy_url))
|
|
cfg.set('proxy', 'port', text_type(settings_proxy_port))
|
|
cfg.set('proxy', 'username', text_type(settings_proxy_username))
|
|
cfg.set('proxy', 'password', text_type(settings_proxy_password))
|
|
cfg.set('proxy', 'exclude', text_type(settings_proxy_exclude))
|
|
|
|
settings_auth = get_auth_settings()
|
|
|
|
settings_auth_type = request.forms.get('settings_auth_type')
|
|
settings_auth_username = request.forms.get('settings_auth_username')
|
|
settings_auth_password = request.forms.get('settings_auth_password')
|
|
|
|
if get_auth_settings()[0] != settings_auth_type:
|
|
configured()
|
|
if settings_auth[2] == settings_auth_password:
|
|
cfg.set('auth', 'type', text_type(settings_auth_type))
|
|
cfg.set('auth', 'username', text_type(settings_auth_username))
|
|
else:
|
|
cfg.set('auth', 'type', text_type(settings_auth_type))
|
|
cfg.set('auth', 'username', text_type(settings_auth_username))
|
|
cfg.set('auth', 'password', hashlib.md5(settings_auth_password).hexdigest())
|
|
if settings_auth_username not in aaa._store.users:
|
|
cork = Cork(os.path.normpath(os.path.join(config_dir, 'config')), initialize=True)
|
|
cork._store.roles[''] = 100
|
|
cork._store.save_roles()
|
|
cork._store.users[settings_auth_username] = {
|
|
'role': '',
|
|
'hash': cork._hash(settings_auth_username, settings_auth_password),
|
|
'email_addr': '',
|
|
'desc': '',
|
|
'creation_date': time.time()
|
|
}
|
|
cork._store.save_users()
|
|
if settings_auth_type == 'basic' or settings_auth_type == 'None':
|
|
pass
|
|
else:
|
|
aaa._beaker_session.delete()
|
|
else:
|
|
if settings_auth[2] != settings_auth_password:
|
|
aaa.user(settings_auth_username).update(role='', pwd=settings_auth_password)
|
|
if settings_auth_type == 'basic' or settings_auth_type == 'None':
|
|
pass
|
|
else:
|
|
aaa._beaker_session.delete()
|
|
|
|
settings_sonarr_ip = request.forms.get('settings_sonarr_ip')
|
|
settings_sonarr_port = request.forms.get('settings_sonarr_port')
|
|
settings_sonarr_baseurl = request.forms.get('settings_sonarr_baseurl')
|
|
settings_sonarr_ssl = request.forms.get('settings_sonarr_ssl')
|
|
if settings_sonarr_ssl is None:
|
|
settings_sonarr_ssl = 'False'
|
|
else:
|
|
settings_sonarr_ssl = 'True'
|
|
settings_sonarr_apikey = request.forms.get('settings_sonarr_apikey')
|
|
settings_sonarr_sync = request.forms.get('settings_sonarr_sync')
|
|
|
|
cfg.set('sonarr', 'ip', text_type(settings_sonarr_ip))
|
|
cfg.set('sonarr', 'port', text_type(settings_sonarr_port))
|
|
cfg.set('sonarr', 'base_url', text_type(settings_sonarr_baseurl))
|
|
cfg.set('sonarr', 'ssl', text_type(settings_sonarr_ssl))
|
|
cfg.set('sonarr', 'apikey', text_type(settings_sonarr_apikey))
|
|
cfg.set('sonarr', 'full_update', text_type(settings_sonarr_sync))
|
|
|
|
settings_radarr_ip = request.forms.get('settings_radarr_ip')
|
|
settings_radarr_port = request.forms.get('settings_radarr_port')
|
|
settings_radarr_baseurl = request.forms.get('settings_radarr_baseurl')
|
|
settings_radarr_ssl = request.forms.get('settings_radarr_ssl')
|
|
if settings_radarr_ssl is None:
|
|
settings_radarr_ssl = 'False'
|
|
else:
|
|
settings_radarr_ssl = 'True'
|
|
settings_radarr_apikey = request.forms.get('settings_radarr_apikey')
|
|
settings_radarr_sync = request.forms.get('settings_radarr_sync')
|
|
|
|
cfg.set('radarr', 'ip', text_type(settings_radarr_ip))
|
|
cfg.set('radarr', 'port', text_type(settings_radarr_port))
|
|
cfg.set('radarr', 'base_url', text_type(settings_radarr_baseurl))
|
|
cfg.set('radarr', 'ssl', text_type(settings_radarr_ssl))
|
|
cfg.set('radarr', 'apikey', text_type(settings_radarr_apikey))
|
|
cfg.set('radarr', 'full_update', text_type(settings_radarr_sync))
|
|
|
|
settings_subliminal_providers = request.forms.getall('settings_subliminal_providers')
|
|
c.execute("UPDATE table_settings_providers SET enabled = 0")
|
|
for item in settings_subliminal_providers:
|
|
c.execute("UPDATE table_settings_providers SET enabled = '1' WHERE name = ?", (item,))
|
|
|
|
settings_addic7ed_username = request.forms.get('settings_addic7ed_username')
|
|
settings_addic7ed_password = request.forms.get('settings_addic7ed_password')
|
|
c.execute("UPDATE table_settings_providers SET username = ?, password = ? WHERE name = 'addic7ed'", (settings_addic7ed_username, settings_addic7ed_password))
|
|
settings_legendastv_username = request.forms.get('settings_legendastv_username')
|
|
settings_legendastv_password = request.forms.get('settings_legendastv_password')
|
|
c.execute("UPDATE table_settings_providers SET username = ?, password = ? WHERE name = 'legendastv'", (settings_legendastv_username, settings_legendastv_password))
|
|
settings_opensubtitles_username = request.forms.get('settings_opensubtitles_username')
|
|
settings_opensubtitles_password = request.forms.get('settings_opensubtitles_password')
|
|
c.execute("UPDATE table_settings_providers SET username = ?, password = ? WHERE name = 'opensubtitles'", (settings_opensubtitles_username, settings_opensubtitles_password))
|
|
|
|
settings_subliminal_languages = request.forms.getall('settings_subliminal_languages')
|
|
c.execute("UPDATE table_settings_languages SET enabled = 0")
|
|
for item in settings_subliminal_languages:
|
|
c.execute("UPDATE table_settings_languages SET enabled = '1' WHERE code2 = ?", (item,))
|
|
|
|
settings_serie_default_enabled = request.forms.get('settings_serie_default_enabled')
|
|
if settings_serie_default_enabled is None:
|
|
settings_serie_default_enabled = 'False'
|
|
else:
|
|
settings_serie_default_enabled = 'True'
|
|
cfg.set('general', 'serie_default_enabled', text_type(settings_serie_default_enabled))
|
|
|
|
settings_serie_default_languages = str(request.forms.getall('settings_serie_default_languages'))
|
|
if settings_serie_default_languages == "['None']":
|
|
settings_serie_default_languages = 'None'
|
|
cfg.set('general', 'serie_default_language', text_type(settings_serie_default_languages))
|
|
|
|
settings_serie_default_hi = request.forms.get('settings_serie_default_hi')
|
|
if settings_serie_default_hi is None:
|
|
settings_serie_default_hi = 'False'
|
|
else:
|
|
settings_serie_default_hi = 'True'
|
|
cfg.set('general', 'serie_default_hi', text_type(settings_serie_default_hi))
|
|
|
|
settings_movie_default_enabled = request.forms.get('settings_movie_default_enabled')
|
|
if settings_movie_default_enabled is None:
|
|
settings_movie_default_enabled = 'False'
|
|
else:
|
|
settings_movie_default_enabled = 'True'
|
|
cfg.set('general', 'movie_default_enabled', text_type(settings_movie_default_enabled))
|
|
|
|
settings_movie_default_languages = str(request.forms.getall('settings_movie_default_languages'))
|
|
if settings_movie_default_languages == "['None']":
|
|
settings_movie_default_languages = 'None'
|
|
cfg.set('general', 'movie_default_language', text_type(settings_movie_default_languages))
|
|
|
|
settings_movie_default_hi = request.forms.get('settings_movie_default_hi')
|
|
if settings_movie_default_hi is None:
|
|
settings_movie_default_hi = 'False'
|
|
else:
|
|
settings_movie_default_hi = 'True'
|
|
cfg.set('general', 'movie_default_hi', text_type(settings_movie_default_hi))
|
|
|
|
with open(config_file, 'wb') as f:
|
|
cfg.write(f)
|
|
|
|
settings_notifier_Boxcar_enabled = request.forms.get('settings_notifier_Boxcar_enabled')
|
|
if settings_notifier_Boxcar_enabled == 'on':
|
|
settings_notifier_Boxcar_enabled = 1
|
|
else:
|
|
settings_notifier_Boxcar_enabled = 0
|
|
settings_notifier_Boxcar_url = request.forms.get('settings_notifier_Boxcar_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Boxcar'", (settings_notifier_Boxcar_enabled, settings_notifier_Boxcar_url))
|
|
|
|
settings_notifier_Faast_enabled = request.forms.get('settings_notifier_Faast_enabled')
|
|
if settings_notifier_Faast_enabled == 'on':
|
|
settings_notifier_Faast_enabled = 1
|
|
else:
|
|
settings_notifier_Faast_enabled = 0
|
|
settings_notifier_Faast_url = request.forms.get('settings_notifier_Faast_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Faast'", (settings_notifier_Faast_enabled, settings_notifier_Faast_url))
|
|
|
|
settings_notifier_Growl_enabled = request.forms.get('settings_notifier_Growl_enabled')
|
|
if settings_notifier_Growl_enabled == 'on':
|
|
settings_notifier_Growl_enabled = 1
|
|
else:
|
|
settings_notifier_Growl_enabled = 0
|
|
settings_notifier_Growl_url = request.forms.get('settings_notifier_Growl_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Growl'", (settings_notifier_Growl_enabled, settings_notifier_Growl_url))
|
|
|
|
settings_notifier_Join_enabled = request.forms.get('settings_notifier_Join_enabled')
|
|
if settings_notifier_Join_enabled == 'on':
|
|
settings_notifier_Join_enabled = 1
|
|
else:
|
|
settings_notifier_Join_enabled = 0
|
|
settings_notifier_Join_url = request.forms.get('settings_notifier_Join_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Join'", (settings_notifier_Join_enabled, settings_notifier_Join_url))
|
|
|
|
settings_notifier_KODI_enabled = request.forms.get('settings_notifier_KODI_enabled')
|
|
if settings_notifier_KODI_enabled == 'on':
|
|
settings_notifier_KODI_enabled = 1
|
|
else:
|
|
settings_notifier_KODI_enabled = 0
|
|
settings_notifier_KODI_url = request.forms.get('settings_notifier_KODI_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'KODI'", (settings_notifier_KODI_enabled, settings_notifier_KODI_url))
|
|
|
|
settings_notifier_Mattermost_enabled = request.forms.get('settings_notifier_Mattermost_enabled')
|
|
if settings_notifier_Mattermost_enabled == 'on':
|
|
settings_notifier_Mattermost_enabled = 1
|
|
else:
|
|
settings_notifier_Mattermost_enabled = 0
|
|
settings_notifier_Mattermost_url = request.forms.get('settings_notifier_Mattermost_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Mattermost'", (settings_notifier_Mattermost_enabled, settings_notifier_Mattermost_url))
|
|
|
|
settings_notifier_NMA_enabled = request.forms.get('settings_notifier_Notify My Android_enabled')
|
|
if settings_notifier_NMA_enabled == 'on':
|
|
settings_notifier_NMA_enabled = 1
|
|
else:
|
|
settings_notifier_NMA_enabled = 0
|
|
settings_notifier_NMA_url = request.forms.get('settings_notifier_Notify My Android_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Notify My Android'", (settings_notifier_NMA_enabled, settings_notifier_NMA_url))
|
|
|
|
settings_notifier_Prowl_enabled = request.forms.get('settings_notifier_Prowl_enabled')
|
|
if settings_notifier_Prowl_enabled == 'on':
|
|
settings_notifier_Prowl_enabled = 1
|
|
else:
|
|
settings_notifier_Prowl_enabled = 0
|
|
settings_notifier_Prowl_url = request.forms.get('settings_notifier_Prowl_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Prowl'", (settings_notifier_Prowl_enabled, settings_notifier_Prowl_url))
|
|
|
|
settings_notifier_Pushalot_enabled = request.forms.get('settings_notifier_Pushalot_enabled')
|
|
if settings_notifier_Pushalot_enabled == 'on':
|
|
settings_notifier_Pushalot_enabled = 1
|
|
else:
|
|
settings_notifier_Pushalot_enabled = 0
|
|
settings_notifier_Pushalot_url = request.forms.get('settings_notifier_Pushalot_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Pushalot'", (settings_notifier_Pushalot_enabled, settings_notifier_Pushalot_url))
|
|
|
|
settings_notifier_PushBullet_enabled = request.forms.get('settings_notifier_PushBullet_enabled')
|
|
if settings_notifier_PushBullet_enabled == 'on':
|
|
settings_notifier_PushBullet_enabled = 1
|
|
else:
|
|
settings_notifier_PushBullet_enabled = 0
|
|
settings_notifier_PushBullet_url = request.forms.get('settings_notifier_PushBullet_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'PushBullet'", (settings_notifier_PushBullet_enabled, settings_notifier_PushBullet_url))
|
|
|
|
settings_notifier_Pushjet_enabled = request.forms.get('settings_notifier_Pushjet_enabled')
|
|
if settings_notifier_Pushjet_enabled == 'on':
|
|
settings_notifier_Pushjet_enabled = 1
|
|
else:
|
|
settings_notifier_Pushjet_enabled = 0
|
|
settings_notifier_Pushjet_url = request.forms.get('settings_notifier_Pushjet_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Pushjet'", (settings_notifier_Pushjet_enabled, settings_notifier_Pushjet_url))
|
|
|
|
settings_notifier_Pushover_enabled = request.forms.get('settings_notifier_Pushover_enabled')
|
|
if settings_notifier_Pushover_enabled == 'on':
|
|
settings_notifier_Pushover_enabled = 1
|
|
else:
|
|
settings_notifier_Pushover_enabled = 0
|
|
settings_notifier_Pushover_url = request.forms.get('settings_notifier_Pushover_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Pushover'", (settings_notifier_Pushover_enabled, settings_notifier_Pushover_url))
|
|
|
|
settings_notifier_RocketChat_enabled = request.forms.get('settings_notifier_Rocket.Chat_enabled')
|
|
if settings_notifier_RocketChat_enabled == 'on':
|
|
settings_notifier_RocketChat_enabled = 1
|
|
else:
|
|
settings_notifier_RocketChat_enabled = 0
|
|
settings_notifier_RocketChat_url = request.forms.get('settings_notifier_Rocket.Chat_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Rocket.Chat'", (settings_notifier_RocketChat_enabled, settings_notifier_RocketChat_url))
|
|
|
|
settings_notifier_Slack_enabled = request.forms.get('settings_notifier_Slack_enabled')
|
|
if settings_notifier_Slack_enabled == 'on':
|
|
settings_notifier_Slack_enabled = 1
|
|
else:
|
|
settings_notifier_Slack_enabled = 0
|
|
settings_notifier_Slack_url = request.forms.get('settings_notifier_Slack_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Slack'", (settings_notifier_Slack_enabled, settings_notifier_Slack_url))
|
|
|
|
settings_notifier_SuperToasty_enabled = request.forms.get('settings_notifier_Super Toasty_enabled')
|
|
if settings_notifier_SuperToasty_enabled == 'on':
|
|
settings_notifier_SuperToasty_enabled = 1
|
|
else:
|
|
settings_notifier_SuperToasty_enabled = 0
|
|
settings_notifier_SuperToasty_url = request.forms.get('settings_notifier_Super Toasty_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Super Toasty'", (settings_notifier_SuperToasty_enabled, settings_notifier_SuperToasty_url))
|
|
|
|
settings_notifier_Telegram_enabled = request.forms.get('settings_notifier_Telegram_enabled')
|
|
if settings_notifier_Telegram_enabled == 'on':
|
|
settings_notifier_Telegram_enabled = 1
|
|
else:
|
|
settings_notifier_Telegram_enabled = 0
|
|
settings_notifier_Telegram_url = request.forms.get('settings_notifier_Telegram_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Telegram'", (settings_notifier_Telegram_enabled, settings_notifier_Telegram_url))
|
|
|
|
settings_notifier_Twitter_enabled = request.forms.get('settings_notifier_Twitter_enabled')
|
|
if settings_notifier_Twitter_enabled == 'on':
|
|
settings_notifier_Twitter_enabled = 1
|
|
else:
|
|
settings_notifier_Twitter_enabled = 0
|
|
settings_notifier_Twitter_url = request.forms.get('settings_notifier_Twitter_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Twitter'", (settings_notifier_Twitter_enabled, settings_notifier_Twitter_url))
|
|
|
|
settings_notifier_XBMC_enabled = request.forms.get('settings_notifier_XBMC_enabled')
|
|
if settings_notifier_XBMC_enabled == 'on':
|
|
settings_notifier_XBMC_enabled = 1
|
|
else:
|
|
settings_notifier_XBMC_enabled = 0
|
|
settings_notifier_XBMC_url = request.forms.get('settings_notifier_XBMC_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'XBMC'", (settings_notifier_XBMC_enabled, settings_notifier_XBMC_url))
|
|
|
|
settings_notifier_Discord_enabled = request.forms.get('settings_notifier_Discord_enabled')
|
|
if settings_notifier_Discord_enabled == 'on':
|
|
settings_notifier_Discord_enabled = 1
|
|
else:
|
|
settings_notifier_Discord_enabled = 0
|
|
settings_notifier_Discord_url = request.forms.get('settings_notifier_Discord_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Discord'", (settings_notifier_Discord_enabled, settings_notifier_Discord_url))
|
|
|
|
settings_notifier_E_Mail_enabled = request.forms.get('settings_notifier_E-Mail_enabled')
|
|
if settings_notifier_E_Mail_enabled == 'on':
|
|
settings_notifier_E_Mail_enabled = 1
|
|
else:
|
|
settings_notifier_E_Mail_enabled = 0
|
|
settings_notifier_E_Mail_url = request.forms.get('settings_notifier_E-Mail_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'E-Mail'", (settings_notifier_E_Mail_enabled, settings_notifier_E_Mail_url))
|
|
|
|
settings_notifier_Emby_enabled = request.forms.get('settings_notifier_Emby_enabled')
|
|
if settings_notifier_Emby_enabled == 'on':
|
|
settings_notifier_Emby_enabled = 1
|
|
else:
|
|
settings_notifier_Emby_enabled = 0
|
|
settings_notifier_Emby_url = request.forms.get('settings_notifier_Emby_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Emby'", (settings_notifier_Emby_enabled, settings_notifier_Emby_url))
|
|
|
|
settings_notifier_IFTTT_enabled = request.forms.get('settings_notifier_IFTTT_enabled')
|
|
if settings_notifier_IFTTT_enabled == 'on':
|
|
settings_notifier_IFTTT_enabled = 1
|
|
else:
|
|
settings_notifier_IFTTT_enabled = 0
|
|
settings_notifier_IFTTT_url = request.forms.get('settings_notifier_IFTTT_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'IFTTT'", (settings_notifier_IFTTT_enabled, settings_notifier_IFTTT_url))
|
|
|
|
settings_notifier_Stride_enabled = request.forms.get('settings_notifier_Stride_enabled')
|
|
if settings_notifier_Stride_enabled == 'on':
|
|
settings_notifier_Stride_enabled = 1
|
|
else:
|
|
settings_notifier_Stride_enabled = 0
|
|
settings_notifier_Stride_url = request.forms.get('settings_notifier_Stride_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Stride'", (settings_notifier_Stride_enabled, settings_notifier_Stride_url))
|
|
|
|
settings_notifier_Windows_enabled = request.forms.get('settings_notifier_Windows_enabled')
|
|
if settings_notifier_Windows_enabled == 'on':
|
|
settings_notifier_Windows_enabled = 1
|
|
else:
|
|
settings_notifier_Windows_enabled = 0
|
|
settings_notifier_Windows_url = request.forms.get('settings_notifier_Windows_url')
|
|
c.execute("UPDATE table_settings_notifier SET enabled = ?, url = ? WHERE name = 'Windows'", (settings_notifier_Windows_enabled, settings_notifier_Windows_url))
|
|
|
|
conn.commit()
|
|
c.close()
|
|
|
|
sonarr_full_update()
|
|
radarr_full_update()
|
|
|
|
logging.info('Settings saved succesfully.')
|
|
|
|
# reschedule full update task according to settings
|
|
sonarr_full_update()
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'check_update')
|
|
@custom_auth_basic(check_credentials)
|
|
def check_update():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
if no_update is False:
|
|
check_and_apply_update()
|
|
|
|
redirect(ref)
|
|
|
|
@route(base_url + 'system')
|
|
@custom_auth_basic(check_credentials)
|
|
def system():
|
|
authorize()
|
|
def get_time_from_interval(interval):
|
|
interval_clean = interval.split('[')
|
|
interval_clean = interval_clean[1][:-1]
|
|
interval_split = interval_clean.split(':')
|
|
|
|
hour = interval_split[0]
|
|
minute = interval_split[1].lstrip("0")
|
|
second = interval_split[2].lstrip("0")
|
|
|
|
text = "every "
|
|
if hour != "0":
|
|
text = text + hour
|
|
if hour == "1":
|
|
text = text + " hour"
|
|
else:
|
|
text = text + " hours"
|
|
|
|
if minute != "" and second != "":
|
|
text = text + ", "
|
|
elif minute == "" and second != "":
|
|
text = text + " and "
|
|
elif minute != "" and second == "":
|
|
text = text + " and "
|
|
if minute != "":
|
|
text = text + minute
|
|
if minute == "1":
|
|
text = text + " minute"
|
|
else:
|
|
text = text + " minutes"
|
|
|
|
if second != "":
|
|
text = text + " and "
|
|
if second != "":
|
|
text = text + second
|
|
if second == "1":
|
|
text = text + " second"
|
|
else:
|
|
text = text + " seconds"
|
|
|
|
return text
|
|
|
|
def get_time_from_cron(cron):
|
|
text = "at "
|
|
hour = str(cron[5])
|
|
minute = str(cron[6])
|
|
second = str(cron[7])
|
|
|
|
if hour != "0" and hour != "*":
|
|
text = text + hour
|
|
if hour == "0" or hour == "1":
|
|
text = text + " hour"
|
|
else:
|
|
text = text + " hours"
|
|
|
|
if minute != "*" and second != "0":
|
|
text = text + ", "
|
|
elif minute == "*" and second != "0":
|
|
text = text + " and "
|
|
elif minute != "0" and minute != "*" and second == "0":
|
|
text = text + " and "
|
|
if minute != "0" and minute != "*":
|
|
text = text + minute
|
|
if minute == "0" or minute == "1":
|
|
text = text + " minute"
|
|
else:
|
|
text = text + " minutes"
|
|
|
|
if second != "0" and second != "*":
|
|
text = text + " and "
|
|
if second != "0" and second != "*":
|
|
text = text + second
|
|
if second == "0" or second == "1":
|
|
text = text + " second"
|
|
else:
|
|
text = text + " seconds"
|
|
|
|
return text
|
|
|
|
|
|
task_list = []
|
|
for job in scheduler.get_jobs():
|
|
if job.next_run_time is not None:
|
|
next_run = pretty.date(job.next_run_time.replace(tzinfo=None))
|
|
else:
|
|
next_run = "Never"
|
|
|
|
if job.trigger.__str__().startswith('interval'):
|
|
task_list.append([job.name, get_time_from_interval(str(job.trigger)), next_run, job.id])
|
|
elif job.trigger.__str__().startswith('cron'):
|
|
task_list.append([job.name, get_time_from_cron(job.trigger.fields), next_run, job.id])
|
|
|
|
i = 0
|
|
with open(os.path.join(config_dir, 'log/bazarr.log')) as f:
|
|
for i, l in enumerate(f, 1):
|
|
pass
|
|
row_count = i
|
|
page_size = int(get_general_settings()[21])
|
|
max_page = int(math.ceil(row_count / (page_size + 0.0)))
|
|
|
|
releases = []
|
|
url_releases = 'https://api.github.com/repos/morpheus65535/Bazarr/releases'
|
|
try:
|
|
r = requests.get(url_releases, timeout=15)
|
|
r.raise_for_status()
|
|
except requests.exceptions.HTTPError as errh:
|
|
logging.exception("Error trying to get releases from Github. Http error.")
|
|
except requests.exceptions.ConnectionError as errc:
|
|
logging.exception("Error trying to get releases from Github. Connection Error.")
|
|
except requests.exceptions.Timeout as errt:
|
|
logging.exception("Error trying to get releases from Github. Timeout Error.")
|
|
except requests.exceptions.RequestException as err:
|
|
logging.exception("Error trying to get releases from Github.")
|
|
else:
|
|
for release in r.json():
|
|
releases.append([release['name'],release['body']])
|
|
|
|
return template('system', __file__=__file__, bazarr_version=bazarr_version, base_url=base_url, task_list=task_list, row_count=row_count, max_page=max_page, page_size=page_size, releases=releases)
|
|
|
|
@route(base_url + 'logs/<page:int>')
|
|
@custom_auth_basic(check_credentials)
|
|
def get_logs(page):
|
|
authorize()
|
|
page_size = int(get_general_settings()[21])
|
|
begin = (page * page_size) - page_size
|
|
end = (page * page_size) - 1
|
|
logs_complete = []
|
|
for line in reversed(open(os.path.join(config_dir, 'log/bazarr.log')).readlines()):
|
|
logs_complete.append(line.rstrip())
|
|
logs = logs_complete[begin:end]
|
|
|
|
return template('logs', logs=logs, base_url=base_url)
|
|
|
|
@route(base_url + 'execute/<taskid>')
|
|
@custom_auth_basic(check_credentials)
|
|
def execute_task(taskid):
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
execute_now(taskid)
|
|
|
|
redirect(ref)
|
|
|
|
|
|
@route(base_url + 'remove_subtitles', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def remove_subtitles():
|
|
authorize()
|
|
episodePath = request.forms.get('episodePath')
|
|
language = request.forms.get('language')
|
|
subtitlesPath = request.forms.get('subtitlesPath')
|
|
sonarrSeriesId = request.forms.get('sonarrSeriesId')
|
|
sonarrEpisodeId = request.forms.get('sonarrEpisodeId')
|
|
|
|
try:
|
|
os.remove(subtitlesPath)
|
|
result = language_from_alpha3(language) + " subtitles deleted from disk."
|
|
history_log(0, sonarrSeriesId, sonarrEpisodeId, result)
|
|
except OSError:
|
|
pass
|
|
store_subtitles(unicode(episodePath))
|
|
list_missing_subtitles(sonarrSeriesId)
|
|
|
|
|
|
@route(base_url + 'remove_subtitles_movie', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def remove_subtitles_movie():
|
|
authorize()
|
|
moviePath = request.forms.get('moviePath')
|
|
language = request.forms.get('language')
|
|
subtitlesPath = request.forms.get('subtitlesPath')
|
|
radarrId = request.forms.get('radarrId')
|
|
|
|
try:
|
|
os.remove(subtitlesPath)
|
|
result = language_from_alpha3(language) + " subtitles deleted from disk."
|
|
history_log_movie(0, radarrId, result)
|
|
except OSError:
|
|
pass
|
|
store_subtitles_movie(unicode(moviePath))
|
|
list_missing_subtitles_movies(radarrId)
|
|
|
|
|
|
@route(base_url + 'get_subtitle', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def get_subtitle():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
episodePath = request.forms.get('episodePath')
|
|
sceneName = request.forms.get('sceneName')
|
|
language = request.forms.get('language')
|
|
hi = request.forms.get('hi')
|
|
sonarrSeriesId = request.forms.get('sonarrSeriesId')
|
|
sonarrEpisodeId = request.forms.get('sonarrEpisodeId')
|
|
# tvdbid = request.forms.get('tvdbid')
|
|
|
|
providers_list = get_providers()
|
|
providers_auth = get_providers_auth()
|
|
|
|
try:
|
|
result = download_subtitle(episodePath, language, hi, providers_list, providers_auth, sceneName, 'series')
|
|
if result is not None:
|
|
history_log(1, sonarrSeriesId, sonarrEpisodeId, result)
|
|
send_notifications(sonarrSeriesId, sonarrEpisodeId, result)
|
|
store_subtitles(unicode(episodePath))
|
|
list_missing_subtitles(sonarrSeriesId)
|
|
redirect(ref)
|
|
except OSError:
|
|
pass
|
|
|
|
@route(base_url + 'manual_search', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def manual_search_json():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
episodePath = request.forms.get('episodePath')
|
|
sceneName = request.forms.get('sceneName')
|
|
language = request.forms.get('language')
|
|
hi = request.forms.get('hi')
|
|
|
|
providers_list = get_providers()
|
|
providers_auth = get_providers_auth()
|
|
|
|
data = manual_search(episodePath, language, hi, providers_list, providers_auth, sceneName, 'series')
|
|
return dict(data=data)
|
|
|
|
@route(base_url + 'manual_get_subtitle', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def manual_get_subtitle():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
episodePath = request.forms.get('episodePath')
|
|
sceneName = request.forms.get('sceneName')
|
|
language = request.forms.get('language')
|
|
hi = request.forms.get('hi')
|
|
selected_provider = request.forms.get('provider')
|
|
subtitle = request.forms.get('subtitle')
|
|
sonarrSeriesId = request.forms.get('sonarrSeriesId')
|
|
sonarrEpisodeId = request.forms.get('sonarrEpisodeId')
|
|
|
|
providers_list = get_providers()
|
|
providers_auth = get_providers_auth()
|
|
|
|
try:
|
|
result = manual_download_subtitle(episodePath, language, hi, subtitle, selected_provider, providers_auth, sceneName, 'series')
|
|
if result is not None:
|
|
history_log(1, sonarrSeriesId, sonarrEpisodeId, result)
|
|
send_notifications(sonarrSeriesId, sonarrEpisodeId, result)
|
|
store_subtitles(unicode(episodePath))
|
|
list_missing_subtitles(sonarrSeriesId)
|
|
redirect(ref)
|
|
except OSError:
|
|
pass
|
|
|
|
@route(base_url + 'get_subtitle_movie', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def get_subtitle_movie():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
moviePath = request.forms.get('moviePath')
|
|
sceneName = request.forms.get('sceneName')
|
|
language = request.forms.get('language')
|
|
hi = request.forms.get('hi')
|
|
radarrId = request.forms.get('radarrId')
|
|
# tmdbid = request.forms.get('tmdbid')
|
|
|
|
providers_list = get_providers()
|
|
providers_auth = get_providers_auth()
|
|
|
|
try:
|
|
result = download_subtitle(moviePath, language, hi, providers_list, providers_auth, sceneName, 'movie')
|
|
if result is not None:
|
|
history_log_movie(1, radarrId, result)
|
|
send_notifications_movie(radarrId, result)
|
|
store_subtitles_movie(unicode(moviePath))
|
|
list_missing_subtitles_movies(radarrId)
|
|
redirect(ref)
|
|
except OSError:
|
|
pass
|
|
|
|
@route(base_url + 'manual_search_movie', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def manual_search_movie_json():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
moviePath = request.forms.get('moviePath')
|
|
sceneName = request.forms.get('sceneName')
|
|
language = request.forms.get('language')
|
|
hi = request.forms.get('hi')
|
|
|
|
providers_list = get_providers()
|
|
providers_auth = get_providers_auth()
|
|
|
|
data = manual_search(moviePath, language, hi, providers_list, providers_auth, sceneName, 'movie')
|
|
return dict(data=data)
|
|
|
|
@route(base_url + 'manual_get_subtitle_movie', method='POST')
|
|
@custom_auth_basic(check_credentials)
|
|
def manual_get_subtitle_movie():
|
|
authorize()
|
|
ref = request.environ['HTTP_REFERER']
|
|
|
|
moviePath = request.forms.get('moviePath')
|
|
sceneName = request.forms.get('sceneName')
|
|
language = request.forms.get('language')
|
|
hi = request.forms.get('hi')
|
|
selected_provider = request.forms.get('provider')
|
|
subtitle = request.forms.get('subtitle')
|
|
radarrId = request.forms.get('radarrId')
|
|
|
|
providers_list = get_providers()
|
|
providers_auth = get_providers_auth()
|
|
|
|
try:
|
|
result = manual_download_subtitle(moviePath, language, hi, subtitle, selected_provider, providers_auth, sceneName, 'movie')
|
|
if result is not None:
|
|
history_log_movie(1, radarrId, result)
|
|
send_notifications_movie(radarrId, result)
|
|
store_subtitles_movie(unicode(moviePath))
|
|
list_missing_subtitles_movies(radarrId)
|
|
redirect(ref)
|
|
except OSError:
|
|
pass
|
|
|
|
def configured():
|
|
conn = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = conn.cursor()
|
|
c.execute("UPDATE system SET configured = 1")
|
|
conn.commit()
|
|
c.close()
|
|
|
|
@route(base_url + 'api/series/wanted')
|
|
def api_wanted():
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = db.cursor()
|
|
data = c.execute("SELECT table_shows.title, table_episodes.season || 'x' || table_episodes.episode, table_episodes.title, table_episodes.missing_subtitles FROM table_episodes INNER JOIN table_shows on table_shows.sonarrSeriesId = table_episodes.sonarrSeriesId WHERE table_episodes.missing_subtitles != '[]' ORDER BY table_episodes._rowid_ DESC").fetchall()
|
|
c.close()
|
|
return dict(subtitles=data)
|
|
|
|
@route(base_url + 'api/series/history')
|
|
def api_history():
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = db.cursor()
|
|
data = c.execute("SELECT table_shows.title, table_episodes.season || 'x' || table_episodes.episode, table_episodes.title, strftime('%Y-%m-%d', datetime(table_history.timestamp, 'unixepoch')), table_history.description FROM table_history INNER JOIN table_shows on table_shows.sonarrSeriesId = table_history.sonarrSeriesId INNER JOIN table_episodes on table_episodes.sonarrEpisodeId = table_history.sonarrEpisodeId WHERE table_history.action = '1' ORDER BY id DESC").fetchall()
|
|
c.close()
|
|
return dict(subtitles=data)
|
|
|
|
@route(base_url + 'api/movies/wanted')
|
|
def api_wanted():
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = db.cursor()
|
|
data = c.execute("SELECT table_movies.title, table_movies.missing_subtitles FROM table_movies WHERE table_movies.missing_subtitles != '[]' ORDER BY table_movies._rowid_ DESC").fetchall()
|
|
c.close()
|
|
return dict(subtitles=data)
|
|
|
|
@route(base_url + 'api/movies/history')
|
|
def api_history():
|
|
db = sqlite3.connect(os.path.join(config_dir, 'db/bazarr.db'), timeout=30)
|
|
c = db.cursor()
|
|
data = c.execute("SELECT table_movies.title, strftime('%Y-%m-%d', datetime(table_history_movie.timestamp, 'unixepoch')), table_history_movie.description FROM table_history_movie INNER JOIN table_movies on table_movies.radarrId = table_history_movie.radarrId WHERE table_history_movie.action = '1' ORDER BY id DESC").fetchall()
|
|
c.close()
|
|
return dict(subtitles=data)
|
|
|
|
@route(base_url + 'test_url/<protocol>/<url:path>', method='GET')
|
|
@custom_auth_basic(check_credentials)
|
|
def test_url(protocol, url):
|
|
try:
|
|
result = requests.get(protocol + "://" + url, allow_redirects=False).json()['version']
|
|
except:
|
|
return dict(status=False)
|
|
else:
|
|
return dict(status=True, version=result)
|
|
|
|
import warnings
|
|
# Mute DeprecationWarning
|
|
warnings.simplefilter("ignore", DeprecationWarning)
|
|
|
|
logging.info('Bazarr is started and waiting for request on http://' + str(ip) + ':' + str(port) + str(base_url))
|
|
run(host=ip, port=port, server='waitress', app=app)
|
|
logging.info('Bazarr has been stopped.')
|