Browse Source
* 📝 Update Release Notes with issue templates * ✨ Add HTTPException with support for headers Including docs and tests * 📝 Update Security docs to use new HTTPExceptionpull/41/head
committed by
GitHub
16 changed files with 452 additions and 14 deletions
@ -0,0 +1,12 @@ |
|||||
|
from fastapi import FastAPI, HTTPException |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
items = {"foo": "The Foo Wrestlers"} |
||||
|
|
||||
|
|
||||
|
@app.get("/items/{item_id}") |
||||
|
async def create_item(item_id: str): |
||||
|
if item_id not in items: |
||||
|
raise HTTPException(status_code=404, detail="Item not found") |
||||
|
return {"item": items[item_id]} |
@ -0,0 +1,16 @@ |
|||||
|
from fastapi import FastAPI, HTTPException |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
items = {"foo": "The Foo Wrestlers"} |
||||
|
|
||||
|
|
||||
|
@app.get("/items-header/{item_id}") |
||||
|
async def create_item_header(item_id: str): |
||||
|
if item_id not in items: |
||||
|
raise HTTPException( |
||||
|
status_code=404, |
||||
|
detail="Item not found", |
||||
|
headers={"X-Error": "There goes my error"}, |
||||
|
) |
||||
|
return {"item": items[item_id]} |
@ -0,0 +1,57 @@ |
|||||
|
There are many situations in where you need to notify an error to the client that is using your API. |
||||
|
|
||||
|
This client could be a browser with a frontend, the code from someone else, an IoT device, etc. |
||||
|
|
||||
|
You could need to tell that client that: |
||||
|
|
||||
|
* He doesn't have enough privileges for that operation. |
||||
|
* He doesn't have access to that resource. |
||||
|
* The item he was trying to access doesn't exist. |
||||
|
* etc. |
||||
|
|
||||
|
In these cases, you would normally return an **HTTP status code** in the range of **400** (from 400 to 499). |
||||
|
|
||||
|
This is similar to the 200 HTTP status codes (from 200 to 299). Those "200" status codes mean that somehow there was a "success" in the request. |
||||
|
|
||||
|
The status codes in the 400 range mean that there was an error from the client. |
||||
|
|
||||
|
Remember all those **"404 Not Found"** errors (and jokes)? |
||||
|
|
||||
|
## Use `HTTPException` |
||||
|
|
||||
|
To return HTTP responses with errors to the client you use `HTTPException`. |
||||
|
|
||||
|
### Import `HTTPException` |
||||
|
|
||||
|
```Python hl_lines="1" |
||||
|
{!./src/handling_errors/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
### Raise an `HTTPException` in your code |
||||
|
|
||||
|
`HTTPException` is a normal Python exception with additional data relevant for APIs. |
||||
|
|
||||
|
Because it's a Python exception, you don't `return` it, you `raise` it. |
||||
|
|
||||
|
This also means that if you are inside a utility function that you are calling inside of your path operation function, and you raise the `HTTPException` from inside of that utility function, it won't run the rest of the code in the path operation function, it will terminate that request right away and send the HTTP error from the `HTTPException` to the client. |
||||
|
|
||||
|
The benefit of raising an exception over `return`ing a value will be more evident in the section about Dependencies and Security. |
||||
|
|
||||
|
In this example, when the client request an item by an ID that doesn't exist, raise an exception with a status code of `404`: |
||||
|
|
||||
|
```Python hl_lines="11" |
||||
|
{!./src/handling_errors/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
### Adding custom headers |
||||
|
|
||||
|
There are some situations in where it's useful to be able to add custom headers to the HTTP error. For example, for some types of security. |
||||
|
|
||||
|
You probably won't need to use it directly in your code. |
||||
|
|
||||
|
But in case you needed it for an advanced scenario, you can add custom headers: |
||||
|
|
||||
|
|
||||
|
```Python hl_lines="14" |
||||
|
{!./src/handling_errors/tutorial002.py!} |
||||
|
``` |
@ -0,0 +1,9 @@ |
|||||
|
from starlette.exceptions import HTTPException as StarletteHTTPException |
||||
|
|
||||
|
|
||||
|
class HTTPException(StarletteHTTPException): |
||||
|
def __init__( |
||||
|
self, status_code: int, detail: str = None, headers: dict = None |
||||
|
) -> None: |
||||
|
super().__init__(status_code=status_code, detail=detail) |
||||
|
self.headers = headers |
@ -0,0 +1,156 @@ |
|||||
|
from fastapi import FastAPI, HTTPException |
||||
|
from starlette.exceptions import HTTPException as StarletteHTTPException |
||||
|
from starlette.testclient import TestClient |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
items = {"foo": "The Foo Wrestlers"} |
||||
|
|
||||
|
|
||||
|
@app.get("/items/{item_id}") |
||||
|
async def create_item(item_id: str): |
||||
|
if item_id not in items: |
||||
|
raise HTTPException( |
||||
|
status_code=404, |
||||
|
detail="Item not found", |
||||
|
headers={"X-Error": "Some custom header"}, |
||||
|
) |
||||
|
return {"item": items[item_id]} |
||||
|
|
||||
|
|
||||
|
@app.get("/starlette-items/{item_id}") |
||||
|
async def create_item(item_id: str): |
||||
|
if item_id not in items: |
||||
|
raise StarletteHTTPException(status_code=404, detail="Item not found") |
||||
|
return {"item": items[item_id]} |
||||
|
|
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "Fast API", "version": "0.1.0"}, |
||||
|
"paths": { |
||||
|
"/items/{item_id}": { |
||||
|
"get": { |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": {"application/json": {"schema": {}}}, |
||||
|
}, |
||||
|
"422": { |
||||
|
"description": "Validation Error", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": { |
||||
|
"$ref": "#/components/schemas/HTTPValidationError" |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
"summary": "Create Item Get", |
||||
|
"operationId": "create_item_items__item_id__get", |
||||
|
"parameters": [ |
||||
|
{ |
||||
|
"required": True, |
||||
|
"schema": {"title": "Item_Id", "type": "string"}, |
||||
|
"name": "item_id", |
||||
|
"in": "path", |
||||
|
} |
||||
|
], |
||||
|
} |
||||
|
}, |
||||
|
"/starlette-items/{item_id}": { |
||||
|
"get": { |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": {"application/json": {"schema": {}}}, |
||||
|
}, |
||||
|
"422": { |
||||
|
"description": "Validation Error", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": { |
||||
|
"$ref": "#/components/schemas/HTTPValidationError" |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
"summary": "Create Item Get", |
||||
|
"operationId": "create_item_starlette-items__item_id__get", |
||||
|
"parameters": [ |
||||
|
{ |
||||
|
"required": True, |
||||
|
"schema": {"title": "Item_Id", "type": "string"}, |
||||
|
"name": "item_id", |
||||
|
"in": "path", |
||||
|
} |
||||
|
], |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
"components": { |
||||
|
"schemas": { |
||||
|
"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_get_item(): |
||||
|
response = client.get("/items/foo") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == {"item": "The Foo Wrestlers"} |
||||
|
|
||||
|
|
||||
|
def test_get_item_not_found(): |
||||
|
response = client.get("/items/bar") |
||||
|
assert response.status_code == 404 |
||||
|
assert response.headers.get("x-error") == "Some custom header" |
||||
|
assert response.json() == {"detail": "Item not found"} |
||||
|
|
||||
|
|
||||
|
def test_get_starlette_item(): |
||||
|
response = client.get("/starlette-items/foo") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == {"item": "The Foo Wrestlers"} |
||||
|
|
||||
|
|
||||
|
def test_get_starlette_item_not_found(): |
||||
|
response = client.get("/starlette-items/bar") |
||||
|
assert response.status_code == 404 |
||||
|
assert response.headers.get("x-error") is None |
||||
|
assert response.json() == {"detail": "Item not found"} |
@ -0,0 +1,90 @@ |
|||||
|
from starlette.testclient import TestClient |
||||
|
|
||||
|
from handling_errors.tutorial001 import app |
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "Fast API", "version": "0.1.0"}, |
||||
|
"paths": { |
||||
|
"/items/{item_id}": { |
||||
|
"get": { |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": {"application/json": {"schema": {}}}, |
||||
|
}, |
||||
|
"422": { |
||||
|
"description": "Validation Error", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": { |
||||
|
"$ref": "#/components/schemas/HTTPValidationError" |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
"summary": "Create Item Get", |
||||
|
"operationId": "create_item_items__item_id__get", |
||||
|
"parameters": [ |
||||
|
{ |
||||
|
"required": True, |
||||
|
"schema": {"title": "Item_Id", "type": "string"}, |
||||
|
"name": "item_id", |
||||
|
"in": "path", |
||||
|
} |
||||
|
], |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
"components": { |
||||
|
"schemas": { |
||||
|
"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_get_item(): |
||||
|
response = client.get("/items/foo") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == {"item": "The Foo Wrestlers"} |
||||
|
|
||||
|
|
||||
|
def test_get_item_not_found(): |
||||
|
response = client.get("/items/bar") |
||||
|
assert response.status_code == 404 |
||||
|
assert response.headers.get("x-error") is None |
||||
|
assert response.json() == {"detail": "Item not found"} |
@ -0,0 +1,90 @@ |
|||||
|
from starlette.testclient import TestClient |
||||
|
|
||||
|
from handling_errors.tutorial002 import app |
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "Fast API", "version": "0.1.0"}, |
||||
|
"paths": { |
||||
|
"/items-header/{item_id}": { |
||||
|
"get": { |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": {"application/json": {"schema": {}}}, |
||||
|
}, |
||||
|
"422": { |
||||
|
"description": "Validation Error", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": { |
||||
|
"$ref": "#/components/schemas/HTTPValidationError" |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
"summary": "Create Item Header Get", |
||||
|
"operationId": "create_item_header_items-header__item_id__get", |
||||
|
"parameters": [ |
||||
|
{ |
||||
|
"required": True, |
||||
|
"schema": {"title": "Item_Id", "type": "string"}, |
||||
|
"name": "item_id", |
||||
|
"in": "path", |
||||
|
} |
||||
|
], |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
"components": { |
||||
|
"schemas": { |
||||
|
"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_get_item_header(): |
||||
|
response = client.get("/items-header/foo") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == {"item": "The Foo Wrestlers"} |
||||
|
|
||||
|
|
||||
|
def test_get_item_not_found_header(): |
||||
|
response = client.get("/items-header/bar") |
||||
|
assert response.status_code == 404 |
||||
|
assert response.headers.get("x-error") == "There goes my error" |
||||
|
assert response.json() == {"detail": "Item not found"} |
Loading…
Reference in new issue