4 Commits

Author SHA1 Message Date
bc50f0160e feat(cli): Add CLI support for RevPi configuration object (revpi-config)
This implements a new command "config" in the CLI to handle RevPi
configuration. It includes parsing and subparser setup for
configuration-related operations. The change improves usability by
extending CLI functionality to manage RevPi configuration objects.
2025-04-21 10:50:45 +02:00
6cd351d8b8 feat(cli): Add 'list-features' action to CLI config commands
Introduce a new 'list-features' action to display all available features
via CLI. Updated argument handling to make 'feature' optional for
'list-features' and added a validation step to ensure the feature name
is provided for other actions.
2025-04-21 10:50:20 +02:00
0e513cd179 feat(cli): Add get_properties helper function for DBus interactions
This function facilitates retrieving specific properties from a DBus
interface, improving code modularity and reusability. It supports both
system and session bus types, streamlining access to DBus resources.
2025-04-21 10:49:47 +02:00
b2a6a184a4 fixup 2025-04-21 10:34:44 +02:00
2 changed files with 37 additions and 41 deletions

View File

@@ -13,7 +13,7 @@ from .revpi_config import (
configure_dphys_swapfile, configure_dphys_swapfile,
configure_external_antenna, configure_external_antenna,
configure_gui, configure_gui,
configure_wlan, configure_wifi,
simple_systemd, simple_systemd,
) )
from ..dbus_helper import DbusInterface from ..dbus_helper import DbusInterface
@@ -94,7 +94,7 @@ AVAILABLE_FEATURES = {
), ),
"revpipyload": FeatureFunction(simple_systemd, ["revpipyload.service"]), "revpipyload": FeatureFunction(simple_systemd, ["revpipyload.service"]),
"bluetooth": FeatureFunction(configure_bluetooth, []), "bluetooth": FeatureFunction(configure_bluetooth, []),
"wlan": FeatureFunction(configure_wlan, []), "ieee80211": FeatureFunction(configure_wifi, []),
"avahi": FeatureFunction(configure_avahi_daemon, []), "avahi": FeatureFunction(configure_avahi_daemon, []),
"external-antenna": FeatureFunction(configure_external_antenna, []), "external-antenna": FeatureFunction(configure_external_antenna, []),
} }

View File

