2020-02-23 16:22:13 +00:00
|
|
|
from . import Controller
|
2020-03-21 21:40:39 +00:00
|
|
|
from owrx.config import Config
|
2020-06-21 20:35:40 +00:00
|
|
|
from datetime import datetime, timezone
|
2020-02-23 16:22:13 +00:00
|
|
|
import mimetypes
|
|
|
|
import os
|
|
|
|
import pkg_resources
|
2020-05-02 11:35:42 +00:00
|
|
|
from abc import ABCMeta, abstractmethod
|
2020-09-04 13:44:25 +00:00
|
|
|
import gzip
|
|
|
|
|
|
|
|
import logging
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class GzipMixin(object):
|
|
|
|
def send_response(self, content, headers=None, content_type="text/html", *args, **kwargs):
|
|
|
|
if self.zipable(content_type) and "accept-encoding" in self.request.headers:
|
|
|
|
accepted = [s.strip().lower() for s in self.request.headers['accept-encoding'].split(",")]
|
|
|
|
if "gzip" in accepted:
|
|
|
|
if type(content) == str:
|
|
|
|
content = content.encode()
|
|
|
|
content = self.gzip(content)
|
|
|
|
if headers is None:
|
|
|
|
headers = {}
|
|
|
|
headers["Content-Encoding"] = "gzip"
|
|
|
|
super().send_response(content, headers=headers, content_type=content_type, *args, **kwargs)
|
|
|
|
|
|
|
|
def zipable(self, content_type):
|
|
|
|
types = [
|
|
|
|
"application/javascript",
|
|
|
|
"text/css",
|
|
|
|
"text/html"
|
|
|
|
]
|
|
|
|
return content_type in types
|
|
|
|
|
|
|
|
def gzip(self, content):
|
|
|
|
return gzip.compress(content)
|
2020-02-23 16:22:13 +00:00
|
|
|
|
2020-05-02 11:35:42 +00:00
|
|
|
|
2020-09-04 13:44:25 +00:00
|
|
|
class ModificationAwareController(Controller, metaclass=ABCMeta):
|
2020-06-21 20:35:40 +00:00
|
|
|
@abstractmethod
|
|
|
|
def getModified(self, file):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def wasModified(self, file):
|
|
|
|
try:
|
2020-07-01 17:10:46 +00:00
|
|
|
modified = self.getModified(file).replace(microsecond=0)
|
2020-06-21 20:35:40 +00:00
|
|
|
|
|
|
|
if modified is not None and "If-Modified-Since" in self.handler.headers:
|
|
|
|
client_modified = datetime.strptime(
|
|
|
|
self.handler.headers["If-Modified-Since"], "%a, %d %b %Y %H:%M:%S %Z"
|
|
|
|
).replace(tzinfo=timezone.utc)
|
|
|
|
if modified <= client_modified:
|
|
|
|
return False
|
|
|
|
except FileNotFoundError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2020-09-04 13:44:25 +00:00
|
|
|
class AssetsController(GzipMixin, ModificationAwareController, metaclass=ABCMeta):
|
2020-02-23 16:22:13 +00:00
|
|
|
def getModified(self, file):
|
2020-07-01 17:10:46 +00:00
|
|
|
return datetime.fromtimestamp(os.path.getmtime(self.getFilePath(file)), timezone.utc)
|
2020-02-23 16:22:13 +00:00
|
|
|
|
|
|
|
def openFile(self, file):
|
2020-05-02 11:35:42 +00:00
|
|
|
return open(self.getFilePath(file), "rb")
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def getFilePath(self, file):
|
2020-02-23 16:22:13 +00:00
|
|
|
pass
|
|
|
|
|
|
|
|
def serve_file(self, file, content_type=None):
|
|
|
|
try:
|
|
|
|
modified = self.getModified(file)
|
|
|
|
|
2020-06-21 20:35:40 +00:00
|
|
|
if not self.wasModified(file):
|
|
|
|
self.send_response("", code=304)
|
|
|
|
return
|
2020-02-23 16:22:13 +00:00
|
|
|
|
|
|
|
f = self.openFile(file)
|
|
|
|
data = f.read()
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
if content_type is None:
|
|
|
|
(content_type, encoding) = mimetypes.MimeTypes().guess_type(file)
|
|
|
|
self.send_response(data, content_type=content_type, last_modified=modified, max_age=3600)
|
|
|
|
except FileNotFoundError:
|
|
|
|
self.send_response("file not found", code=404)
|
|
|
|
|
2020-02-23 17:32:37 +00:00
|
|
|
def indexAction(self):
|
2020-02-23 16:22:13 +00:00
|
|
|
filename = self.request.matches.group(1)
|
|
|
|
self.serve_file(filename)
|
|
|
|
|
|
|
|
|
|
|
|
class OwrxAssetsController(AssetsController):
|
2020-05-02 11:35:42 +00:00
|
|
|
def getFilePath(self, file):
|
|
|
|
return pkg_resources.resource_filename("htdocs", file)
|
2020-02-23 16:22:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AprsSymbolsController(AssetsController):
|
2020-02-27 17:43:44 +00:00
|
|
|
def __init__(self, handler, request, options):
|
2020-03-21 21:40:39 +00:00
|
|
|
pm = Config.get()
|
2020-02-23 16:22:13 +00:00
|
|
|
path = pm["aprs_symbols_path"]
|
|
|
|
if not path.endswith("/"):
|
|
|
|
path += "/"
|
|
|
|
self.path = path
|
2020-02-27 17:43:44 +00:00
|
|
|
super().__init__(handler, request, options)
|
2020-02-23 16:22:13 +00:00
|
|
|
|
|
|
|
def getFilePath(self, file):
|
|
|
|
return self.path + file
|
|
|
|
|
|
|
|
|
2020-09-04 13:44:25 +00:00
|
|
|
class CompiledAssetsController(GzipMixin, ModificationAwareController):
|
2020-05-02 11:35:42 +00:00
|
|
|
profiles = {
|
|
|
|
"receiver.js": [
|
2020-09-17 19:33:11 +00:00
|
|
|
"lib/chroma.min.js",
|
2020-05-02 11:35:42 +00:00
|
|
|
"openwebrx.js",
|
|
|
|
"lib/jquery-3.2.1.min.js",
|
2020-11-04 21:32:13 +00:00
|
|
|
"lib/jquery.nanoscroller.min.js",
|
2020-05-08 22:11:20 +00:00
|
|
|
"lib/Header.js",
|
2020-05-02 11:35:42 +00:00
|
|
|
"lib/Demodulator.js",
|
|
|
|
"lib/DemodulatorPanel.js",
|
|
|
|
"lib/BookmarkBar.js",
|
2020-05-03 21:56:22 +00:00
|
|
|
"lib/BookmarkDialog.js",
|
2020-05-02 11:35:42 +00:00
|
|
|
"lib/AudioEngine.js",
|
|
|
|
"lib/ProgressBar.js",
|
|
|
|
"lib/Measurement.js",
|
|
|
|
"lib/FrequencyDisplay.js",
|
|
|
|
"lib/Js8Threads.js",
|
|
|
|
"lib/Modes.js",
|
2020-05-02 11:57:19 +00:00
|
|
|
],
|
2020-05-10 14:12:37 +00:00
|
|
|
"map.js": [
|
|
|
|
"lib/jquery-3.2.1.min.js",
|
|
|
|
"lib/chroma.min.js",
|
|
|
|
"lib/Header.js",
|
|
|
|
"map.js",
|
|
|
|
],
|
2020-05-31 18:25:41 +00:00
|
|
|
"settings.js": [
|
|
|
|
"lib/jquery-3.2.1.min.js",
|
|
|
|
"lib/Header.js",
|
|
|
|
"lib/settings/Input.js",
|
|
|
|
"lib/settings/SdrDevice.js",
|
|
|
|
"settings.js",
|
|
|
|
]
|
2020-05-02 11:35:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def indexAction(self):
|
|
|
|
profileName = self.request.matches.group(1)
|
|
|
|
if profileName not in CompiledAssetsController.profiles:
|
|
|
|
self.send_response("profile not found", code=404)
|
2020-05-02 14:59:27 +00:00
|
|
|
return
|
2020-05-02 11:35:42 +00:00
|
|
|
|
|
|
|
files = CompiledAssetsController.profiles[profileName]
|
|
|
|
files = [pkg_resources.resource_filename("htdocs", f) for f in files]
|
|
|
|
|
|
|
|
modified = self.getModified(files)
|
|
|
|
|
2020-06-21 20:35:40 +00:00
|
|
|
if not self.wasModified(files):
|
|
|
|
self.send_response("", code=304)
|
|
|
|
return
|
2020-05-02 11:35:42 +00:00
|
|
|
|
|
|
|
contents = [self.getContents(f) for f in files]
|
|
|
|
|
|
|
|
(content_type, encoding) = mimetypes.MimeTypes().guess_type(profileName)
|
|
|
|
self.send_response("\n".join(contents), content_type=content_type, last_modified=modified, max_age=3600)
|
|
|
|
|
|
|
|
def getContents(self, file):
|
|
|
|
with open(file) as f:
|
|
|
|
return f.read()
|
|
|
|
|
|
|
|
def getModified(self, files):
|
2020-07-01 17:10:46 +00:00
|
|
|
modified = [os.path.getmtime(f) for f in files]
|
|
|
|
return datetime.fromtimestamp(max(*modified), timezone.utc)
|