From bbec2807ccfeae15e8367764586d952953dcc12e Mon Sep 17 00:00:00 2001 From: Joao Mario Lago Date: Thu, 3 Oct 2024 04:36:52 -0300 Subject: [PATCH] core: autopilot-manager: Add flight_controller module * Move flight controller / flight controller detector to its own module as preparation for platform spliting --- .../ardupilot_manager/api/v1/routers/index.py | 3 +- .../ardupilot_manager/autopilot_manager.py | 24 +++--- .../firmware/FirmwareDownload.py | 3 +- .../firmware/FirmwareInstall.py | 3 +- .../firmware/FirmwareManagement.py | 11 +-- .../firmware/test_FirmwareDownload.py | 3 +- .../firmware/test_FirmwareInstall.py | 3 +- .../flight_controller/__init__.py | 8 ++ .../flight_controller/detector/__init__.py | 3 + .../detector}/board_identification.py | 2 +- .../detector/detector.py} | 16 ++-- .../detector/linux}/__init__.py | 0 .../detector}/linux/argonot.py | 4 +- .../detector}/linux/detector.py | 6 +- .../detector}/linux/linux_boards.py | 3 +- .../detector}/linux/navigator.py | 5 +- .../flight_controller/flight_controller.py | 76 ++++++++++++++++++ .../linux/__init__.py | 0 core/services/ardupilot_manager/main.py | 4 +- core/services/ardupilot_manager/typedefs.py | 78 +------------------ 20 files changed, 134 insertions(+), 121 deletions(-) create mode 100644 core/services/ardupilot_manager/flight_controller/__init__.py create mode 100644 core/services/ardupilot_manager/flight_controller/detector/__init__.py rename core/services/ardupilot_manager/{flight_controller_detector => flight_controller/detector}/board_identification.py (97%) rename core/services/ardupilot_manager/{flight_controller_detector/Detector.py => flight_controller/detector/detector.py} (85%) rename core/services/ardupilot_manager/{flight_controller_detector => flight_controller/detector/linux}/__init__.py (100%) rename core/services/ardupilot_manager/{flight_controller_detector => flight_controller/detector}/linux/argonot.py (63%) rename core/services/ardupilot_manager/{flight_controller_detector => flight_controller/detector}/linux/detector.py (83%) rename core/services/ardupilot_manager/{flight_controller_detector => flight_controller/detector}/linux/linux_boards.py (91%) rename core/services/ardupilot_manager/{flight_controller_detector => flight_controller/detector}/linux/navigator.py (94%) create mode 100644 core/services/ardupilot_manager/flight_controller/flight_controller.py delete mode 100644 core/services/ardupilot_manager/flight_controller_detector/linux/__init__.py diff --git a/core/services/ardupilot_manager/api/v1/routers/index.py b/core/services/ardupilot_manager/api/v1/routers/index.py index 23bbf55a99..19a6f80c40 100644 --- a/core/services/ardupilot_manager/api/v1/routers/index.py +++ b/core/services/ardupilot_manager/api/v1/routers/index.py @@ -18,7 +18,8 @@ from autopilot_manager import AutoPilotManager from exceptions import InvalidFirmwareFile -from typedefs import Firmware, FlightController, Parameters, Serial, SITLFrame, Vehicle +from flight_controller import FlightController +from typedefs import Firmware, Parameters, Serial, SITLFrame, Vehicle index_router_v1 = APIRouter( tags=["index_v1"], diff --git a/core/services/ardupilot_manager/autopilot_manager.py b/core/services/ardupilot_manager/autopilot_manager.py index 3bfeb691d4..847ede9cc1 100644 --- a/core/services/ardupilot_manager/autopilot_manager.py +++ b/core/services/ardupilot_manager/autopilot_manager.py @@ -18,23 +18,19 @@ NoPreferredBoardSet, ) from firmware.FirmwareManagement import FirmwareManager -from flight_controller_detector.Detector import Detector as BoardDetector -from flight_controller_detector.linux.linux_boards import LinuxFlightController -from mavlink_proxy.Endpoint import Endpoint, EndpointType -from mavlink_proxy.exceptions import EndpointAlreadyExists -from mavlink_proxy.Manager import Manager as MavlinkManager -from settings import Settings -from typedefs import ( - Firmware, +from flight_controller import ( FlightController, FlightControllerFlags, - Parameters, Platform, PlatformType, - Serial, - SITLFrame, - Vehicle, ) +from flight_controller.detector import FlightControllerDetector +from flight_controller.detector.linux.linux_boards import LinuxFlightController +from mavlink_proxy.Endpoint import Endpoint, EndpointType +from mavlink_proxy.exceptions import EndpointAlreadyExists +from mavlink_proxy.Manager import Manager as MavlinkManager +from settings import Settings +from typedefs import Firmware, Parameters, Serial, SITLFrame, Vehicle class AutoPilotManager(metaclass=Singleton): @@ -294,7 +290,7 @@ def get_available_routers(self) -> List[str]: return [router.name() for router in self.mavlink_manager.available_interfaces()] async def start_sitl(self) -> None: - self._current_board = BoardDetector.detect_sitl() + self._current_board = FlightControllerDetector.detect_sitl() if not self.firmware_manager.is_firmware_installed(self._current_board): self.firmware_manager.install_firmware_from_params(Vehicle.Sub, self._current_board) frame = self.load_sitl_frame() @@ -404,7 +400,7 @@ async def start_mavlink_manager(self, device: Endpoint) -> None: @staticmethod async def available_boards(include_bootloaders: bool = False) -> List[FlightController]: - all_boards = await BoardDetector.detect(True) + all_boards = await FlightControllerDetector.detect(True) if include_bootloaders: return all_boards return [board for board in all_boards if FlightControllerFlags.is_bootloader not in board.flags] diff --git a/core/services/ardupilot_manager/firmware/FirmwareDownload.py b/core/services/ardupilot_manager/firmware/FirmwareDownload.py index 1c6591b1f5..565a17ee9f 100644 --- a/core/services/ardupilot_manager/firmware/FirmwareDownload.py +++ b/core/services/ardupilot_manager/firmware/FirmwareDownload.py @@ -21,7 +21,8 @@ NoCandidate, NoVersionAvailable, ) -from typedefs import FirmwareFormat, Platform, PlatformType, Vehicle +from flight_controller import Platform, PlatformType +from typedefs import FirmwareFormat, Vehicle # TODO: This should be not necessary # Disable SSL verification diff --git a/core/services/ardupilot_manager/firmware/FirmwareInstall.py b/core/services/ardupilot_manager/firmware/FirmwareInstall.py index b14e7b4738..a170e0a2bd 100644 --- a/core/services/ardupilot_manager/firmware/FirmwareInstall.py +++ b/core/services/ardupilot_manager/firmware/FirmwareInstall.py @@ -12,7 +12,8 @@ from exceptions import FirmwareInstallFail, InvalidFirmwareFile, UnsupportedPlatform from firmware.FirmwareDownload import FirmwareDownloader from firmware.FirmwareUpload import FirmwareUploader -from typedefs import FirmwareFormat, FlightController, Platform, PlatformType +from flight_controller import FlightController, Platform, PlatformType +from typedefs import FirmwareFormat def get_board_id(platform: Platform) -> int: diff --git a/core/services/ardupilot_manager/firmware/FirmwareManagement.py b/core/services/ardupilot_manager/firmware/FirmwareManagement.py index 1d063a1a07..e0d808bf62 100644 --- a/core/services/ardupilot_manager/firmware/FirmwareManagement.py +++ b/core/services/ardupilot_manager/firmware/FirmwareManagement.py @@ -15,15 +15,8 @@ ) from firmware.FirmwareDownload import FirmwareDownloader from firmware.FirmwareInstall import FirmwareInstaller -from typedefs import ( - Firmware, - FirmwareFormat, - FlightController, - Parameters, - Platform, - PlatformType, - Vehicle, -) +from flight_controller import FlightController, Platform, PlatformType +from typedefs import Firmware, FirmwareFormat, Parameters, Vehicle class FirmwareManager: diff --git a/core/services/ardupilot_manager/firmware/test_FirmwareDownload.py b/core/services/ardupilot_manager/firmware/test_FirmwareDownload.py index 1246b7a417..b2ab672ff1 100644 --- a/core/services/ardupilot_manager/firmware/test_FirmwareDownload.py +++ b/core/services/ardupilot_manager/firmware/test_FirmwareDownload.py @@ -4,7 +4,8 @@ import pytest from firmware.FirmwareDownload import FirmwareDownloader -from typedefs import Platform, Vehicle +from flight_controller import Platform +from typedefs import Vehicle def test_static() -> None: diff --git a/core/services/ardupilot_manager/firmware/test_FirmwareInstall.py b/core/services/ardupilot_manager/firmware/test_FirmwareInstall.py index 559e5aa30c..f279babff3 100644 --- a/core/services/ardupilot_manager/firmware/test_FirmwareInstall.py +++ b/core/services/ardupilot_manager/firmware/test_FirmwareInstall.py @@ -6,7 +6,8 @@ from exceptions import InvalidFirmwareFile from firmware.FirmwareDownload import FirmwareDownloader from firmware.FirmwareInstall import FirmwareInstaller -from typedefs import FlightController, Platform, Vehicle +from flight_controller import FlightController, Platform +from typedefs import Vehicle def test_firmware_validation() -> None: diff --git a/core/services/ardupilot_manager/flight_controller/__init__.py b/core/services/ardupilot_manager/flight_controller/__init__.py new file mode 100644 index 0000000000..fea7544df8 --- /dev/null +++ b/core/services/ardupilot_manager/flight_controller/__init__.py @@ -0,0 +1,8 @@ +from flight_controller.flight_controller import ( + FlightController, + FlightControllerFlags, + Platform, + PlatformType, +) + +__all__ = ["FlightController", "FlightControllerFlags", "Platform", "PlatformType"] diff --git a/core/services/ardupilot_manager/flight_controller/detector/__init__.py b/core/services/ardupilot_manager/flight_controller/detector/__init__.py new file mode 100644 index 0000000000..5a19a91e2a --- /dev/null +++ b/core/services/ardupilot_manager/flight_controller/detector/__init__.py @@ -0,0 +1,3 @@ +from flight_controller.detector.detector import FlightControllerDetector + +__all__ = ["FlightControllerDetector"] diff --git a/core/services/ardupilot_manager/flight_controller_detector/board_identification.py b/core/services/ardupilot_manager/flight_controller/detector/board_identification.py similarity index 97% rename from core/services/ardupilot_manager/flight_controller_detector/board_identification.py rename to core/services/ardupilot_manager/flight_controller/detector/board_identification.py index d536497b0e..0e393264d6 100644 --- a/core/services/ardupilot_manager/flight_controller_detector/board_identification.py +++ b/core/services/ardupilot_manager/flight_controller/detector/board_identification.py @@ -3,7 +3,7 @@ from pydantic import BaseModel -from typedefs import Platform +from flight_controller import Platform class SerialAttr(str, Enum): diff --git a/core/services/ardupilot_manager/flight_controller_detector/Detector.py b/core/services/ardupilot_manager/flight_controller/detector/detector.py similarity index 85% rename from core/services/ardupilot_manager/flight_controller_detector/Detector.py rename to core/services/ardupilot_manager/flight_controller/detector/detector.py index a1753a14b8..443f20f146 100644 --- a/core/services/ardupilot_manager/flight_controller_detector/Detector.py +++ b/core/services/ardupilot_manager/flight_controller/detector/detector.py @@ -4,12 +4,12 @@ from commonwealth.utils.general import is_running_as_root from serial.tools.list_ports_linux import SysFS, comports -from flight_controller_detector.board_identification import identifiers -from flight_controller_detector.linux.detector import LinuxFlightControllerDetector -from typedefs import FlightController, FlightControllerFlags, Platform +from flight_controller import FlightController, FlightControllerFlags, Platform +from flight_controller.detector.board_identification import identifiers +from flight_controller.detector.linux.detector import LinuxFlightControllerDetector -class Detector: +class FlightControllerDetector: @classmethod async def detect_linux_board(cls) -> Optional[FlightController]: for _i in range(5): @@ -59,16 +59,16 @@ def detect_serial_flight_controllers() -> List[FlightController]: FlightController( name=port.product or port.name, manufacturer=port.manufacturer, - platform=Detector.detect_serial_platform(port) + platform=FlightControllerDetector.detect_serial_platform(port) or Platform(), # this is just to make CI happy. check line 82 path=port.device, ) for port in unique_serial_devices - if Detector.detect_serial_platform(port) is not None + if FlightControllerDetector.detect_serial_platform(port) is not None ] for port in unique_serial_devices: for board in boards: - if board.path == port.device and Detector.is_serial_bootloader(port): + if board.path == port.device and FlightControllerDetector.is_serial_bootloader(port): board.flags.append(FlightControllerFlags.is_bootloader) return boards @@ -97,6 +97,6 @@ async def detect(cls, include_sitl: bool = True) -> List[FlightController]: available.extend(cls().detect_serial_flight_controllers()) if include_sitl: - available.append(Detector.detect_sitl()) + available.append(FlightControllerDetector.detect_sitl()) return available diff --git a/core/services/ardupilot_manager/flight_controller_detector/__init__.py b/core/services/ardupilot_manager/flight_controller/detector/linux/__init__.py similarity index 100% rename from core/services/ardupilot_manager/flight_controller_detector/__init__.py rename to core/services/ardupilot_manager/flight_controller/detector/linux/__init__.py diff --git a/core/services/ardupilot_manager/flight_controller_detector/linux/argonot.py b/core/services/ardupilot_manager/flight_controller/detector/linux/argonot.py similarity index 63% rename from core/services/ardupilot_manager/flight_controller_detector/linux/argonot.py rename to core/services/ardupilot_manager/flight_controller/detector/linux/argonot.py index e486d39429..c7c23d79b8 100644 --- a/core/services/ardupilot_manager/flight_controller_detector/linux/argonot.py +++ b/core/services/ardupilot_manager/flight_controller/detector/linux/argonot.py @@ -1,5 +1,5 @@ -from flight_controller_detector.linux.navigator import NavigatorPi4 -from typedefs import Platform +from flight_controller import Platform +from flight_controller.detector.linux.navigator import NavigatorPi4 class Argonot(NavigatorPi4): diff --git a/core/services/ardupilot_manager/flight_controller_detector/linux/detector.py b/core/services/ardupilot_manager/flight_controller/detector/linux/detector.py similarity index 83% rename from core/services/ardupilot_manager/flight_controller_detector/linux/detector.py rename to core/services/ardupilot_manager/flight_controller/detector/linux/detector.py index 3b1dd5c258..6020dea146 100644 --- a/core/services/ardupilot_manager/flight_controller_detector/linux/detector.py +++ b/core/services/ardupilot_manager/flight_controller/detector/linux/detector.py @@ -3,9 +3,9 @@ from loguru import logger -from flight_controller_detector.linux.argonot import Argonot -from flight_controller_detector.linux.linux_boards import LinuxFlightController -from flight_controller_detector.linux.navigator import NavigatorPi4, NavigatorPi5 +from flight_controller.detector.linux.argonot import Argonot +from flight_controller.detector.linux.linux_boards import LinuxFlightController +from flight_controller.detector.linux.navigator import NavigatorPi4, NavigatorPi5 class LinuxFlightControllerDetector: diff --git a/core/services/ardupilot_manager/flight_controller_detector/linux/linux_boards.py b/core/services/ardupilot_manager/flight_controller/detector/linux/linux_boards.py similarity index 91% rename from core/services/ardupilot_manager/flight_controller_detector/linux/linux_boards.py rename to core/services/ardupilot_manager/flight_controller/detector/linux/linux_boards.py index 97f21ab8c5..652b49adbc 100644 --- a/core/services/ardupilot_manager/flight_controller_detector/linux/linux_boards.py +++ b/core/services/ardupilot_manager/flight_controller/detector/linux/linux_boards.py @@ -2,7 +2,8 @@ from smbus2 import SMBus -from typedefs import FlightController, PlatformType, Serial +from flight_controller import FlightController, PlatformType +from typedefs import Serial class LinuxFlightController(FlightController): diff --git a/core/services/ardupilot_manager/flight_controller_detector/linux/navigator.py b/core/services/ardupilot_manager/flight_controller/detector/linux/navigator.py similarity index 94% rename from core/services/ardupilot_manager/flight_controller_detector/linux/navigator.py rename to core/services/ardupilot_manager/flight_controller/detector/linux/navigator.py index 0ebf9da4e4..2cd3fc9a1e 100644 --- a/core/services/ardupilot_manager/flight_controller_detector/linux/navigator.py +++ b/core/services/ardupilot_manager/flight_controller/detector/linux/navigator.py @@ -2,8 +2,9 @@ from commonwealth.utils.commands import load_file -from flight_controller_detector.linux.linux_boards import LinuxFlightController -from typedefs import Platform, Serial +from flight_controller import Platform +from flight_controller.detector.linux.linux_boards import LinuxFlightController +from typedefs import Serial class Navigator(LinuxFlightController): diff --git a/core/services/ardupilot_manager/flight_controller/flight_controller.py b/core/services/ardupilot_manager/flight_controller/flight_controller.py new file mode 100644 index 0000000000..2fca844ffe --- /dev/null +++ b/core/services/ardupilot_manager/flight_controller/flight_controller.py @@ -0,0 +1,76 @@ +from enum import Enum, auto +from platform import machine +from typing import List, Optional + +from pydantic import BaseModel + + +def get_sitl_platform_name(machine_arch: str) -> str: + """Get SITL platform name based on machine architecture.""" + + if "arm" not in machine_arch.lower() and "aarch" not in machine_arch.lower(): + return "SITL_x86_64_linux_gnu" + return "SITL_arm_linux_gnueabihf" + + +# TODO: This class can be deprecated once we move to Python 3.11, which introduces the equivalent StrEnum +class LowerStringEnum(str, Enum): + def __str__(self) -> str: + return self.name.lower() + + +class PlatformType(LowerStringEnum): + Serial = auto() + Linux = auto() + SITL = auto() + Unknown = auto() + + +class Platform(str, Enum): + """Valid Ardupilot platform types. + The Enum values are 1:1 representations of the platforms available on the ArduPilot manifest.""" + + Pixhawk1 = "Pixhawk1" + Pixhawk4 = "Pixhawk4" + Pixhawk6X = "Pixhawk6X" + Pixhawk6C = "Pixhawk6C" + CubeOrange = "CubeOrange" + GenericSerial = "GenericSerial" + Navigator = "navigator" + Argonot = "argonot" + SITL = get_sitl_platform_name(machine()) + + @property + def type(self) -> PlatformType: + platform_types = { + Platform.Pixhawk1: PlatformType.Serial, + Platform.Pixhawk4: PlatformType.Serial, + Platform.Pixhawk6X: PlatformType.Serial, + Platform.Pixhawk6C: PlatformType.Serial, + Platform.CubeOrange: PlatformType.Serial, + Platform.GenericSerial: PlatformType.Serial, + Platform.Navigator: PlatformType.Linux, + Platform.Argonot: PlatformType.Linux, + Platform.SITL: PlatformType.SITL, + } + return platform_types.get(self, PlatformType.Unknown) + + +class FlightControllerFlags(str, Enum): + """Flags for the Flight-controller class.""" + + is_bootloader = "is_bootloader" + + +class FlightController(BaseModel): + """Flight-controller board.""" + + name: str + manufacturer: Optional[str] + platform: Platform + path: Optional[str] + flags: List[FlightControllerFlags] = [] + + @property + def type(self) -> PlatformType: + return self.platform.type diff --git a/core/services/ardupilot_manager/flight_controller_detector/linux/__init__.py b/core/services/ardupilot_manager/flight_controller_detector/linux/__init__.py deleted file mode 100644 index e69de29bb2..0000000000 diff --git a/core/services/ardupilot_manager/main.py b/core/services/ardupilot_manager/main.py index 92b693d1a0..3c6df31ed0 100755 --- a/core/services/ardupilot_manager/main.py +++ b/core/services/ardupilot_manager/main.py @@ -9,7 +9,7 @@ from args import CommandLineArgs from autopilot_manager import AutoPilotManager -from flight_controller_detector.Detector import Detector as BoardDetector +from flight_controller.detector.detector import FlightControllerDetector from settings import SERVICE_NAME logging.basicConfig(handlers=[InterceptHandler()], level=0) @@ -36,7 +36,7 @@ server = Server(config) if args.sitl: - autopilot.set_preferred_board(BoardDetector.detect_sitl()) + autopilot.set_preferred_board(FlightControllerDetector.detect_sitl()) try: loop.run_until_complete(autopilot.start_ardupilot()) except Exception as start_error: diff --git a/core/services/ardupilot_manager/typedefs.py b/core/services/ardupilot_manager/typedefs.py index 9f14f92f6e..ecab042b9d 100644 --- a/core/services/ardupilot_manager/typedefs.py +++ b/core/services/ardupilot_manager/typedefs.py @@ -1,12 +1,13 @@ import ipaddress import re -from enum import Enum, auto +from enum import Enum from pathlib import Path -from platform import machine -from typing import Any, Dict, List, Optional +from typing import Any, Dict, List from pydantic import BaseModel, validator +from flight_controller import FlightController + class SITLFrame(str, Enum): """Valid SITL frame types""" @@ -61,14 +62,6 @@ class SITLFrame(str, Enum): UNDEFINED = " undefined" -def get_sitl_platform_name(machine_arch: str) -> str: - """Get SITL platform name based on machine architecture.""" - - if "arm" not in machine_arch.lower() and "aarch" not in machine_arch.lower(): - return "SITL_x86_64_linux_gnu" - return "SITL_arm_linux_gnueabihf" - - class Firmware(BaseModel): """Simplified representation of a firmware, as available on Ardupilot's manifest.""" @@ -86,73 +79,10 @@ class Vehicle(str, Enum): Copter = "Copter" -# TODO: This class can be deprecated once we move to Python 3.11, which introduces the equivalent StrEnum -class LowerStringEnum(str, Enum): - def __str__(self) -> str: - return self.name.lower() - - -class PlatformType(LowerStringEnum): - Serial = auto() - Linux = auto() - SITL = auto() - Unknown = auto() - - -class Platform(str, Enum): - """Valid Ardupilot platform types. - The Enum values are 1:1 representations of the platforms available on the ArduPilot manifest.""" - - Pixhawk1 = "Pixhawk1" - Pixhawk4 = "Pixhawk4" - Pixhawk6X = "Pixhawk6X" - Pixhawk6C = "Pixhawk6C" - CubeOrange = "CubeOrange" - GenericSerial = "GenericSerial" - Navigator = "navigator" - Argonot = "argonot" - SITL = get_sitl_platform_name(machine()) - - @property - def type(self) -> PlatformType: - platform_types = { - Platform.Pixhawk1: PlatformType.Serial, - Platform.Pixhawk4: PlatformType.Serial, - Platform.Pixhawk6X: PlatformType.Serial, - Platform.Pixhawk6C: PlatformType.Serial, - Platform.CubeOrange: PlatformType.Serial, - Platform.GenericSerial: PlatformType.Serial, - Platform.Navigator: PlatformType.Linux, - Platform.Argonot: PlatformType.Linux, - Platform.SITL: PlatformType.SITL, - } - return platform_types.get(self, PlatformType.Unknown) - - -class FlightControllerFlags(str, Enum): - """Flags for the Flight-controller class.""" - - is_bootloader = "is_bootloader" - - class Parameters(BaseModel): params: Dict[str, float] -class FlightController(BaseModel): - """Flight-controller board.""" - - name: str - manufacturer: Optional[str] - platform: Platform - path: Optional[str] - flags: List[FlightControllerFlags] = [] - - @property - def type(self) -> PlatformType: - return self.platform.type - - class AvailableBoards(BaseModel): regular: List[FlightController] bootloaders: List[FlightController]