2014-06-11 08:34:28 +00:00
|
|
|
import os
|
2014-07-02 18:51:14 +00:00
|
|
|
import socket
|
|
|
|
import time
|
|
|
|
import threading
|
|
|
|
import sys
|
2014-03-10 05:18:05 +00:00
|
|
|
import sickbeard
|
2014-06-11 08:34:28 +00:00
|
|
|
import webserve
|
2015-02-06 11:39:10 +00:00
|
|
|
import browser
|
2014-06-15 21:45:09 +00:00
|
|
|
import webapi
|
2014-06-15 11:08:41 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
from sickbeard import logger
|
|
|
|
from sickbeard.helpers import create_https_certificates
|
2015-02-06 11:39:10 +00:00
|
|
|
from tornado.web import Application, StaticFileHandler, HTTPError
|
2014-06-15 07:16:55 +00:00
|
|
|
from tornado.httpserver import HTTPServer
|
2014-06-16 12:19:07 +00:00
|
|
|
from tornado.ioloop import IOLoop
|
2014-06-11 08:34:28 +00:00
|
|
|
|
2014-07-09 18:41:04 +00:00
|
|
|
|
2014-06-11 08:34:28 +00:00
|
|
|
class MultiStaticFileHandler(StaticFileHandler):
|
|
|
|
def initialize(self, paths, default_filename=None):
|
|
|
|
self.paths = paths
|
2014-06-18 15:06:50 +00:00
|
|
|
self.default_filename = default_filename
|
2014-06-11 08:34:28 +00:00
|
|
|
|
|
|
|
def get(self, path, include_body=True):
|
|
|
|
for p in self.paths:
|
|
|
|
try:
|
|
|
|
# Initialize the Static file with a path
|
|
|
|
super(MultiStaticFileHandler, self).initialize(p)
|
|
|
|
# Try to get the file
|
|
|
|
return super(MultiStaticFileHandler, self).get(path)
|
|
|
|
except HTTPError as exc:
|
|
|
|
# File not found, carry on
|
|
|
|
if exc.status_code == 404:
|
|
|
|
continue
|
|
|
|
raise
|
|
|
|
|
|
|
|
# Oops file not found anywhere!
|
|
|
|
raise HTTPError(404)
|
|
|
|
|
2014-07-09 18:41:04 +00:00
|
|
|
|
2014-11-13 05:36:47 +00:00
|
|
|
class WebServer(threading.Thread):
|
2014-07-06 00:57:43 +00:00
|
|
|
def __init__(self, options={}, io_loop=None):
|
2014-07-02 18:51:14 +00:00
|
|
|
threading.Thread.__init__(self)
|
|
|
|
self.daemon = True
|
|
|
|
self.alive = True
|
2015-02-06 11:39:10 +00:00
|
|
|
self.name = 'TORNADO'
|
2014-07-02 18:51:14 +00:00
|
|
|
self.io_loop = io_loop or IOLoop.current()
|
|
|
|
|
|
|
|
self.options = options
|
|
|
|
self.options.setdefault('port', 8081)
|
|
|
|
self.options.setdefault('host', '0.0.0.0')
|
|
|
|
self.options.setdefault('log_dir', None)
|
|
|
|
self.options.setdefault('username', '')
|
|
|
|
self.options.setdefault('password', '')
|
2014-07-09 18:41:04 +00:00
|
|
|
self.options.setdefault('web_root', None)
|
2014-07-02 18:51:14 +00:00
|
|
|
assert isinstance(self.options['port'], int)
|
|
|
|
assert 'data_root' in self.options
|
|
|
|
|
2014-07-09 18:41:04 +00:00
|
|
|
# web root
|
|
|
|
self.options['web_root'] = ('/' + self.options['web_root'].lstrip('/')) if self.options[
|
|
|
|
'web_root'] else ''
|
|
|
|
|
2014-07-02 18:51:14 +00:00
|
|
|
# tornado setup
|
|
|
|
self.enable_https = self.options['enable_https']
|
|
|
|
self.https_cert = self.options['https_cert']
|
|
|
|
self.https_key = self.options['https_key']
|
|
|
|
|
|
|
|
if self.enable_https:
|
|
|
|
# If either the HTTPS certificate or key do not exist, make some self-signed ones.
|
2014-07-09 18:41:04 +00:00
|
|
|
if not (self.https_cert and os.path.exists(self.https_cert)) or not (
|
|
|
|
self.https_key and os.path.exists(self.https_key)):
|
2014-07-02 18:51:14 +00:00
|
|
|
if not create_https_certificates(self.https_cert, self.https_key):
|
2015-02-06 11:39:10 +00:00
|
|
|
logger.log(u'Unable to create CERT/KEY files, disabling HTTPS')
|
2014-07-02 18:51:14 +00:00
|
|
|
sickbeard.ENABLE_HTTPS = False
|
2014-07-06 00:57:43 +00:00
|
|
|
self.enable_https = False
|
2014-07-02 18:51:14 +00:00
|
|
|
|
|
|
|
if not (os.path.exists(self.https_cert) and os.path.exists(self.https_key)):
|
2015-02-06 11:39:10 +00:00
|
|
|
logger.log(u'Disabled HTTPS because of missing CERT and KEY files', logger.WARNING)
|
2014-03-10 05:18:05 +00:00
|
|
|
sickbeard.ENABLE_HTTPS = False
|
2014-07-06 00:57:43 +00:00
|
|
|
self.enable_https = False
|
2014-03-10 05:18:05 +00:00
|
|
|
|
2014-07-02 18:51:14 +00:00
|
|
|
# Load the app
|
|
|
|
self.app = Application([],
|
2014-07-09 18:41:04 +00:00
|
|
|
debug=True,
|
|
|
|
autoreload=False,
|
|
|
|
gzip=True,
|
|
|
|
xheaders=sickbeard.HANDLE_REVERSE_PROXY,
|
2015-02-06 11:39:10 +00:00
|
|
|
cookie_secret=sickbeard.COOKIE_SECRET,
|
|
|
|
login_url='%s/login/' % self.options['web_root'],
|
2014-07-02 18:51:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# Main Handler
|
2015-02-06 11:39:10 +00:00
|
|
|
self.app.add_handlers('.*$', [
|
|
|
|
(r'%s/api/builder(/?)(.*)' % self.options['web_root'], webapi.ApiBuilder),
|
|
|
|
(r'%s/api(/?.*)' % self.options['web_root'], webapi.Api),
|
|
|
|
(r'%s/config/general(/?.*)' % self.options['web_root'], webserve.ConfigGeneral),
|
|
|
|
(r'%s/config/search(/?.*)' % self.options['web_root'], webserve.ConfigSearch),
|
|
|
|
(r'%s/config/providers(/?.*)' % self.options['web_root'], webserve.ConfigProviders),
|
|
|
|
(r'%s/config/subtitles(/?.*)' % self.options['web_root'], webserve.ConfigSubtitles),
|
|
|
|
(r'%s/config/postProcessing(/?.*)' % self.options['web_root'], webserve.ConfigPostProcessing),
|
|
|
|
(r'%s/config/notifications(/?.*)' % self.options['web_root'], webserve.ConfigNotifications),
|
|
|
|
(r'%s/config/anime(/?.*)' % self.options['web_root'], webserve.ConfigAnime),
|
|
|
|
(r'%s/config(/?.*)' % self.options['web_root'], webserve.Config),
|
|
|
|
(r'%s/errorlogs(/?.*)' % self.options['web_root'], webserve.ErrorLogs),
|
|
|
|
(r'%s/history(/?.*)' % self.options['web_root'], webserve.History),
|
|
|
|
(r'%s/home/is_alive(/?.*)' % self.options['web_root'], webserve.IsAliveHandler),
|
|
|
|
(r'%s/home/addShows(/?.*)' % self.options['web_root'], webserve.NewHomeAddShows),
|
|
|
|
(r'%s/home/postprocess(/?.*)' % self.options['web_root'], webserve.HomePostProcess),
|
|
|
|
(r'%s/home(/?.*)' % self.options['web_root'], webserve.Home),
|
|
|
|
(r'%s/manage/manageSearches(/?.*)' % self.options['web_root'], webserve.ManageSearches),
|
|
|
|
(r'%s/manage/(/?.*)' % self.options['web_root'], webserve.Manage),
|
|
|
|
(r'%s/ui(/?.*)' % self.options['web_root'], webserve.UI),
|
|
|
|
(r'%s/browser(/?.*)' % self.options['web_root'], browser.WebFileBrowser),
|
|
|
|
(r'%s(/?.*)' % self.options['web_root'], webserve.MainHandler),
|
2014-07-02 18:51:14 +00:00
|
|
|
])
|
|
|
|
|
2015-02-06 11:39:10 +00:00
|
|
|
# webui login/logout handlers
|
|
|
|
self.app.add_handlers('.*$', [
|
|
|
|
(r'%s/login(/?)' % self.options['web_root'], webserve.LoginHandler),
|
|
|
|
(r'%s/logout(/?)' % self.options['web_root'], webserve.LogoutHandler),
|
|
|
|
])
|
|
|
|
|
|
|
|
# Web calendar handler (Needed because option Unprotected calendar)
|
|
|
|
self.app.add_handlers('.*$', [
|
|
|
|
(r'%s/calendar' % self.options['web_root'], webserve.CalendarHandler),
|
|
|
|
])
|
|
|
|
|
|
|
|
# Static File Handlers
|
|
|
|
self.app.add_handlers('.*$', [
|
|
|
|
# favicon
|
|
|
|
(r'%s/(favicon\.ico)' % self.options['web_root'], StaticFileHandler,
|
|
|
|
{'path': os.path.join(self.options['data_root'], 'images/ico/favicon.ico')}),
|
|
|
|
|
|
|
|
# images
|
|
|
|
(r'%s/images/(.*)' % self.options['web_root'], StaticFileHandler,
|
|
|
|
{'path': os.path.join(self.options['data_root'], 'images')}),
|
|
|
|
|
|
|
|
# cached images
|
|
|
|
(r'%s/cache/images/(.*)' % self.options['web_root'], StaticFileHandler,
|
|
|
|
{'path': os.path.join(sickbeard.CACHE_DIR, 'images')}),
|
|
|
|
|
|
|
|
# css
|
|
|
|
(r'%s/css/(.*)' % self.options['web_root'], StaticFileHandler,
|
|
|
|
{'path': os.path.join(self.options['data_root'], 'css')}),
|
|
|
|
|
|
|
|
# javascript
|
|
|
|
(r'%s/js/(.*)' % self.options['web_root'], StaticFileHandler,
|
|
|
|
{'path': os.path.join(self.options['data_root'], 'js')}),
|
2014-07-02 18:51:14 +00:00
|
|
|
])
|
|
|
|
|
2014-07-09 18:41:04 +00:00
|
|
|
|
2014-07-02 18:51:14 +00:00
|
|
|
def run(self):
|
|
|
|
if self.enable_https:
|
2015-02-06 11:39:10 +00:00
|
|
|
protocol = 'https'
|
|
|
|
self.server = HTTPServer(self.app, ssl_options={'certfile': self.https_cert, 'keyfile': self.https_key})
|
2014-07-02 18:51:14 +00:00
|
|
|
else:
|
2015-02-06 11:39:10 +00:00
|
|
|
protocol = 'http'
|
2014-07-24 06:36:44 +00:00
|
|
|
self.server = HTTPServer(self.app)
|
2014-07-02 18:51:14 +00:00
|
|
|
|
2015-02-06 11:39:10 +00:00
|
|
|
logger.log(u'Starting SickGear on ' + protocol + '://' + str(self.options['host']) + ':' + str(
|
|
|
|
self.options['port']) + '/')
|
2014-07-02 18:51:14 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
self.server.listen(self.options['port'], self.options['host'])
|
|
|
|
except:
|
|
|
|
etype, evalue, etb = sys.exc_info()
|
2014-07-09 18:41:04 +00:00
|
|
|
logger.log(
|
2015-02-06 11:39:10 +00:00
|
|
|
'Could not start webserver on %s. Excpeption: %s, Error: %s' % (self.options['port'], etype, evalue),
|
2014-07-09 18:41:04 +00:00
|
|
|
logger.ERROR)
|
2014-07-02 18:51:14 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.io_loop.start()
|
|
|
|
self.io_loop.close(True)
|
2014-07-15 02:00:53 +00:00
|
|
|
except (IOError, ValueError):
|
2015-02-06 11:39:10 +00:00
|
|
|
# Ignore errors like 'ValueError: I/O operation on closed kqueue fd'. These might be thrown during a reload.
|
2014-07-02 18:51:14 +00:00
|
|
|
pass
|
|
|
|
|
|
|
|
def shutDown(self):
|
|
|
|
self.alive = False
|
2014-12-14 19:01:32 +00:00
|
|
|
self.io_loop.stop()
|