From 4a234fa7c267f89db8660066935cb74c8ce092cb Mon Sep 17 00:00:00 2001 From: Denys Fedoryshchenko Date: Wed, 30 Oct 2024 13:11:56 +0200 Subject: [PATCH] feat(main.py): Add events endpoint Add initial implementation of events endpoint Signed-off-by: Denys Fedoryshchenko --- api/main.py | 60 ++++++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 59 insertions(+), 1 deletion(-) diff --git a/api/main.py b/api/main.py index d6994d19..50ae3654 100644 --- a/api/main.py +++ b/api/main.py @@ -24,6 +24,7 @@ Query, Body, ) +from fastapi.encoders import jsonable_encoder from fastapi.responses import JSONResponse, PlainTextResponse, FileResponse from fastapi.security import OAuth2PasswordRequestForm from fastapi_pagination import add_pagination @@ -54,6 +55,7 @@ UserUpdate, UserGroup, ) +import traceback @asynccontextmanager @@ -166,6 +168,19 @@ async def invalid_id_exception_handler( ) +@app.exception_handler(Exception) +async def server_error(request: Request, error: Exception): + """ + Global exception handler for all exceptions + """ + print(f"Internal server error: {error} \n{traceback.format_exc()}") + return JSONResponse( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, + content={"detail": "Internal server error (exception)"}, + ) + + + @app.get('/') async def root(): """Root endpoint handler""" @@ -498,9 +513,52 @@ def _get_eventhistory(evdict): return evhist +# TBD: Restrict response by Pydantic model +@app.get('/events') +async def get_events(request: Request): + """Get all the events if no request parameters have passed. + Format: [{event1}, {event2}, ...] or if recursive is set to true, + then we add to each event the node information. + Get all the matching events otherwise. + Query parameters can be used to filter the events: + - limit: Number of events to return + - from: Start timestamp (unix epoch) to filter events + - kind: Event kind to filter events + - recursive: Retrieve node together with event + This API endpoint is under development and may change in future. + """ + metrics.add('http_requests_total', 1) + query_params = dict(request.query_params) + recursive = query_params.pop('recursive', None) + limit = query_params.pop('limit', None) + kind = query_params.pop('kind', None) + state = query_params.pop('state', None) + from_ts = query_params.pop('from', None) + if from_ts: + query_params['timestamp'] = {'$gte': int(from_ts)} + if kind: + query_params['data.kind'] = kind + if state: + query_params['data.state'] = state + if limit: + query_params['limit'] = int(limit) + resp = await db.find_by_attributes_nonpaginated(EventHistory, query_params) + resp_list = [] + for item in resp: + item['id'] = str(item['_id']) + item.pop('_id') + if recursive: + node = await db.find_by_id(Node, item['data']['id']) + if node: + item['node'] = node + resp_list.append(item) + json_comp = jsonable_encoder(resp_list) + return JSONResponse(content=json_comp) + + + # ----------------------------------------------------------------------------- # Nodes - def _get_node_event_data(operation, node, is_hierarchy=False): return { 'op': operation,