@@ -21,7 +21,7 @@ log = getLogger(__name__)
ConfigVariable = namedtuple("ConfigVariable", ["name", "value", "line_index"]) ConfigVariable = namedtuple("ConfigVariable", ["name", "value", "line_index"])
LINUX_BT_CLASS_PATH = "/sys/class/bluetooth" LINUX_BT_CLASS_PATH = "/sys/class/bluetooth"
LINUX_WLAN_CLASS_PATH = "/sys/class/ieee80211" LINUX_WIFI_CLASS_PATH = "/sys/class/ieee80211"
CONFIG_TXT_LOCATIONS = ("/boot/firmware/config.txt", "/boot/config.txt") CONFIG_TXT_LOCATIONS = ("/boot/firmware/config.txt", "/boot/config.txt")
@@ -81,19 +81,19 @@ class RevPiConfig:
except ValueError: except ValueError:
pass pass
# Detect WLAN on CM module # Detect WiFi on CM module
could_have_wlan = self._cm_type in (ComputeModuleTypes.CM4, ComputeModuleTypes.CM5) could_have_wifi = self._cm_type in (ComputeModuleTypes.CM4, ComputeModuleTypes.CM5)
if could_have_wlan: if could_have_wifi:
wlan_interface = join(LINUX_WLAN_CLASS_PATH, "phy0") wifi_interface = join(LINUX_WIFI_CLASS_PATH, "phy0")
if grep("DRIVER=brcmfmac", join(wlan_interface, "device", "uevent")): if grep("DRIVER=brcmfmac", join(wifi_interface, "device", "uevent")):
self._wlan_class_path = wlan_interface self._wlan_class_path = wifi_interface
# If no build in WLAN on the CM, detect third party WLAN on RevPi Flat # If no build in Wi-Fi on the CM, detect third party Wi-Fi on RevPi Flat
if not self._wlan_class_path and grep("revpi-flat", "/proc/device-tree/compatible"): if not self._wlan_class_path and grep("revpi-flat", "/proc/device-tree/compatible"):
lst_wlan_interfaces = glob("/sys/class/ieee80211/*") lst_wifi_interfaces = glob("/sys/class/ieee80211/*")
for wlan_interface in lst_wlan_interfaces: for wifi_interface in lst_wifi_interfaces:
if grep("DRIVER=mwifiex_sdio", join(wlan_interface, "device", "uevent")): if grep("DRIVER=mwifiex_sdio", join(wifi_interface, "device", "uevent")):
self._wlan_class_path = wlan_interface self._wlan_class_path = wifi_interface
# Detect ConBridge # Detect ConBridge
could_have_con_bridge = self._cm_type in (ComputeModuleTypes.CM3, ComputeModuleTypes.CM4S) could_have_con_bridge = self._cm_type in (ComputeModuleTypes.CM3, ComputeModuleTypes.CM4S)
@@ -102,7 +102,7 @@ class RevPiConfig:
self._revpi_with_con_bridge = len(lst_grep) > 0 self._revpi_with_con_bridge = len(lst_grep) > 0
@property @property
def class_path_wlan(self) -> str: def class_path_wifi(self) -> str:
return self._wlan_class_path return self._wlan_class_path
@property @property
@@ -114,7 +114,7 @@ class RevPiConfig:
return self._revpi_with_con_bridge return self._revpi_with_con_bridge
@property @property
def with_wlan(self) -> bool: def with_wifi(self) -> bool:
return bool(self._wlan_class_path) return bool(self._wlan_class_path)
@@ -224,13 +224,11 @@ def configure_bluetooth(action: ConfigActions):
if action is ConfigActions.ENABLE: if action is ConfigActions.ENABLE:
if bt_rfkill_index is not None: if bt_rfkill_index is not None:
with open(f"/sys/class/rfkill/rfkill{bt_rfkill_index}/soft", "w") as f: subprocess.call(["rfkill", "unblock", str(bt_rfkill_index)])
f.write("0")
elif action is ConfigActions.DISABLE: elif action is ConfigActions.DISABLE:
if bt_rfkill_index is not None: if bt_rfkill_index is not None:
with open(f"/sys/class/rfkill/rfkill{bt_rfkill_index}/soft", "w") as f: subprocess.call(["rfkill", "block", str(bt_rfkill_index)])
f.write("1")
elif action is ConfigActions.STATUS: elif action is ConfigActions.STATUS:
if bt_rfkill_index is None: if bt_rfkill_index is None:
@@ -295,21 +293,21 @@ def configure_dphys_swapfile(action: ConfigActions):
def configure_external_antenna(action: ConfigActions): def configure_external_antenna(action: ConfigActions):
revpi = RevPiConfig() revpi = RevPiConfig()
if action is ConfigActions.AVAILABLE: if action is ConfigActions.AVAILABLE:
return revpi.with_wlan return revpi.with_wifi
config_txt = ConfigTxt() config_txt = ConfigTxt()
if action is ConfigActions.ENABLE and revpi.with_wlan: if action is ConfigActions.ENABLE and revpi.with_wifi:
config_txt.clear_dtparams(["ant1", "ant2"]) config_txt.clear_dtparams(["ant1", "ant2"])
config_txt.add_name_value("dtparam", "ant2") config_txt.add_name_value("dtparam", "ant2")
config_txt.save_config() config_txt.save_config()
elif action is ConfigActions.DISABLE and revpi.with_wlan: elif action is ConfigActions.DISABLE and revpi.with_wifi:
config_txt.clear_dtparams(["ant1", "ant2"]) config_txt.clear_dtparams(["ant1", "ant2"])
config_txt.save_config() config_txt.save_config()
elif action is ConfigActions.STATUS: elif action is ConfigActions.STATUS:
return revpi.with_wlan and "ant2" in config_txt.get_values("dtparam") return revpi.with_wifi and "ant2" in config_txt.get_values("dtparam")
else: else:
raise ValueError(f"action {action} not supported") raise ValueError(f"action {action} not supported")
@@ -339,30 +337,28 @@ def configure_gui(action: ConfigActions):
raise ValueError(f"action {action} not supported") raise ValueError(f"action {action} not supported")
def configure_wlan(action: ConfigActions): def configure_wifi(action: ConfigActions):
revpi = RevPiConfig() revpi = RevPiConfig()
if action is ConfigActions.ENABLE: if action is ConfigActions.ENABLE:
if revpi.with_wlan: if revpi.with_wifi:
wlan_rfkill_index = get_rfkill_index(revpi.class_path_wlan) wifi_rfkill_index = get_rfkill_index(revpi.class_path_wifi)
with open(f"/sys/class/rfkill/rfkill{wlan_rfkill_index}/soft", "w") as f: subprocess.call(["rfkill", "unblock", str(wifi_rfkill_index)])
f.write("0")
elif action is ConfigActions.DISABLE: elif action is ConfigActions.DISABLE:
if revpi.with_wlan: if revpi.with_wifi:
wlan_rfkill_index = get_rfkill_index(revpi.class_path_wlan) wifi_rfkill_index = get_rfkill_index(revpi.class_path_wifi)
with open(f"/sys/class/rfkill/rfkill{wlan_rfkill_index}/soft", "w") as f: subprocess.call(["rfkill", "block", str(wifi_rfkill_index)])
f.write("1")
elif action is ConfigActions.AVAILABLE: elif action is ConfigActions.AVAILABLE:
return revpi.with_wlan return revpi.with_wifi
elif action is ConfigActions.STATUS: elif action is ConfigActions.STATUS:
if not revpi.with_wlan: if not revpi.with_wifi:
return False return False
wlan_rfkill_index = get_rfkill_index(revpi.class_path_wlan) wifi_rfkill_index = get_rfkill_index(revpi.class_path_wifi)
with open(f"/sys/class/rfkill/rfkill{wlan_rfkill_index}/soft", "r") as f: with open(f"/sys/class/rfkill/rfkill{wifi_rfkill_index}/soft", "r") as f:
buffer = f.read().strip() buffer = f.read().strip()
return buffer == "0" return buffer == "0"
@@ -424,10 +420,10 @@ if __name__ == "__main__":
print("Model:", rc.model) print("Model:", rc.model)
print("Serial: ", rc.serial) print("Serial: ", rc.serial)
print("CM Type: ", rc.cm_type.name) print("CM Type: ", rc.cm_type.name)
print("With WLAN: ", rc.with_wlan) print("With wifi: ", rc.with_wifi)
if rc.with_wlan: if rc.with_wifi:
print(" class path: ", rc.class_path_wlan) print(" class path: ", rc.class_path_wifi)
print(" rfkill index: ", get_rfkill_index(rc.class_path_wlan)) print(" rfkill index: ", get_rfkill_index(rc.class_path_wifi))
print("With con-bridge:", rc.with_con_bridge) print("With con-bridge:", rc.with_con_bridge)
config_txt = ConfigTxt() config_txt = ConfigTxt()