diff --git a/setup.py b/setup.py index c9359a1a..1405094d 100644 --- a/setup.py +++ b/setup.py @@ -33,7 +33,7 @@ "backoff", "click", "dacite", - "fastapi-azure-auth==4.0.0", + "fastapi-azure-auth", "fastapi", "injector", "opencensus-ext-logging", @@ -41,7 +41,8 @@ "opencensus-ext-azure", "numpy", "paho-mqtt", - "pydantic==1.10.11", + "pydantic", + "pydantic_settings", "PyJWT", "python-dotenv", "PyYAML", diff --git a/src/isar/apis/models/models.py b/src/isar/apis/models/models.py index baae7692..93d211f7 100644 --- a/src/isar/apis/models/models.py +++ b/src/isar/apis/models/models.py @@ -11,7 +11,7 @@ class StepResponse(BaseModel): class TaskResponse(BaseModel): id: str - tag_id: Optional[str] + tag_id: Optional[str] = None steps: List[StepResponse] diff --git a/src/isar/apis/models/start_mission_definition.py b/src/isar/apis/models/start_mission_definition.py index fda348e0..b5dcd1ef 100644 --- a/src/isar/apis/models/start_mission_definition.py +++ b/src/isar/apis/models/start_mission_definition.py @@ -32,22 +32,22 @@ class InspectionTypes(str, Enum): class StartMissionInspectionDefinition(BaseModel): type: InspectionTypes = Field(default=InspectionTypes.image) inspection_target: InputPosition - duration: Optional[float] - metadata: Optional[dict] - id: Optional[str] + duration: Optional[float] = None + metadata: Optional[dict] = None + id: Optional[str] = None class StartMissionTaskDefinition(BaseModel): pose: InputPose inspections: List[StartMissionInspectionDefinition] - tag: Optional[str] - id: Optional[str] + tag: Optional[str] = None + id: Optional[str] = None class StartMissionDefinition(BaseModel): tasks: List[StartMissionTaskDefinition] - id: Optional[str] - name: Optional[str] + id: Optional[str] = None + name: Optional[str] = None def to_isar_mission(mission_definition: StartMissionDefinition) -> Mission: diff --git a/src/isar/config/settings.py b/src/isar/config/settings.py index 2a73d5fd..fa85ab0b 100644 --- a/src/isar/config/settings.py +++ b/src/isar/config/settings.py @@ -1,9 +1,10 @@ import importlib.resources as pkg_resources import os -from typing import List, Optional +from typing import Any, List, Optional from dotenv import load_dotenv -from pydantic import BaseSettings, Field, validator +from pydantic import Field, ValidationInfo, field_validator +from pydantic_settings import BaseSettings, SettingsConfigDict from isar.config import predefined_missions from robot_interface.models.robots.robot_model import RobotModel @@ -11,6 +12,14 @@ class Settings(BaseSettings): + def __init__(self) -> None: + try: + with pkg_resources.path(f"isar.config", "settings.env") as path: + env_file_path = path + except ModuleNotFoundError: + env_file_path = None + super().__init__(_env_file=env_file_path) + # Determines which robot package ISAR will attempt to import # Name must match with an installed python package in the local environment ROBOT_PACKAGE: str = Field(default="isar_robot") @@ -32,7 +41,7 @@ class Settings(BaseSettings): FSM_SLEEP_TIME: float = Field(default=0.1) # Location of JSON files containing predefined missions for the Local Planner to use - path = os.path.dirname(predefined_missions.__file__) + path: str = os.path.dirname(predefined_missions.__file__) PREDEFINED_MISSIONS_FOLDER: str = Field(default=path + "/") # Name of default map transformation @@ -217,14 +226,18 @@ class Settings(BaseSettings): DATA_CLASSIFICATION: str = Field(default="internal") # List of MQTT Topics - TOPIC_ISAR_STATE: str = Field(default="state") - TOPIC_ISAR_MISSION: str = Field(default="mission") - TOPIC_ISAR_TASK: str = Field(default="task") - TOPIC_ISAR_STEP: str = Field(default="step") - TOPIC_ISAR_INSPECTION_RESULT = Field(default="inspection_result") - TOPIC_ISAR_ROBOT_STATUS: str = Field(default="robot_status") - TOPIC_ISAR_ROBOT_INFO: str = Field(default="robot_info") - TOPIC_ISAR_ROBOT_HEARTBEAT: str = Field(default="robot_heartbeat") + TOPIC_ISAR_STATE: str = Field(default="state", validate_default=True) + TOPIC_ISAR_MISSION: str = Field(default="mission", validate_default=True) + TOPIC_ISAR_TASK: str = Field(default="task", validate_default=True) + TOPIC_ISAR_STEP: str = Field(default="step", validate_default=True) + TOPIC_ISAR_INSPECTION_RESULT: str = Field( + default="inspection_result", validate_default=True + ) + TOPIC_ISAR_ROBOT_STATUS: str = Field(default="robot_status", validate_default=True) + TOPIC_ISAR_ROBOT_INFO: str = Field(default="robot_info", validate_default=True) + TOPIC_ISAR_ROBOT_HEARTBEAT: str = Field( + default="robot_heartbeat", validate_default=True + ) # Logging @@ -252,21 +265,22 @@ class Settings(BaseSettings): REQUIRED_ROLE: str = Field(default="Mission.Control") - @validator("LOG_LEVELS", pre=True, always=True) - def set_log_levels(cls, v, values) -> dict: + @field_validator("LOG_LEVELS") + @classmethod + def set_log_levels(cls, v: Any, info: ValidationInfo) -> dict: return { - "api": values["API_LOG_LEVEL"], - "main": values["MAIN_LOG_LEVEL"], - "mqtt": values["MQTT_LOG_LEVEL"], - "state_machine": values["STATE_MACHINE_LOG_LEVEL"], - "uploader": values["UPLOADER_LOG_LEVEL"], - "console": values["CONSOLE_LOG_LEVEL"], - "urllib3": values["URLLIB3_LOG_LEVEL"], - "uvicorn": values["UVICORN_LOG_LEVEL"], - "azure": values["AZURE_LOG_LEVEL"], + "api": info.data["API_LOG_LEVEL"], + "main": info.data["MAIN_LOG_LEVEL"], + "mqtt": info.data["MQTT_LOG_LEVEL"], + "state_machine": info.data["STATE_MACHINE_LOG_LEVEL"], + "uploader": info.data["UPLOADER_LOG_LEVEL"], + "console": info.data["CONSOLE_LOG_LEVEL"], + "urllib3": info.data["URLLIB3_LOG_LEVEL"], + "uvicorn": info.data["UVICORN_LOG_LEVEL"], + "azure": info.data["AZURE_LOG_LEVEL"], } - @validator( + @field_validator( "TOPIC_ISAR_STATE", "TOPIC_ISAR_MISSION", "TOPIC_ISAR_TASK", @@ -275,20 +289,16 @@ def set_log_levels(cls, v, values) -> dict: "TOPIC_ISAR_ROBOT_INFO", "TOPIC_ISAR_ROBOT_HEARTBEAT", "TOPIC_ISAR_INSPECTION_RESULT", - pre=True, - always=True, ) - def prefix_isar_topics(cls, v, values): - return f"isar/{values['ISAR_ID']}/{v}" - - class Config: - with pkg_resources.path("isar.config", "settings.env") as path: - package_path = path - - env_prefix = "ISAR_" - env_file = package_path - env_file_encoding = "utf-8" - case_sensitive = True + @classmethod + def prefix_isar_topics(cls, v: Any, info: ValidationInfo): + return f"isar/{info.data['ISAR_ID']}/{v}" + + model_config = SettingsConfigDict( + env_prefix="ISAR_", + env_file_encoding="utf-8", + case_sensitive=True, + ) load_dotenv() @@ -319,10 +329,10 @@ def __init__(self) -> None: # Note that if the robot does not support moving an arm this will be None and # the functionality will be unavailable VALID_ARM_POSES: Optional[List[str]] = Field(default=None) - - class Config: - env_file_encoding = "utf-8" - case_sensitive = True + model_config = SettingsConfigDict( + env_file_encoding="utf-8", + case_sensitive=True, + ) robot_settings = RobotSettings()