2020-03-26 20:52:34 +00:00
|
|
|
from abc import ABC, abstractmethod
|
|
|
|
from .admin import AdminController
|
2020-03-26 21:08:24 +00:00
|
|
|
from owrx.config import Config
|
2020-03-29 17:50:37 +00:00
|
|
|
from owrx.service import ServiceDetector
|
2020-03-26 22:04:02 +00:00
|
|
|
from urllib.parse import parse_qs
|
|
|
|
import logging
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
2020-03-26 20:52:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Input(ABC):
|
|
|
|
def __init__(self, id, label, infotext=None):
|
|
|
|
self.id = id
|
|
|
|
self.label = label
|
|
|
|
self.infotext = infotext
|
|
|
|
|
|
|
|
def bootstrap_decorate(self, input):
|
|
|
|
infotext = "<small>{text}</small>".format(text=self.infotext) if self.infotext else ""
|
|
|
|
return """
|
|
|
|
<div class="form-group row">
|
|
|
|
<label class="col-form-label col-form-label-sm col-3" for="{id}">{label}</label>
|
|
|
|
<div class="col-9 p-0">
|
|
|
|
{input}
|
|
|
|
{infotext}
|
|
|
|
</div>
|
|
|
|
</div>
|
2020-03-27 20:11:33 +00:00
|
|
|
""".format(
|
|
|
|
id=self.id, label=self.label, input=input, infotext=infotext
|
|
|
|
)
|
2020-03-26 20:52:34 +00:00
|
|
|
|
|
|
|
def input_classes(self):
|
|
|
|
return " ".join(["form-control", "form-control-sm"])
|
|
|
|
|
|
|
|
@abstractmethod
|
2020-03-26 21:08:24 +00:00
|
|
|
def render_input(self, value):
|
2020-03-26 20:52:34 +00:00
|
|
|
pass
|
|
|
|
|
2020-03-26 21:08:24 +00:00
|
|
|
def render(self, config):
|
|
|
|
return self.bootstrap_decorate(self.render_input(config[self.id]))
|
2020-03-26 20:52:34 +00:00
|
|
|
|
2020-03-27 00:14:38 +00:00
|
|
|
def parse(self, data):
|
2020-03-26 22:04:02 +00:00
|
|
|
return {self.id: data[self.id][0]} if self.id in data else {}
|
|
|
|
|
2020-03-26 20:52:34 +00:00
|
|
|
|
|
|
|
class TextInput(Input):
|
2020-03-26 21:08:24 +00:00
|
|
|
def render_input(self, value):
|
2020-03-26 20:52:34 +00:00
|
|
|
return """
|
2020-03-26 21:08:24 +00:00
|
|
|
<input type="text" class="{classes}" id="{id}" name="{id}" placeholder="{label}" value="{value}">
|
2020-03-27 20:11:33 +00:00
|
|
|
""".format(
|
|
|
|
id=self.id, label=self.label, classes=self.input_classes(), value=value
|
|
|
|
)
|
2020-03-26 20:52:34 +00:00
|
|
|
|
|
|
|
|
2020-03-27 21:00:10 +00:00
|
|
|
class NumberInput(Input):
|
|
|
|
def render_input(self, value):
|
|
|
|
return """
|
|
|
|
<input type="number" class="{classes}" id="{id}" name="{id}" placeholder="{label}" value="{value}">
|
|
|
|
""".format(
|
|
|
|
id=self.id, label=self.label, classes=self.input_classes(), value=value
|
|
|
|
)
|
|
|
|
|
|
|
|
def convert_value(self, v):
|
|
|
|
return int(v)
|
|
|
|
|
|
|
|
def parse(self, data):
|
|
|
|
return {k: self.convert_value(v) for k, v in super().parse(data).items()}
|
|
|
|
|
|
|
|
|
|
|
|
class FloatInput(NumberInput):
|
|
|
|
def convert_value(self, v):
|
|
|
|
return float(v)
|
|
|
|
|
|
|
|
|
2020-03-26 20:52:34 +00:00
|
|
|
class LocationInput(Input):
|
2020-03-26 21:08:24 +00:00
|
|
|
def render_input(self, value):
|
2020-03-26 20:52:34 +00:00
|
|
|
# TODO make this work and pretty
|
|
|
|
return "Placeholder for a map widget to select receiver location"
|
|
|
|
|
|
|
|
|
|
|
|
class TextAreaInput(Input):
|
2020-03-26 21:08:24 +00:00
|
|
|
def render_input(self, value):
|
2020-03-26 20:52:34 +00:00
|
|
|
return """
|
2020-03-26 21:08:24 +00:00
|
|
|
<textarea class="{classes}" id="{id}" name="{id}" style="height:200px;">{value}</textarea>
|
2020-03-27 20:11:33 +00:00
|
|
|
""".format(
|
|
|
|
id=self.id, classes=self.input_classes(), value=value
|
|
|
|
)
|
2020-03-26 20:52:34 +00:00
|
|
|
|
|
|
|
|
2020-03-27 00:14:38 +00:00
|
|
|
class CheckboxInput(Input):
|
|
|
|
def __init__(self, id, label, checkboxText, infotext=None):
|
|
|
|
super().__init__(id, label, infotext=infotext)
|
|
|
|
self.checkboxText = checkboxText
|
2020-03-26 20:52:34 +00:00
|
|
|
|
2020-03-27 00:14:38 +00:00
|
|
|
def render_input(self, value):
|
|
|
|
return """
|
|
|
|
<div class="{classes}">
|
|
|
|
<input class="form-check-input" type="checkbox" id="{id}" name="{id}" {checked}>
|
|
|
|
<label class="form-check-label" for="{id}">
|
|
|
|
{checkboxText}
|
|
|
|
</label>
|
|
|
|
</div>
|
2020-03-27 20:11:33 +00:00
|
|
|
""".format(
|
|
|
|
id=self.id, classes=self.input_classes(), checked="checked" if value else "", checkboxText=self.checkboxText
|
|
|
|
)
|
2020-03-26 20:52:34 +00:00
|
|
|
|
2020-03-27 00:14:38 +00:00
|
|
|
def input_classes(self):
|
|
|
|
return " ".join(["form-check", "form-control-sm"])
|
|
|
|
|
|
|
|
def parse(self, data):
|
|
|
|
return {self.id: self.id in data and data[self.id][0] == "on"}
|
|
|
|
|
|
|
|
|
2020-03-29 17:52:56 +00:00
|
|
|
class Option(object):
|
|
|
|
# used for both MultiCheckboxInput and DropdownInput
|
|
|
|
def __init__(self, value, text):
|
|
|
|
self.value = value
|
|
|
|
self.text = text
|
|
|
|
|
|
|
|
|
2020-03-29 17:50:37 +00:00
|
|
|
class MultiCheckboxInput(Input):
|
|
|
|
def __init__(self, id, label, options, infotext=None):
|
|
|
|
super().__init__(id, label, infotext=infotext)
|
|
|
|
self.options = options
|
|
|
|
|
|
|
|
def render_input(self, value):
|
|
|
|
return "".join(self.render_checkbox(o, value) for o in self.options)
|
|
|
|
|
|
|
|
def checkbox_id(self, option):
|
|
|
|
return "{0}-{1}".format(self.id, option.value)
|
|
|
|
|
|
|
|
def render_checkbox(self, option, value):
|
|
|
|
return """
|
|
|
|
<div class="{classes}">
|
|
|
|
<input class="form-check-input" type="checkbox" id="{id}" name="{id}" {checked}>
|
|
|
|
<label class="form-check-label" for="{id}">
|
|
|
|
{checkboxText}
|
|
|
|
</label>
|
|
|
|
</div>
|
|
|
|
""".format(
|
|
|
|
id=self.checkbox_id(option),
|
|
|
|
classes=self.input_classes(),
|
|
|
|
checked="checked" if option.value in value else "",
|
|
|
|
checkboxText=option.text
|
|
|
|
)
|
|
|
|
|
|
|
|
def parse(self, data):
|
|
|
|
def in_response(option):
|
|
|
|
boxid = self.checkbox_id(option)
|
|
|
|
return boxid in data and data[boxid][0] == "on"
|
|
|
|
return {self.id: [o.value for o in self.options if in_response(o)]}
|
|
|
|
|
|
|
|
def input_classes(self):
|
|
|
|
return " ".join(["form-check", "form-control-sm"])
|
|
|
|
|
|
|
|
|
|
|
|
class ServicesCheckboxInput(MultiCheckboxInput):
|
|
|
|
def __init__(self, id, label, infotext=None):
|
2020-03-29 17:52:56 +00:00
|
|
|
services = [Option(s, s.upper()) for s in ServiceDetector.getAvailableServices()]
|
2020-03-29 17:50:37 +00:00
|
|
|
super().__init__(id, label, services, infotext)
|
|
|
|
|
|
|
|
|
2020-03-27 20:11:33 +00:00
|
|
|
class DropdownInput(Input):
|
|
|
|
def __init__(self, id, label, options, infotext = None):
|
|
|
|
super().__init__(id, label, infotext=infotext)
|
|
|
|
self.options = options
|
|
|
|
|
|
|
|
def render_input(self, value):
|
|
|
|
return """
|
|
|
|
<select class="{classes}" id="{id}" name="{id}">{options}</select>
|
|
|
|
""".format(classes=self.input_classes(), id=self.id, options=self.render_options(value))
|
|
|
|
|
|
|
|
def render_options(self, value):
|
|
|
|
options = [
|
|
|
|
"""
|
|
|
|
<option value="{value}" {selected}>{text}</option>
|
|
|
|
""".format(
|
|
|
|
text=o.text, value=o.value, selected="selected" if o.value == value else ""
|
|
|
|
) for o in self.options
|
|
|
|
]
|
|
|
|
return "".join(options)
|
|
|
|
|
|
|
|
|
2020-03-27 00:14:38 +00:00
|
|
|
class Section(object):
|
|
|
|
def __init__(self, title, *inputs):
|
|
|
|
self.title = title
|
|
|
|
self.inputs = inputs
|
|
|
|
|
|
|
|
def render_inputs(self):
|
2020-03-26 21:08:24 +00:00
|
|
|
config = Config.get()
|
2020-03-27 00:14:38 +00:00
|
|
|
return "".join([i.render(config) for i in self.inputs])
|
|
|
|
|
|
|
|
def render(self):
|
2020-03-26 22:04:02 +00:00
|
|
|
return """
|
2020-03-27 00:14:38 +00:00
|
|
|
<div class="col-12 settings-category">
|
|
|
|
<h3 class="settings-header">
|
|
|
|
{title}
|
|
|
|
</h3>
|
2020-03-26 22:04:02 +00:00
|
|
|
{inputs}
|
2020-03-27 00:14:38 +00:00
|
|
|
</div>
|
2020-03-27 20:11:33 +00:00
|
|
|
""".format(
|
|
|
|
title=self.title, inputs=self.render_inputs()
|
|
|
|
)
|
2020-03-27 00:14:38 +00:00
|
|
|
|
|
|
|
def parse(self, data):
|
|
|
|
return {k: v for i in self.inputs for k, v in i.parse(data).items()}
|
|
|
|
|
|
|
|
|
|
|
|
class SettingsController(AdminController):
|
|
|
|
sections = [
|
|
|
|
Section(
|
2020-03-27 20:11:33 +00:00
|
|
|
"General settings",
|
2020-03-27 00:14:38 +00:00
|
|
|
TextInput("receiver_name", "Receiver name"),
|
|
|
|
TextInput("receiver_location", "Receiver location"),
|
2020-03-27 21:00:10 +00:00
|
|
|
NumberInput("receiver_asl", "Receiver elevation", infotext="Elevation in meters above mean see level"),
|
2020-03-27 00:14:38 +00:00
|
|
|
TextInput("receiver_admin", "Receiver admin"),
|
|
|
|
LocationInput("receiver_gps", "Receiver coordinates"),
|
|
|
|
TextInput("photo_title", "Photo title"),
|
|
|
|
TextAreaInput("photo_desc", "Photo description"),
|
|
|
|
),
|
2020-03-27 20:11:33 +00:00
|
|
|
Section(
|
|
|
|
"Waterfall settings",
|
2020-03-27 21:00:10 +00:00
|
|
|
NumberInput(
|
2020-03-27 20:11:33 +00:00
|
|
|
"fft_fps",
|
|
|
|
"FFT frames per second",
|
|
|
|
infotext="This setting specifies how many lines are being added to the waterfall per second. "
|
|
|
|
+ "Higher values will give you a faster waterfall, but will also use more CPU.",
|
|
|
|
),
|
2020-03-27 21:00:10 +00:00
|
|
|
NumberInput("fft_size", "FFT size"),
|
|
|
|
FloatInput(
|
2020-03-27 20:11:33 +00:00
|
|
|
"fft_voverlap_factor",
|
|
|
|
"FFT vertical overlap factor",
|
|
|
|
infotext="If fft_voverlap_factor is above 0, multiple FFTs will be used for creating a line on the "
|
|
|
|
+ "diagram.",
|
|
|
|
),
|
2020-03-27 21:00:10 +00:00
|
|
|
NumberInput("waterfall_min_level", "Lowest waterfall level"),
|
|
|
|
NumberInput("waterfall_max_level", "Highest waterfall level"),
|
2020-03-27 20:11:33 +00:00
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"Compression",
|
|
|
|
DropdownInput("audio_compression", "Audio compression", options=[
|
2020-03-29 17:52:56 +00:00
|
|
|
Option("adpcm", "ADPCM"),
|
|
|
|
Option("none", "None"),
|
2020-03-27 20:11:33 +00:00
|
|
|
]),
|
|
|
|
DropdownInput("fft_compression", "Waterfall compression", options=[
|
2020-03-29 17:52:56 +00:00
|
|
|
Option("adpcm", "ADPCM"),
|
|
|
|
Option("none", "None"),
|
2020-03-27 20:11:33 +00:00
|
|
|
]),
|
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"Digimodes",
|
|
|
|
CheckboxInput("digimodes_enable", "", checkboxText="Enable Digimodes"),
|
2020-03-27 21:00:10 +00:00
|
|
|
NumberInput("digimodes_fft_size", "Digimodes FFT size"),
|
2020-03-27 20:11:33 +00:00
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"Digital voice",
|
2020-03-27 21:00:10 +00:00
|
|
|
NumberInput(
|
2020-03-27 20:11:33 +00:00
|
|
|
"digital_voice_unvoiced_quality",
|
|
|
|
"Quality of unvoiced sounds in synthesized voice",
|
|
|
|
infotext="Determines the quality, and thus the cpu usage, for the ambe codec used by digital voice"
|
|
|
|
+ "modes.<br />If you're running on a Raspi (up to 3B+) you should leave this set at 1"
|
|
|
|
),
|
|
|
|
CheckboxInput(
|
|
|
|
"digital_voice_dmr_id_lookup",
|
|
|
|
"DMR id lookup",
|
|
|
|
checkboxText="Enable lookup of DMR ids in the radioid database to show callsigns and names"
|
|
|
|
),
|
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"Experimental pipe settings",
|
|
|
|
CheckboxInput(
|
|
|
|
"csdr_dynamic_bufsize",
|
|
|
|
"",
|
|
|
|
checkboxText="Enable dynamic buffer sizes",
|
|
|
|
infotext="This allows you to change the buffering mode of csdr."
|
|
|
|
),
|
|
|
|
CheckboxInput(
|
|
|
|
"csdr_print_bufsizes",
|
|
|
|
"",
|
|
|
|
checkboxText="Print buffer sizez",
|
|
|
|
infotext="This prints the buffer sizes used for csdr processes."
|
|
|
|
),
|
|
|
|
CheckboxInput(
|
|
|
|
"csdr_through",
|
|
|
|
"",
|
|
|
|
checkboxText="Print throughput",
|
|
|
|
infotext="Enabling this will print out how much data is going into the DSP chains."
|
|
|
|
),
|
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"Map settings",
|
|
|
|
TextInput(
|
|
|
|
"google_maps_api_key",
|
|
|
|
"Google Maps API key",
|
|
|
|
infotext="Google Maps requires an API key, check out "
|
|
|
|
+ '<a href="https://developers.google.com/maps/documentation/embed/get-api-key" target="_blank">'
|
|
|
|
+ "their documentation</a> on how to obtain one."
|
|
|
|
),
|
2020-03-27 21:00:10 +00:00
|
|
|
NumberInput(
|
2020-03-27 20:11:33 +00:00
|
|
|
"map_position_retention_time",
|
|
|
|
"Map retention time",
|
|
|
|
infotext="Unit is seconds<br/>Specifies how log markers / grids will remain visible on the map"
|
|
|
|
),
|
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"WSJT-X settings",
|
2020-03-27 21:00:10 +00:00
|
|
|
NumberInput("wsjt_queue_workers", "Number of WSJT decoding workers"),
|
|
|
|
NumberInput("wsjt_queue_length", "Maximum length of WSJT job queue"),
|
|
|
|
NumberInput(
|
2020-03-27 20:11:33 +00:00
|
|
|
"wsjt_decoding_depth",
|
|
|
|
"WSJT decoding depth",
|
|
|
|
infotext="A higher decoding depth will allow more results, but will also consume more cpu"
|
|
|
|
)
|
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"Background decoding",
|
|
|
|
CheckboxInput("services_enabled", "Service", checkboxText="Enable background decoding services"),
|
2020-03-29 17:50:37 +00:00
|
|
|
ServicesCheckboxInput("services_decoders", "Enabled services")
|
2020-03-27 20:11:33 +00:00
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"APRS settings",
|
|
|
|
TextInput(
|
|
|
|
"aprs_callsign",
|
|
|
|
"APRS callsign",
|
|
|
|
infotext="This callsign will be used to send data to the APRS-IS network"
|
|
|
|
),
|
|
|
|
CheckboxInput(
|
|
|
|
"aprs_igate_enabled",
|
|
|
|
"APRS I-Gate",
|
|
|
|
checkboxText="Enable APRS receive-only I-Gate"
|
|
|
|
),
|
|
|
|
TextInput("aprs_igate_server", "APRS-IS server"),
|
|
|
|
TextInput("aprs_igate_password", "APRS-IS network password"),
|
|
|
|
CheckboxInput(
|
|
|
|
"aprs_igate_beacon",
|
|
|
|
"APRS beacon",
|
|
|
|
checkboxText="Send the receiver position to the APRS-IS network",
|
|
|
|
infotext="Please check that your receiver location is setup correctly"
|
|
|
|
),
|
|
|
|
),
|
|
|
|
Section(
|
|
|
|
"pskreporter settings",
|
|
|
|
CheckboxInput("pskreporter_enabled", "Reporting", checkboxText="Enable sending spots to pskreporter.info"),
|
|
|
|
TextInput(
|
|
|
|
"pskreporter_callsign",
|
|
|
|
"pskreporter callsign",
|
|
|
|
infotext="This callsign will be used to send spots to pskreporter.info"
|
|
|
|
),
|
|
|
|
),
|
2020-03-27 00:14:38 +00:00
|
|
|
Section(
|
|
|
|
"sdr.hu",
|
2020-03-27 20:11:33 +00:00
|
|
|
TextInput(
|
|
|
|
"sdrhu_key",
|
|
|
|
"sdr.hu key",
|
|
|
|
infotext='Please obtain your personal key on <a href="https://sdr.hu" target="_blank">sdr.hu</a>',
|
|
|
|
),
|
2020-03-27 23:40:36 +00:00
|
|
|
CheckboxInput("sdrhu_public_listing", "List on sdr.hu", "List my receiver on sdr.hu"),
|
2020-03-27 00:14:38 +00:00
|
|
|
TextInput("server_hostname", "Hostname"),
|
|
|
|
),
|
|
|
|
]
|
|
|
|
|
|
|
|
def render_sections(self):
|
|
|
|
sections = "".join(section.render() for section in SettingsController.sections)
|
|
|
|
return """
|
|
|
|
<form class="settings-body" method="POST">
|
|
|
|
{sections}
|
2020-03-26 22:04:02 +00:00
|
|
|
<div class="buttons">
|
|
|
|
<button type="submit" class="btn btn-primary">Apply</button>
|
|
|
|
</div>
|
|
|
|
</form>
|
2020-03-27 20:11:33 +00:00
|
|
|
""".format(
|
|
|
|
sections=sections
|
|
|
|
)
|
2020-03-27 00:14:38 +00:00
|
|
|
|
|
|
|
def indexAction(self):
|
|
|
|
self.serve_template("admin.html", **self.template_variables())
|
2020-03-26 20:52:34 +00:00
|
|
|
|
|
|
|
def template_variables(self):
|
|
|
|
variables = super().template_variables()
|
2020-03-27 00:14:38 +00:00
|
|
|
variables["sections"] = self.render_sections()
|
2020-03-26 20:52:34 +00:00
|
|
|
return variables
|
2020-03-26 22:04:02 +00:00
|
|
|
|
|
|
|
def processFormData(self):
|
|
|
|
data = parse_qs(self.get_body().decode("utf-8"))
|
2020-03-27 00:14:38 +00:00
|
|
|
data = {k: v for i in SettingsController.sections for k, v in i.parse(data).items()}
|
2020-03-26 22:04:02 +00:00
|
|
|
config = Config.get()
|
|
|
|
for k, v in data.items():
|
|
|
|
config[k] = v
|
2020-03-27 22:44:03 +00:00
|
|
|
Config.store()
|
2020-03-26 22:04:02 +00:00
|
|
|
self.send_redirect("/admin")
|