From 40e2097d425a9be6bdac6b8cd5636c84166bace2 Mon Sep 17 00:00:00 2001 From: samuel-sirven-bib Date: Fri, 5 Jul 2024 16:01:25 +0200 Subject: [PATCH 1/4] Fix Internal Server Error 500 for ValidationError - Change the middleware in responses.py for ValidationError to raise an HTTPUnprocessableEntity instead of a JsonHttpBadRequest --- app/api/v2/responses.py | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/app/api/v2/responses.py b/app/api/v2/responses.py index 278e4f07e..d1fc58c2a 100644 --- a/app/api/v2/responses.py +++ b/app/api/v2/responses.py @@ -1,5 +1,9 @@ +import json + from aiohttp import web from json import JSONDecodeError + +from aiohttp.web_exceptions import HTTPUnprocessableEntity from marshmallow.exceptions import ValidationError from app.api.v2 import errors @@ -51,9 +55,9 @@ async def apispec_request_validation_middleware(request, handler): ) except ValidationError as ex: # ex: List of objects sent when single object expected - raise JsonHttpBadRequest( - error='Error parsing JSON: Could not validate Schema', - details=str(ex) + formatted_message = json.dumps({"json": ex.messages}, indent=2) + raise HTTPUnprocessableEntity( + text=formatted_message ) except JSONDecodeError as ex: raise JsonHttpBadRequest( From e9c8c1a2257e7b23b1554d2d26abe6d9126ea768 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Louis=20Hackl=C3=A4nder?= Date: Mon, 22 Jul 2024 14:14:39 +0200 Subject: [PATCH 2/4] Add API functionality to update planners --- app/api/v2/handlers/planner_api.py | 18 +++++++++++++++++ app/objects/c_planner.py | 8 ++++++++ tests/api/v2/handlers/test_planners_api.py | 23 ++++++++++++++++++++++ 3 files changed, 49 insertions(+) diff --git a/app/api/v2/handlers/planner_api.py b/app/api/v2/handlers/planner_api.py index f109907a8..8b9b358fd 100644 --- a/app/api/v2/handlers/planner_api.py +++ b/app/api/v2/handlers/planner_api.py @@ -17,6 +17,7 @@ def add_routes(self, app: web.Application): router = app.router router.add_get('/planners', self.get_planners) router.add_get('/planners/{planner_id}', self.get_planner_by_id) + router.add_patch('/planners/{planner_id}', self.update_planner) @aiohttp_apispec.docs(tags=['planners'], summary='Retrieve planners', @@ -49,3 +50,20 @@ async def get_planners(self, request: web.Request): async def get_planner_by_id(self, request: web.Request): planner = await self.get_object(request) return web.json_response(planner) + + @aiohttp_apispec.docs(tags=['planners'], + summary='Updates an existing planner.', + description='Updates a planner based on the `PlannerSchema` value provided in the message body.', + parameters=[{ + 'in': 'path', + 'name': 'planner_id', + 'schema': {'type': 'string'}, + 'required': 'true', + 'description': 'UUID of the Planner to be updated' + }]) + @aiohttp_apispec.request_schema(PlannerSchema(partial=True)) + @aiohttp_apispec.response_schema(PlannerSchema(partial=True), + description='JSON dictionary representation of the replaced Planner.') + async def update_planner(self, request: web.Request): + planner = await self.update_on_disk_object(request) + return web.json_response(planner.display) diff --git a/app/objects/c_planner.py b/app/objects/c_planner.py index 010c08752..ed06f2af0 100644 --- a/app/objects/c_planner.py +++ b/app/objects/c_planner.py @@ -18,6 +18,12 @@ class PlannerSchema(ma.Schema): allow_repeatable_abilities = ma.fields.Boolean() plugin = ma.fields.String(load_default=None) + @ma.pre_load + def fix_id(self, data, **_): + if 'planner_id' in data: + data['id'] = data.pop('planner_id') + return data + @ma.post_load() def build_planner(self, data, **kwargs): return None if kwargs.get('partial') is True else Planner(**data) @@ -54,6 +60,8 @@ def store(self, ram): existing.update('stopping_conditions', self.stopping_conditions) existing.update('params', self.params) existing.update('plugin', self.plugin) + existing.update('description', self.description) + existing.update('allow_repeatable_abilities', self.allow_repeatable_abilities) return existing async def which_plugin(self): diff --git a/tests/api/v2/handlers/test_planners_api.py b/tests/api/v2/handlers/test_planners_api.py index 47afc2b22..09b371115 100644 --- a/tests/api/v2/handlers/test_planners_api.py +++ b/tests/api/v2/handlers/test_planners_api.py @@ -26,6 +26,13 @@ def expected_test_planner_dump(test_planner): return test_planner.display_schema.dump(test_planner) +@pytest.fixture +def updated_planner(test_planner): + planner_dict = test_planner.schema.dump(test_planner) + planner_dict.update(dict(description="a test planner with updated description")) + return planner_dict + + class TestPlannersApi: async def test_get_planners(self, api_v2_client, api_cookies, test_planner, expected_test_planner_dump): resp = await api_v2_client.get('/api/v2/planners', cookies=api_cookies) @@ -57,3 +64,19 @@ async def test_planner_defaults(self, api_v2_client, api_cookies, test_planner, assert len(planners_list) == 2 assert planners_list[0]["id"] == "456" assert planners_list[0]["name"][0] > planners_list[1]["name"][0] # prove that this wasn't an alphabetical sort + + async def test_update_planner(self, api_v2_client, api_cookies, test_planner, updated_planner, mocker): + with mocker.patch('app.api.v2.managers.base_api_manager.BaseApiManager.strip_yml') as mock_strip_yml: + mock_strip_yml.return_value = [test_planner.schema.dump(test_planner)] + resp = await api_v2_client.patch('/api/v2/planners/123', cookies=api_cookies, json=updated_planner) + assert resp.status == HTTPStatus.OK + planner = (await BaseService.get_service('data_svc').locate('planners'))[0] + assert planner.description == updated_planner["description"] + + async def test_unauthorized_update_planner(self, api_v2_client, updated_planner): + resp = await api_v2_client.patch('/api/v2/planners/123', json=updated_planner) + assert resp.status == HTTPStatus.UNAUTHORIZED + + async def test_update_nonexistent_planner(self, api_v2_client, api_cookies, updated_planner): + resp = await api_v2_client.patch('/api/v2/planners/999', cookies=api_cookies, json=updated_planner) + assert resp.status == HTTPStatus.NOT_FOUND From cd1540cb35abc125c7014c1e623c4934a084d385 Mon Sep 17 00:00:00 2001 From: "P.S.i Coder" <82137307+psicoder85@users.noreply.github.com> Date: Wed, 24 Jul 2024 21:55:03 -0600 Subject: [PATCH 3/4] Update README.md Documentation update --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 9e85fa2d9..a5e813263 100644 --- a/README.md +++ b/README.md @@ -141,7 +141,7 @@ Refer to our [contributor documentation](CONTRIBUTING.md). ## Vulnerability Disclosures -Refer to our [vulnerability discolosure documentation](SECURITY.md) for submitting bugs. +Refer to our [Vulnerability Disclosure Documentation](SECURITY.md) for submitting bugs. ## Licensing From b93877dd881c8beb29557f917a2b13ad3f0cf013 Mon Sep 17 00:00:00 2001 From: Michael Kouremetis Date: Sat, 3 Aug 2024 11:29:25 -0400 Subject: [PATCH 4/4] Update stale.yml --- .github/workflows/stale.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/stale.yml b/.github/workflows/stale.yml index 782dac96e..2c99dfbfa 100644 --- a/.github/workflows/stale.yml +++ b/.github/workflows/stale.yml @@ -25,5 +25,5 @@ jobs: stale-pr-message: 'This issue is stale because it has been open 30 days with no activity. Remove stale label or comment or this will be closed in 5 days' stale-issue-message: 'This issue is stale because it has been open 30 days with no activity. Remove stale label or comment or this will be closed in 5 days' exempt-issue-labels: 'feature,keep' - days-before-stale: 45 - days-before-close: 30 + days-before-stale: 60 + days-before-close: 60