import logging import time import typing from typing import Any, List, Optional import httpx from fastapi import APIRouter, BackgroundTasks, Depends, Query from fastapi.responses import HTMLResponse, JSONResponse from sqlalchemy.orm import Session from clan_cli.config import ap_url, c1_url, c2_url, dlg_url, msg_type_to_label from ...errors import ClanError from .. import sql_crud, sql_db, sql_models from ..schemas import ( Entity, EntityCreate, Eventmessage, EventmessageCreate, Resolution, Role, Service, ServiceCreate, ServiceUsageCreate, ) from ..tags import Tags router = APIRouter() log = logging.getLogger(__name__) ######################### # # # Service # # # ######################### @router.post("/api/v1/service", response_model=Service, tags=[Tags.services]) def create_service( service: ServiceCreate, db: Session = Depends(sql_db.get_db) ) -> Service: services = sql_crud.create_service(db=db, service=service) return services @router.post("/api/v1/service_usage", response_model=Service, tags=[Tags.services]) def add_service_usage( usage: ServiceUsageCreate, service_uuid: str = "bdd640fb-0667-1ad1-1c80-317fa3b1799d", db: Session = Depends(sql_db.get_db), ) -> Service: service = sql_crud.add_service_usage(db, service_uuid, usage) return service @router.put("/api/v1/inc_service_usage", response_model=Service, tags=[Tags.services]) def inc_service_usage( usage: ServiceUsageCreate, consumer_entity_did: str = "did:sov:test:120", service_uuid: str = "bdd640fb-0667-1ad1-1c80-317fa3b1799d", db: Session = Depends(sql_db.get_db), ) -> Service: service = sql_crud.increment_service_usage(db, service_uuid, consumer_entity_did) return service @router.put("/api/v1/service", response_model=Service, tags=[Tags.services]) def update_service( service: ServiceCreate, uuid: str = "bdd640fb-0667-1ad1-1c80-317fa3b1799d", db: Session = Depends(sql_db.get_db), ) -> Service: service = sql_crud.set_service(db, uuid, service) return service @router.get("/api/v1/services", response_model=List[Service], tags=[Tags.services]) def get_all_services( skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db) ) -> List[sql_models.Service]: services = sql_crud.get_services(db, skip=skip, limit=limit) return services @router.get( "/api/v1/service_by_did", response_model=List[Service], tags=[Tags.services] ) def get_service_by_did( entity_did: str = "did:sov:test:120", skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db), ) -> List[sql_models.Service]: service = sql_crud.get_services_by_entity_did(db, entity_did=entity_did) return service @router.get("/api/v1/service", response_model=Service, tags=[Tags.services]) def get_service_by_uuid( uuid: str = "bdd640fb-0667-1ad1-1c80-317fa3b1799d", skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db), ) -> Optional[sql_models.Service]: service = sql_crud.get_service_by_uuid(db, uuid=uuid) return service @router.get( "/api/v1/services_without_entity", response_model=List[Service], tags=[Tags.services], ) def get_services_without_entity( entity_did: str = "did:sov:test:120", skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db), ) -> List[sql_models.Service]: service = sql_crud.get_services_without_entity_id(db, entity_did=entity_did) return service @router.delete("/api/v1/service", tags=[Tags.services]) def delete_service( entity_did: str = "did:sov:test:120", db: Session = Depends(sql_db.get_db), ) -> dict[str, str]: sql_crud.delete_service_by_entity_did(db, entity_did) return {"message": "service deleted"} ######################### # # # Entity # # # ######################### @router.post("/api/v1/entity", response_model=Entity, tags=[Tags.entities]) def create_entity( entity: EntityCreate, db: Session = Depends(sql_db.get_db) ) -> sql_models.Entity: return sql_crud.create_entity(db, entity) @router.get( "/api/v1/entity_by_roles", response_model=List[Entity], tags=[Tags.entities] ) def get_entity_by_roles( roles: List[Role] = Query(...), db: Session = Depends(sql_db.get_db) ) -> List[sql_models.Entity]: entity = sql_crud.get_entity_by_role(db, roles=roles) return entity @router.get("/api/v1/entities", response_model=List[Entity], tags=[Tags.entities]) def get_all_entities( skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db) ) -> List[sql_models.Entity]: entities = sql_crud.get_entities(db, skip=skip, limit=limit) return entities @router.get("/api/v1/entity", response_model=Entity, tags=[Tags.entities]) def get_entity_by_did( entity_did: str = "did:sov:test:120", db: Session = Depends(sql_db.get_db), ) -> Optional[sql_models.Entity]: entity = sql_crud.get_entity_by_name_or_did(db, name=entity_did) return entity @router.get( "/api/v1/attached_entities", response_model=List[Entity], tags=[Tags.entities], ) def get_attached_entities( skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db) ) -> List[sql_models.Entity]: entities = sql_crud.get_attached_entities(db, skip=skip, limit=limit) return entities @router.put("/api/v1/detach", tags=[Tags.entities]) def detach_entity( background_tasks: BackgroundTasks, entity_did: str = "did:sov:test:120", skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db), ) -> dict[str, str]: sql_crud.set_stop_health_task(db, entity_did, True) return {"message": f"Detached {entity_did} successfully"} @router.put("/api/v1/attach", tags=[Tags.entities]) def attach_entity( background_tasks: BackgroundTasks, entity_did: str = "did:sov:test:120", skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db), ) -> dict[str, str]: entity = sql_crud.get_entity_by_did(db, did=entity_did) if entity is None: raise ClanError(f"Entity with did '{entity_did}' not found") url = f"http://{entity.ip}/health" sql_crud.set_stop_health_task(db, entity_did, False) print("Start health query at", url) background_tasks.add_task(attach_entity_loc, db, entity_did) return {"message": f"Started attachment task for {entity.name}"} @router.get("/api/v1/is_attached", tags=[Tags.entities]) def is_attached( entity_did: str = "did:sov:test:120", db: Session = Depends(sql_db.get_db) ) -> dict[str, str]: entity = sql_crud.get_entity_by_did(db, did=entity_did) if entity is None: raise ClanError(f"Entity with did '{entity_did}' not found") timer = 0.0 timeout = 2 while not entity.attached: time.sleep(0.1) timer += 0.1 if timer > timeout: url = f"http://{entity.ip}" raise ClanError(f"Entity at {url} not reachable") db.refresh(entity) return {"message": f"Attached to {entity.name} successfully"} def attach_entity_loc(db: Session, entity_did: str) -> None: entity = sql_crud.get_entity_by_did(db, did=entity_did) try: assert entity is not None url = f"http://{entity.ip}/health" while entity.stop_health_task is False: response = httpx.get(url, timeout=2) if response.status_code != 200: raise ClanError( f"Entity with did '{entity_did}' returned {response.status_code}" ) if entity.attached is False: sql_crud.set_attached_by_entity_did(db, entity_did, True) if entity is None: raise ClanError(f"Entity with did '{entity_did}' has been deleted") time.sleep(1) db.refresh(entity) except Exception: print(f"Entity {entity_did} not reachable at {url}") finally: sql_crud.set_attached_by_entity_did(db, entity_did, False) sql_crud.set_stop_health_task(db, entity_did, False) @router.delete("/api/v1/entity", tags=[Tags.entities]) def delete_entity( entity_did: str = "did:sov:test:120", db: Session = Depends(sql_db.get_db), ) -> dict[str, str]: sql_crud.delete_entity_by_did_recursive(db, did=entity_did) return {"message": "Entity deleted and all relations to that entity"} def get_rpc_by_role(db: Session, role: Role, path: str) -> Any: matching_entities = sql_crud.get_entity_by_role(db, roles=[role]) if matching_entities is None: raise ClanError(f"No {role} found") if len(matching_entities) > 1: raise ClanError(f"More than one {role} found") if len(matching_entities) == 0: raise ClanError(f"No {role} found") dlg = matching_entities[0] url = f"http://{dlg.ip}/{path}" try: response = httpx.get(url, timeout=2) except httpx.HTTPError as e: raise ClanError(f"{role} not reachable at {url}") from e if response.status_code != 200: raise ClanError(f"{role} returned {response.status_code}") return response.json() ######################### # # # Resolution # # # ######################### @router.get( "/api/v1/resolutions", response_model=List[Resolution], tags=[Tags.resolutions] ) def get_all_resolutions( skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db) ) -> List[Resolution]: return get_rpc_by_role(db, Role("DLG"), "dlg_list_of_did_resolutions") ######################### # # # Repository # # # ######################### @router.get( "/api/v1/repositories", tags=[Tags.repositories], response_model=List[Service] ) def get_all_repositories( skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db) ) -> List[Service]: return get_rpc_by_role(db, Role("AP"), "ap_list_of_services") ######################### # # # Eventmessage # # # ######################### @router.post( "/api/v1/event_message", response_model=Eventmessage, tags=[Tags.eventmessages] ) def create_eventmessage( eventmsg: EventmessageCreate, db: Session = Depends(sql_db.get_db) ) -> EventmessageCreate: return sql_crud.create_eventmessage(db, eventmsg) @typing.no_type_check @router.get( "/api/v1/event_messages", response_class=JSONResponse, tags=[Tags.eventmessages], ) def get_all_eventmessages( skip: int = 0, limit: int = 100, db: Session = Depends(sql_db.get_db) ) -> JSONResponse: eventmessages = sql_crud.get_eventmessages(db, skip=skip, limit=limit) result: dict[int, dict[int, List[Eventmessage]]] = {} for msg in eventmessages: msg_name = msg_type_to_label.get(msg.msg_type, None) src_name = sql_crud.get_entity_by_did(db, msg.src_did) src_name = src_name if src_name is None else src_name.name des_name = sql_crud.get_entity_by_did(db, msg.des_did) des_name = des_name if des_name is None else des_name.name if result.get(msg.group) is None: result[msg.group] = {} if result[msg.group].get(msg.group_id) is None: result[msg.group][msg.group_id] = [] result[msg.group][msg.group_id].append( Eventmessage( id=msg.id, timestamp=msg.timestamp, group=msg.group, group_id=msg.group_id, msg_type=msg.msg_type, msg_type_name=msg_name, src_did=msg.src_did, src_name=src_name, des_did=msg.des_did, des_name=des_name, msg=msg.msg, ).dict() ) return JSONResponse(content=result, status_code=200) ############################## # # # EMULATED API ENDPOINTS # # # ############################## @router.get("/emulate", response_class=HTMLResponse) def get_emulated_enpoints() -> HTMLResponse: html_content = f""" Emulated API

Emulated API

Emulated API endpoints for testing purposes.

DLG: {dlg_url}

AP: {ap_url}

C1: {c1_url}

C2: {c2_url}

""" return HTMLResponse(content=html_content, status_code=200)