Skip to content

Commit

Permalink
Merge branch 'master' into feature/manage-schedules-with-cron
Browse files Browse the repository at this point in the history
  • Loading branch information
elegantmoose authored Sep 5, 2024
2 parents 24cde12 + 5521615 commit 949668a
Show file tree
Hide file tree
Showing 6 changed files with 59 additions and 6 deletions.
4 changes: 2 additions & 2 deletions .github/workflows/stale.yml
Original file line number Diff line number Diff line change
Expand Up @@ -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
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -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

Expand Down
18 changes: 18 additions & 0 deletions app/api/v2/handlers/planner_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -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',
Expand Down Expand Up @@ -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)
10 changes: 7 additions & 3 deletions app/api/v2/responses.py
Original file line number Diff line number Diff line change
@@ -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
Expand Down Expand Up @@ -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(
Expand Down
8 changes: 8 additions & 0 deletions app/objects/c_planner.py
Original file line number Diff line number Diff line change
Expand Up @@ -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)
Expand Down Expand Up @@ -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):
Expand Down
23 changes: 23 additions & 0 deletions tests/api/v2/handlers/test_planners_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -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)
Expand Down Expand Up @@ -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

0 comments on commit 949668a

Please sign in to comment.