from abc import ABCMeta, abstractmethod from owrx.command import Option from owrx.source.connector import ConnectorSource, ConnectorDeviceDescription from typing import List from owrx.form import Input, TextInput from owrx.form.device import GainInput from owrx.soapy import SoapySettings class SoapyConnectorSource(ConnectorSource, metaclass=ABCMeta): def getCommandMapper(self): return ( super() .getCommandMapper() .setBase("soapy_connector") .setMappings( { "antenna": Option("-a"), "soapy_settings": Option("-t"), } ) ) """ must be implemented by child classes to be able to build a driver-based device selector by default. return value must be the corresponding soapy driver identifier. """ @abstractmethod def getDriver(self): pass def getEventNames(self): return super().getEventNames() + list(self.getSoapySettingsMappings().keys()) def buildSoapyDeviceParameters(self, parsed, values): """ this method always attempts to inject a driver= part into the soapysdr query, depending on what connector was used. this prevents the soapy_connector from using the wrong device in scenarios where there's no same-type sdrs. """ parsed = [v for v in parsed if "driver" not in v] parsed += [{"driver": self.getDriver()}] return parsed def getSoapySettingsMappings(self): return {} def buildSoapySettings(self, values): settings = {} for k, v in self.getSoapySettingsMappings().items(): if k in values and values[k] is not None: settings[v] = self.convertSoapySettingsValue(values[k]) return settings def convertSoapySettingsValue(self, value): if isinstance(value, bool): return "true" if value else "false" return value def getCommandValues(self): values = super().getCommandValues() if "device" in values and values["device"] is not None: parsed = SoapySettings.parse(values["device"]) else: parsed = [] modified = self.buildSoapyDeviceParameters(parsed, values) values["device"] = SoapySettings.encode(modified) settings = ",".join(["{0}={1}".format(k, v) for k, v in self.buildSoapySettings(values).items()]) if len(settings): values["soapy_settings"] = settings return values def onPropertyChange(self, changes): mappings = self.getSoapySettingsMappings() settings = {} for prop, value in changes.items(): if prop in mappings.keys(): settings[mappings[prop]] = self.convertSoapySettingsValue(value) if settings: changes["settings"] = ",".join("{0}={1}".format(k, v) for k, v in settings.items()) super().onPropertyChange(changes) class SoapyConnectorDeviceDescription(ConnectorDeviceDescription): def getInputs(self) -> List[Input]: return super().getInputs() + [ TextInput( "device", "Device Identifier", infotext='SoapySDR device identifier string (example: "serial=123456789")', ), GainInput( "rf_gain", "Device Gain", gain_stages=self.getGainStages(), ), TextInput("antenna", "Antenna"), ] def getOptionalKeys(self): return super().getOptionalKeys() + ["device", "rf_gain", "antenna"] def getGainStages(self): return None