Browse Source
* ✏️ Fix typo in security intro * ✨ Add testing docs and tests * 🐛 Debug Travis coverage * 🐛 Debug Travis coverage, report XML * 💚 Make Travis/Flit use same code install * ⏪ Revert Travis/Codecov debugging changespull/155/head
committed by
GitHub
17 changed files with 241 additions and 5 deletions
@ -0,0 +1,8 @@ |
|||
from fastapi import FastAPI |
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
@app.get("/") |
|||
async def read_main(): |
|||
return {"msg": "Hello World"} |
@ -0,0 +1,11 @@ |
|||
from starlette.testclient import TestClient |
|||
|
|||
from .main import app |
|||
|
|||
client = TestClient(app) |
|||
|
|||
|
|||
def test_read_main(): |
|||
response = client.get("/") |
|||
assert response.status_code == 200 |
|||
assert response.json() == {"msg": "Hello World"} |
@ -0,0 +1,18 @@ |
|||
from fastapi import FastAPI |
|||
from starlette.testclient import TestClient |
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
@app.get("/") |
|||
async def read_main(): |
|||
return {"msg": "Hello World"} |
|||
|
|||
|
|||
client = TestClient(app) |
|||
|
|||
|
|||
def test_read_main(): |
|||
response = client.get("/") |
|||
assert response.status_code == 200 |
|||
assert response.json() == {"msg": "Hello World"} |
@ -0,0 +1,31 @@ |
|||
from fastapi import FastAPI |
|||
from starlette.testclient import TestClient |
|||
from starlette.websockets import WebSocket |
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
@app.get("/") |
|||
async def read_main(): |
|||
return {"msg": "Hello World"} |
|||
|
|||
|
|||
@app.websocket_route("/ws") |
|||
async def websocket(websocket: WebSocket): |
|||
await websocket.accept() |
|||
await websocket.send_json({"msg": "Hello WebSocket"}) |
|||
await websocket.close() |
|||
|
|||
|
|||
def test_read_main(): |
|||
client = TestClient(app) |
|||
response = client.get("/") |
|||
assert response.status_code == 200 |
|||
assert response.json() == {"msg": "Hello World"} |
|||
|
|||
|
|||
def test_websocket(): |
|||
client = TestClient(app) |
|||
with client.websocket_connect("/ws") as websocket: |
|||
data = websocket.receive_json() |
|||
assert data == {"msg": "Hello WebSocket"} |
@ -0,0 +1,24 @@ |
|||
from fastapi import FastAPI |
|||
from starlette.testclient import TestClient |
|||
|
|||
app = FastAPI() |
|||
|
|||
items = {} |
|||
|
|||
|
|||
@app.on_event("startup") |
|||
async def startup_event(): |
|||
items["foo"] = {"name": "Fighters"} |
|||
items["bar"] = {"name": "Tenders"} |
|||
|
|||
|
|||
@app.get("/items/{item_id}") |
|||
async def read_items(item_id: str): |
|||
return items[item_id] |
|||
|
|||
|
|||
def test_read_items(): |
|||
with TestClient(app) as client: |
|||
response = client.get("/items/foo") |
|||
assert response.status_code == 200 |
|||
assert response.json() == {"name": "Fighters"} |
@ -0,0 +1,69 @@ |
|||
Thanks to <a href="https://www.starlette.io/testclient/" target="_blank">Starlette's TestClient</a>, testing **FastAPI** applications is easy and enjoyable. |
|||
|
|||
It is based on <a href="http://docs.python-requests.org" target="_blank">Requests</a>, so it's very familiar and intuitive. |
|||
|
|||
With it, you can use <a href="https://docs.pytest.org/" target="_blank">pytest</a> directly with **FastAPI**. |
|||
|
|||
## Using `TestClient` |
|||
|
|||
Import `TestClient` from `starlette.testclient`. |
|||
|
|||
Create a `TestClient` passing to it your **FastAPI**. |
|||
|
|||
Create functions with a name that starts with `test_` (this is standard `pytest` conventions). |
|||
|
|||
Use the `TestClient` object the same way as you do with `requests`. |
|||
|
|||
Write simple `assert` statements with the standard Python expressions that you need to check (again, standard `pytest`). |
|||
|
|||
```Python hl_lines="2 12 15 16 17 18" |
|||
{!./src/app_testing/tutorial001.py!} |
|||
``` |
|||
|
|||
!!! tip |
|||
Notice that the testing functions are normal `def`, not `async def`. |
|||
|
|||
And the calls to the client are also normal calls, not using `await`. |
|||
|
|||
This allows you to use `pytest` directly without complications. |
|||
|
|||
|
|||
## Separating tests |
|||
|
|||
In a real application, you probably would have your tests in a different file. |
|||
|
|||
And your **FastAPI** application might also be composed of several files/modules, etc. |
|||
|
|||
### **FastAPI** app file |
|||
|
|||
Let's say you have a file `main.py` with your **FastAPI** app: |
|||
|
|||
```Python |
|||
{!./src/app_testing/main.py!} |
|||
``` |
|||
|
|||
### Testing file |
|||
|
|||
Then you could have a file `test_main.py` with your tests, and import your `app` from the `main` module (`main.py`): |
|||
|
|||
```Python |
|||
{!./src/app_testing/test_main.py!} |
|||
``` |
|||
|
|||
## Testing WebSockets |
|||
|
|||
You can use the same `TestClient` to test WebSockets. |
|||
|
|||
For this, you use the `TestClient` in a `with` statement, connecting to the WebSocket: |
|||
|
|||
```Python hl_lines="27 28 29 30 31" |
|||
{!./src/app_testing/tutorial002.py!} |
|||
``` |
|||
|
|||
## Testing Events, `startup` and `shutdown` |
|||
|
|||
When you need your event handlers (`startup` and `shutdown`) to run in your tests, you can use the `TestClient` with a `with` statement: |
|||
|
|||
```Python hl_lines="9 10 11 12 20 21 22 23 24" |
|||
{!./src/app_testing/tutorial003.py!} |
|||
``` |
@ -0,0 +1,30 @@ |
|||
from app_testing.test_main import client, test_read_main |
|||
|
|||
openapi_schema = { |
|||
"openapi": "3.0.2", |
|||
"info": {"title": "Fast API", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/": { |
|||
"get": { |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
} |
|||
}, |
|||
"summary": "Read Main Get", |
|||
"operationId": "read_main__get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
def test_openapi_schema(): |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200 |
|||
assert response.json() == openapi_schema |
|||
|
|||
|
|||
def test_main(): |
|||
test_read_main() |
@ -0,0 +1,30 @@ |
|||
from app_testing.tutorial001 import client, test_read_main |
|||
|
|||
openapi_schema = { |
|||
"openapi": "3.0.2", |
|||
"info": {"title": "Fast API", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/": { |
|||
"get": { |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
} |
|||
}, |
|||
"summary": "Read Main Get", |
|||
"operationId": "read_main__get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
def test_openapi_schema(): |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200 |
|||
assert response.json() == openapi_schema |
|||
|
|||
|
|||
def test_main(): |
|||
test_read_main() |
@ -0,0 +1,9 @@ |
|||
from app_testing.tutorial002 import test_read_main, test_websocket |
|||
|
|||
|
|||
def test_main(): |
|||
test_read_main() |
|||
|
|||
|
|||
def test_ws(): |
|||
test_websocket() |
@ -0,0 +1,5 @@ |
|||
from app_testing.tutorial003 import test_read_items |
|||
|
|||
|
|||
def test_main(): |
|||
test_read_items() |
Loading…
Reference in new issue