2019-05-12 15:56:18 +02:00
|
|
|
import os
|
2019-05-03 22:59:24 +02:00
|
|
|
import mimetypes
|
2019-07-05 19:30:24 +02:00
|
|
|
import json
|
2019-05-12 15:56:18 +02:00
|
|
|
from datetime import datetime
|
2019-07-11 13:40:12 +02:00
|
|
|
from string import Template
|
2019-05-04 16:56:23 +02:00
|
|
|
from owrx.websocket import WebSocketConnection
|
|
|
|
from owrx.config import PropertyManager
|
2019-05-12 18:10:24 +02:00
|
|
|
from owrx.source import ClientRegistry
|
2019-05-12 15:56:18 +02:00
|
|
|
from owrx.connection import WebSocketMessageHandler
|
2019-05-11 14:18:43 +02:00
|
|
|
from owrx.version import openwebrx_version
|
2019-07-05 19:30:24 +02:00
|
|
|
from owrx.feature import FeatureDetector
|
2019-08-04 18:36:03 +02:00
|
|
|
from owrx.metrics import Metrics
|
2019-05-03 22:59:24 +02:00
|
|
|
|
2019-05-10 21:50:58 +02:00
|
|
|
import logging
|
2019-07-21 19:40:28 +02:00
|
|
|
|
2019-05-10 21:50:58 +02:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2019-07-21 19:40:28 +02:00
|
|
|
|
2019-05-03 22:59:24 +02:00
|
|
|
class Controller(object):
|
2019-07-06 13:03:49 +02:00
|
|
|
def __init__(self, handler, request):
|
2019-05-03 22:59:24 +02:00
|
|
|
self.handler = handler
|
2019-07-06 13:03:49 +02:00
|
|
|
self.request = request
|
2019-07-21 19:40:28 +02:00
|
|
|
|
|
|
|
def send_response(self, content, code=200, content_type="text/html", last_modified: datetime = None, max_age=None):
|
2019-05-03 22:59:24 +02:00
|
|
|
self.handler.send_response(code)
|
|
|
|
if content_type is not None:
|
|
|
|
self.handler.send_header("Content-Type", content_type)
|
2019-05-09 16:12:05 +02:00
|
|
|
if last_modified is not None:
|
|
|
|
self.handler.send_header("Last-Modified", last_modified.strftime("%a, %d %b %Y %H:%M:%S GMT"))
|
|
|
|
if max_age is not None:
|
|
|
|
self.handler.send_header("Cache-Control", "max-age: {0}".format(max_age))
|
2019-05-03 22:59:24 +02:00
|
|
|
self.handler.end_headers()
|
2019-07-21 19:40:28 +02:00
|
|
|
if type(content) == str:
|
2019-05-03 22:59:24 +02:00
|
|
|
content = content.encode()
|
|
|
|
self.handler.wfile.write(content)
|
|
|
|
|
|
|
|
|
|
|
|
class StatusController(Controller):
|
|
|
|
def handle_request(self):
|
2019-05-10 23:00:18 +02:00
|
|
|
pm = PropertyManager.getSharedInstance()
|
|
|
|
# TODO keys that have been left out since they are no longer simple strings: sdr_hw, bands, antenna
|
|
|
|
vars = {
|
|
|
|
"status": "active",
|
|
|
|
"name": pm["receiver_name"],
|
|
|
|
"op_email": pm["receiver_admin"],
|
2019-05-12 18:10:24 +02:00
|
|
|
"users": ClientRegistry.getSharedInstance().clientCount(),
|
2019-05-10 23:00:18 +02:00
|
|
|
"users_max": pm["max_clients"],
|
|
|
|
"gps": pm["receiver_gps"],
|
|
|
|
"asl": pm["receiver_asl"],
|
|
|
|
"loc": pm["receiver_location"],
|
2019-05-11 14:18:43 +02:00
|
|
|
"sw_version": openwebrx_version,
|
2019-07-21 19:40:28 +02:00
|
|
|
"avatar_ctime": os.path.getctime("htdocs/gfx/openwebrx-avatar.png"),
|
2019-05-10 23:00:18 +02:00
|
|
|
}
|
2019-07-21 19:40:28 +02:00
|
|
|
self.send_response("\n".join(["{key}={value}".format(key=key, value=value) for key, value in vars.items()]))
|
|
|
|
|
2019-05-03 22:59:24 +02:00
|
|
|
|
|
|
|
class AssetsController(Controller):
|
2019-09-18 17:22:35 +02:00
|
|
|
def __init__(self, handler, request, path):
|
|
|
|
if not path.endswith("/"):
|
|
|
|
path += "/"
|
|
|
|
self.path = path
|
|
|
|
super().__init__(handler, request)
|
|
|
|
|
2019-07-21 19:40:28 +02:00
|
|
|
def serve_file(self, file, content_type=None):
|
2019-05-04 16:56:23 +02:00
|
|
|
try:
|
2019-09-18 17:22:35 +02:00
|
|
|
modified = datetime.fromtimestamp(os.path.getmtime(self.path + file))
|
2019-05-09 16:12:05 +02:00
|
|
|
|
|
|
|
if "If-Modified-Since" in self.handler.headers:
|
2019-07-21 19:40:28 +02:00
|
|
|
client_modified = datetime.strptime(
|
|
|
|
self.handler.headers["If-Modified-Since"], "%a, %d %b %Y %H:%M:%S %Z"
|
|
|
|
)
|
2019-05-09 16:12:05 +02:00
|
|
|
if modified <= client_modified:
|
2019-07-21 19:40:28 +02:00
|
|
|
self.send_response("", code=304)
|
2019-05-09 16:12:05 +02:00
|
|
|
return
|
|
|
|
|
2019-09-18 17:22:35 +02:00
|
|
|
f = open(self.path + file, "rb")
|
2019-05-04 16:56:23 +02:00
|
|
|
data = f.read()
|
|
|
|
f.close()
|
|
|
|
|
2019-05-09 16:12:05 +02:00
|
|
|
if content_type is None:
|
|
|
|
(content_type, encoding) = mimetypes.MimeTypes().guess_type(file)
|
2019-07-21 19:40:28 +02:00
|
|
|
self.send_response(data, content_type=content_type, last_modified=modified, max_age=3600)
|
2019-05-04 16:56:23 +02:00
|
|
|
except FileNotFoundError:
|
2019-07-21 19:40:28 +02:00
|
|
|
self.send_response("file not found", code=404)
|
|
|
|
|
2019-05-03 22:59:24 +02:00
|
|
|
def handle_request(self):
|
2019-07-06 13:03:49 +02:00
|
|
|
filename = self.request.matches.group(1)
|
2019-05-04 16:56:23 +02:00
|
|
|
self.serve_file(filename)
|
|
|
|
|
2019-07-21 19:40:28 +02:00
|
|
|
|
2019-09-18 17:22:35 +02:00
|
|
|
class OwrxAssetsController(AssetsController):
|
|
|
|
def __init__(self, handler, request):
|
|
|
|
super().__init__(handler, request, "htdocs/")
|
|
|
|
|
|
|
|
|
|
|
|
class AprsSymbolsController(AssetsController):
|
|
|
|
def __init__(self, handler, request):
|
|
|
|
pm = PropertyManager.getSharedInstance()
|
|
|
|
super().__init__(handler, request, pm["aprs_symbols_path"])
|
|
|
|
|
|
|
|
|
2019-07-11 13:40:12 +02:00
|
|
|
class TemplateController(Controller):
|
|
|
|
def render_template(self, file, **vars):
|
2019-10-06 16:05:30 +08:00
|
|
|
f = open("htdocs/" + file, "r", encoding="utf-8")
|
2019-07-11 13:40:12 +02:00
|
|
|
template = Template(f.read())
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
return template.safe_substitute(**vars)
|
|
|
|
|
|
|
|
def serve_template(self, file, **vars):
|
2019-07-21 19:40:28 +02:00
|
|
|
self.send_response(self.render_template(file, **vars), content_type="text/html")
|
2019-07-11 13:40:12 +02:00
|
|
|
|
|
|
|
def default_variables(self):
|
|
|
|
return {}
|
|
|
|
|
|
|
|
|
|
|
|
class WebpageController(TemplateController):
|
|
|
|
def template_variables(self):
|
2019-07-21 19:40:28 +02:00
|
|
|
header = self.render_template("include/header.include.html")
|
|
|
|
return {"header": header}
|
2019-07-11 13:40:12 +02:00
|
|
|
|
|
|
|
|
|
|
|
class IndexController(WebpageController):
|
2019-05-09 16:12:05 +02:00
|
|
|
def handle_request(self):
|
2019-07-11 13:40:12 +02:00
|
|
|
self.serve_template("index.html", **self.template_variables())
|
|
|
|
|
2019-05-09 16:12:05 +02:00
|
|
|
|
2019-07-11 13:40:12 +02:00
|
|
|
class MapController(WebpageController):
|
2019-07-01 16:49:39 +02:00
|
|
|
def handle_request(self):
|
2019-07-21 19:40:28 +02:00
|
|
|
# TODO check if we have a google maps api key first?
|
2019-07-11 13:40:12 +02:00
|
|
|
self.serve_template("map.html", **self.template_variables())
|
2019-07-05 19:30:24 +02:00
|
|
|
|
2019-07-21 19:40:28 +02:00
|
|
|
|
2019-07-11 16:44:33 +02:00
|
|
|
class FeatureController(WebpageController):
|
2019-07-05 19:30:24 +02:00
|
|
|
def handle_request(self):
|
2019-07-11 16:44:33 +02:00
|
|
|
self.serve_template("features.html", **self.template_variables())
|
2019-07-05 19:30:24 +02:00
|
|
|
|
2019-07-21 19:40:28 +02:00
|
|
|
|
2019-07-05 19:30:24 +02:00
|
|
|
class ApiController(Controller):
|
|
|
|
def handle_request(self):
|
|
|
|
data = json.dumps(FeatureDetector().feature_report())
|
2019-07-21 19:40:28 +02:00
|
|
|
self.send_response(data, content_type="application/json")
|
|
|
|
|
2019-07-01 16:49:39 +02:00
|
|
|
|
2019-08-04 18:36:03 +02:00
|
|
|
class MetricsController(Controller):
|
|
|
|
def handle_request(self):
|
|
|
|
data = json.dumps(Metrics.getSharedInstance().getMetrics())
|
|
|
|
self.send_response(data, content_type="application/json")
|
|
|
|
|
|
|
|
|
2019-05-04 20:26:11 +02:00
|
|
|
class WebSocketController(Controller):
|
|
|
|
def handle_request(self):
|
|
|
|
conn = WebSocketConnection(self.handler, WebSocketMessageHandler())
|
|
|
|
# enter read loop
|
2019-10-21 21:51:31 +02:00
|
|
|
conn.handle()
|