Browse Source
including refactor of jsonable_encoder to allow other object and model typespull/11/head
10 changed files with 311 additions and 42 deletions
@ -0,0 +1,27 @@ |
|||||
|
from datetime import datetime, time, timedelta |
||||
|
from uuid import UUID |
||||
|
|
||||
|
from fastapi import Body, FastAPI |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.put("/items/{item_id}") |
||||
|
async def read_items( |
||||
|
item_id: UUID, |
||||
|
start_datetime: datetime = Body(None), |
||||
|
end_datetime: datetime = Body(None), |
||||
|
repeat_at: time = Body(None), |
||||
|
process_after: timedelta = Body(None), |
||||
|
): |
||||
|
start_process = start_datetime + process_after |
||||
|
duration = end_datetime - start_process |
||||
|
return { |
||||
|
"item_id": item_id, |
||||
|
"start_datetime": start_datetime, |
||||
|
"end_datetime": end_datetime, |
||||
|
"repeat_at": repeat_at, |
||||
|
"process_after": process_after, |
||||
|
"start_process": start_process, |
||||
|
"duration": duration, |
||||
|
} |
@ -0,0 +1,64 @@ |
|||||
|
Up to now, you have been using common data types, like: |
||||
|
|
||||
|
* `int` |
||||
|
* `float` |
||||
|
* `str` |
||||
|
* `bool` |
||||
|
|
||||
|
But you can also use more complex data types. |
||||
|
|
||||
|
And you will still have the same features as seen up to now: |
||||
|
|
||||
|
* Great editor support. |
||||
|
* Data conversion from incoming requests. |
||||
|
* Data conversion for response data. |
||||
|
* Data validation. |
||||
|
* Automatic annotation and documentation. |
||||
|
|
||||
|
## Other data types |
||||
|
|
||||
|
Here are some of the additional data types you can use: |
||||
|
|
||||
|
* `UUID`: |
||||
|
* A standard "Universally Unique Identifier", common as an ID in many databases and systems. |
||||
|
* In requests and responses will be represented as a `str`. |
||||
|
* `datetime.datetime`: |
||||
|
* A Python `datetime.datetime`. |
||||
|
* In requests and responses will be represented as a `str` in ISO 8601 format, like: `2008-09-15T15:53:00+05:00`. |
||||
|
* `datetime.date`: |
||||
|
* Python `datetime.date`. |
||||
|
* In requests and responses will be represented as a `str` in ISO 8601 format, like: `2008-09-15`. |
||||
|
* `datetime.time`: |
||||
|
* A Python `datetime.time`. |
||||
|
* In requests and responses will be represented as a `str` in ISO 8601 format, like: `14:23:55.003`. |
||||
|
* `datetime.timedelta`: |
||||
|
* A Python `datetime.timedelta`. |
||||
|
* In requests and responses will be represented as a `float` of total seconds. |
||||
|
* Pydantic also allows representing it as a "ISO 8601 time diff encoding", <a href="https://pydantic-docs.helpmanual.io/#json-serialisation" target="_blank">see the docs for more info</a>. |
||||
|
* `frozenset`: |
||||
|
* In requests and responses, treated the same as a `set`: |
||||
|
* In requests, a list will be read, eliminating duplicates and converting it to a `set`. |
||||
|
* In responses, the `set` will be converted to a `list`. |
||||
|
* The generated schema will specify that the `set` values are unique (using JSON Schema's `uniqueItems`). |
||||
|
* `bytes`: |
||||
|
* Standard Python `bytes`. |
||||
|
* In requests and responses will be treated as `str`. |
||||
|
* The generated schema will specify that it's a `str` with `binary` "format". |
||||
|
* `Decimal`: |
||||
|
* Standard Python `Decimal`. |
||||
|
* In requests and responses, handled the same as a `float`. |
||||
|
|
||||
|
|
||||
|
## Example |
||||
|
|
||||
|
Here's an example path operation with parameters using some of the above types. |
||||
|
|
||||
|
```Python hl_lines="1 2 11 12 13 14 15" |
||||
|
{!./src/extra_data_types/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
Note that the parameters inside the function have their natural data type, and you can, for example, perform normal date manipulations, like: |
||||
|
|
||||
|
```Python hl_lines="17 18" |
||||
|
{!./src/extra_data_types/tutorial001.py!} |
||||
|
``` |
@ -0,0 +1,136 @@ |
|||||
|
from starlette.testclient import TestClient |
||||
|
|
||||
|
from extra_data_types.tutorial001 import app |
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "Fast API", "version": "0.1.0"}, |
||||
|
"paths": { |
||||
|
"/items/{item_id}": { |
||||
|
"put": { |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": {"application/json": {"schema": {}}}, |
||||
|
}, |
||||
|
"422": { |
||||
|
"description": "Validation Error", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": { |
||||
|
"$ref": "#/components/schemas/HTTPValidationError" |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
"summary": "Read Items Put", |
||||
|
"operationId": "read_items_items__item_id__put", |
||||
|
"parameters": [ |
||||
|
{ |
||||
|
"required": True, |
||||
|
"schema": { |
||||
|
"title": "Item_Id", |
||||
|
"type": "string", |
||||
|
"format": "uuid", |
||||
|
}, |
||||
|
"name": "item_id", |
||||
|
"in": "path", |
||||
|
} |
||||
|
], |
||||
|
"requestBody": { |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": {"$ref": "#/components/schemas/Body_read_items"} |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
"components": { |
||||
|
"schemas": { |
||||
|
"Body_read_items": { |
||||
|
"title": "Body_read_items", |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"start_datetime": { |
||||
|
"title": "Start_Datetime", |
||||
|
"type": "string", |
||||
|
"format": "date-time", |
||||
|
}, |
||||
|
"end_datetime": { |
||||
|
"title": "End_Datetime", |
||||
|
"type": "string", |
||||
|
"format": "date-time", |
||||
|
}, |
||||
|
"repeat_at": { |
||||
|
"title": "Repeat_At", |
||||
|
"type": "string", |
||||
|
"format": "time", |
||||
|
}, |
||||
|
"process_after": { |
||||
|
"title": "Process_After", |
||||
|
"type": "string", |
||||
|
"format": "time-delta", |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
"ValidationError": { |
||||
|
"title": "ValidationError", |
||||
|
"required": ["loc", "msg", "type"], |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"loc": { |
||||
|
"title": "Location", |
||||
|
"type": "array", |
||||
|
"items": {"type": "string"}, |
||||
|
}, |
||||
|
"msg": {"title": "Message", "type": "string"}, |
||||
|
"type": {"title": "Error Type", "type": "string"}, |
||||
|
}, |
||||
|
}, |
||||
|
"HTTPValidationError": { |
||||
|
"title": "HTTPValidationError", |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"detail": { |
||||
|
"title": "Detail", |
||||
|
"type": "array", |
||||
|
"items": {"$ref": "#/components/schemas/ValidationError"}, |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def test_openapi_schema(): |
||||
|
response = client.get("/openapi.json") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == openapi_schema |
||||
|
|
||||
|
|
||||
|
def test_extra_types(): |
||||
|
item_id = "ff97dd87-a4a5-4a12-b412-cde99f33e00e" |
||||
|
data = { |
||||
|
"start_datetime": "2018-12-22T14:00:00+00:00", |
||||
|
"end_datetime": "2018-12-24T15:00:00+00:00", |
||||
|
"repeat_at": "15:30:00", |
||||
|
"process_after": 300, |
||||
|
} |
||||
|
expected_response = data.copy() |
||||
|
expected_response.update( |
||||
|
{ |
||||
|
"start_process": "2018-12-22T14:05:00+00:00", |
||||
|
"duration": 176_100, |
||||
|
"item_id": item_id, |
||||
|
} |
||||
|
) |
||||
|
response = client.put(f"/items/{item_id}", json=data) |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == expected_response |
Loading…
Reference in new issue