committed by
GitHub
15 changed files with 1065 additions and 16 deletions
@ -0,0 +1,30 @@ |
|||
from . import models, schemas |
|||
|
|||
|
|||
def get_user(user_id: int): |
|||
return models.User.filter(models.User.id == user_id).first() |
|||
|
|||
|
|||
def get_user_by_email(email: str): |
|||
return models.User.filter(models.User.email == email).first() |
|||
|
|||
|
|||
def get_users(skip: int = 0, limit: int = 100): |
|||
return list(models.User.select().offset(skip).limit(limit)) |
|||
|
|||
|
|||
def create_user(user: schemas.UserCreate): |
|||
fake_hashed_password = user.password + "notreallyhashed" |
|||
db_user = models.User(email=user.email, hashed_password=fake_hashed_password) |
|||
db_user.save() |
|||
return db_user |
|||
|
|||
|
|||
def get_items(skip: int = 0, limit: int = 100): |
|||
return list(models.Item.select().offset(skip).limit(limit)) |
|||
|
|||
|
|||
def create_user_item(item: schemas.ItemCreate, user_id: int): |
|||
db_item = models.Item(**item.dict(), owner_id=user_id) |
|||
db_item.save() |
|||
return db_item |
@ -0,0 +1,26 @@ |
|||
from contextvars import ContextVar |
|||
|
|||
import peewee |
|||
|
|||
DATABASE_NAME = "test.db" |
|||
|
|||
|
|||
class PeeweeConnectionState(peewee._ConnectionState): |
|||
def __init__(self, **kwargs): |
|||
super().__setattr__("_state", {}) |
|||
self._state["closed"] = ContextVar("closed", default=True) |
|||
self._state["conn"] = ContextVar("conn", default=None) |
|||
self._state["ctx"] = ContextVar("ctx", default=[]) |
|||
self._state["transactions"] = ContextVar("transactions", default=[]) |
|||
super().__init__(**kwargs) |
|||
|
|||
def __setattr__(self, name, value): |
|||
self._state[name].set(value) |
|||
|
|||
def __getattr__(self, name): |
|||
return self._state[name].get() |
|||
|
|||
|
|||
db = peewee.SqliteDatabase(DATABASE_NAME, check_same_thread=False) |
|||
|
|||
db._state = PeeweeConnectionState() |
@ -0,0 +1,70 @@ |
|||
import time |
|||
from typing import List |
|||
|
|||
from fastapi import Depends, FastAPI, HTTPException |
|||
|
|||
from . import crud, database, models, schemas |
|||
|
|||
database.db.connect() |
|||
database.db.create_tables([models.User, models.Item]) |
|||
database.db.close() |
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
# Dependency |
|||
def get_db(): |
|||
try: |
|||
database.db.connect() |
|||
yield |
|||
finally: |
|||
if not database.db.is_closed(): |
|||
database.db.close() |
|||
|
|||
|
|||
@app.post("/users/", response_model=schemas.User, dependencies=[Depends(get_db)]) |
|||
def create_user(user: schemas.UserCreate): |
|||
db_user = crud.get_user_by_email(email=user.email) |
|||
if db_user: |
|||
raise HTTPException(status_code=400, detail="Email already registered") |
|||
return crud.create_user(user=user) |
|||
|
|||
|
|||
@app.get("/users/", response_model=List[schemas.User], dependencies=[Depends(get_db)]) |
|||
def read_users(skip: int = 0, limit: int = 100): |
|||
users = crud.get_users(skip=skip, limit=limit) |
|||
return users |
|||
|
|||
|
|||
@app.get( |
|||
"/users/{user_id}", response_model=schemas.User, dependencies=[Depends(get_db)] |
|||
) |
|||
def read_user(user_id: int): |
|||
db_user = crud.get_user(user_id=user_id) |
|||
if db_user is None: |
|||
raise HTTPException(status_code=404, detail="User not found") |
|||
return db_user |
|||
|
|||
|
|||
@app.post( |
|||
"/users/{user_id}/items/", |
|||
response_model=schemas.Item, |
|||
dependencies=[Depends(get_db)], |
|||
) |
|||
def create_item_for_user(user_id: int, item: schemas.ItemCreate): |
|||
return crud.create_user_item(item=item, user_id=user_id) |
|||
|
|||
|
|||
@app.get("/items/", response_model=List[schemas.Item], dependencies=[Depends(get_db)]) |
|||
def read_items(skip: int = 0, limit: int = 100): |
|||
items = crud.get_items(skip=skip, limit=limit) |
|||
return items |
|||
|
|||
|
|||
@app.get( |
|||
"/slowusers/", response_model=List[schemas.User], dependencies=[Depends(get_db)] |
|||
) |
|||
def read_slow_users(skip: int = 0, limit: int = 100): |
|||
time.sleep(15) # Fake long processing request |
|||
users = crud.get_users(skip=skip, limit=limit) |
|||
return users |
@ -0,0 +1,21 @@ |
|||
import peewee |
|||
|
|||
from .database import db |
|||
|
|||
|
|||
class User(peewee.Model): |
|||
email = peewee.CharField(unique=True, index=True) |
|||
hashed_password = peewee.CharField() |
|||
is_active = peewee.BooleanField(default=True) |
|||
|
|||
class Meta: |
|||
database = db |
|||
|
|||
|
|||
class Item(peewee.Model): |
|||
title = peewee.CharField(index=True) |
|||
description = peewee.CharField(index=True) |
|||
owner = peewee.ForeignKeyField(User, backref="items") |
|||
|
|||
class Meta: |
|||
database = db |
@ -0,0 +1,49 @@ |
|||
from typing import Any, List |
|||
|
|||
import peewee |
|||
from pydantic import BaseModel |
|||
from pydantic.utils import GetterDict |
|||
|
|||
|
|||
class PeeweeGetterDict(GetterDict): |
|||
def get(self, key: Any, default: Any = None): |
|||
res = getattr(self._obj, key, default) |
|||
if isinstance(res, peewee.ModelSelect): |
|||
return list(res) |
|||
return res |
|||
|
|||
|
|||
class ItemBase(BaseModel): |
|||
title: str |
|||
description: str = None |
|||
|
|||
|
|||
class ItemCreate(ItemBase): |
|||
pass |
|||
|
|||
|
|||
class Item(ItemBase): |
|||
id: int |
|||
owner_id: int |
|||
|
|||
class Config: |
|||
orm_mode = True |
|||
getter_dict = PeeweeGetterDict |
|||
|
|||
|
|||
class UserBase(BaseModel): |
|||
email: str |
|||
|
|||
|
|||
class UserCreate(UserBase): |
|||
password: str |
|||
|
|||
|
|||
class User(UserBase): |
|||
id: int |
|||
is_active: bool |
|||
items: List[Item] = [] |
|||
|
|||
class Config: |
|||
orm_mode = True |
|||
getter_dict = PeeweeGetterDict |
@ -0,0 +1,408 @@ |
|||
!!! warning |
|||
If you are just starting, the <a href="https://fastapi.tiangolo.com/tutorial/sql-databases/" target="_blank">SQLAlchemy tutorial</a> should be enough. |
|||
|
|||
Feel free to skip this. |
|||
|
|||
If you are starting a project from scratch, you are probably better off with <a href="https://fastapi.tiangolo.com/tutorial/sql-databases/" target="_blank">SQLAlchemy ORM</a>, or any other async ORM. |
|||
|
|||
If you already have a code base that uses <a href="http://docs.peewee-orm.com/en/latest/" target="_blank">Peewee ORM</a>, you can check here how to use it with **FastAPI**. |
|||
|
|||
!!! warning "Python 3.7+ required" |
|||
You will need Python 3.7 or above to safely use Peewee with FastAPI. |
|||
|
|||
## Peewee for async |
|||
|
|||
Peewee was not designed for async frameworks, or with them in mind. |
|||
|
|||
Peewee has some heavy assumptions about its defaults and about how it should be used. |
|||
|
|||
If you are developing an application with an older non-async framework, and can work with all its defaults, **it can be a great tool**. |
|||
|
|||
But if you need to change some of the defaults, support more than one predefined database, work with an async framework (like FastAPI), etc, you will need to add quite some complex extra code to override those defaults. |
|||
|
|||
Nevertheless, it's possible to do it, and here you'll see exactly what code you have to add to be able to use Peewee with FastAPI. |
|||
|
|||
!!! note "Technical Details" |
|||
You can read more about Peewee's stand about async in Python <a href="http://docs.peewee-orm.com/en/latest/peewee/database.html#async-with-gevent" target="_blank">in the docs</a>, <a href="https://github.com/coleifer/peewee/issues/263#issuecomment-517347032" target="_blank">an issue</a>, <a href="https://github.com/coleifer/peewee/pull/2072#issuecomment-563215132" target="_blank">a PR</a>. |
|||
|
|||
## The same app |
|||
|
|||
We are going to create the same application as in the <a href="https://fastapi.tiangolo.com/tutorial/sql-databases/" target="_blank">SQLAlchemy tutorial</a>. |
|||
|
|||
Most of the code is actually the same. |
|||
|
|||
So, we are going to focus only on the differences. |
|||
|
|||
## File structure |
|||
|
|||
Let's say you have a directory named `my_super_project` that contains a sub-directory called `sql_app` with a structure like this: |
|||
|
|||
``` |
|||
. |
|||
└── sql_app |
|||
├── __init__.py |
|||
├── crud.py |
|||
├── database.py |
|||
├── main.py |
|||
└── schemas.py |
|||
``` |
|||
|
|||
This is almost the same structure as we had for the SQLAlchemy tutorial. |
|||
|
|||
Now let's see what each file/module does. |
|||
|
|||
## Create the Peewee parts |
|||
|
|||
Let's refer to the file `sql_app/database.py`. |
|||
|
|||
### The standard Peewee code |
|||
|
|||
Let's first check all the normal Peewee code, create a Peewee database: |
|||
|
|||
```Python hl_lines="3 5 24" |
|||
{!./src/sql_databases_peewee/sql_app/database.py!} |
|||
``` |
|||
|
|||
!!! tip |
|||
Have in mind that if you wanted to use a different database, like PostgreSQL, you couldn't just change the string. You would need to use a different Peewee database class. |
|||
|
|||
#### Note |
|||
|
|||
The argument: |
|||
|
|||
```Python |
|||
check_same_thread=False |
|||
``` |
|||
|
|||
is equivalent to the one in the SQLAlchemy tutorial: |
|||
|
|||
```Python |
|||
connect_args={"check_same_thread": False} |
|||
``` |
|||
|
|||
...it is needed only for `SQLite`. |
|||
|
|||
!!! info "Technical Details" |
|||
|
|||
Exactly the same technical details as in the <a href="https://fastapi.tiangolo.com/tutorial/sql-databases/#note" target="_blank">SQLAlchemy tutorial</a> apply. |
|||
|
|||
### Make Peewee async-compatible `PeeweeConnectionState` |
|||
|
|||
The main issue with Peewee and FastAPI is that Peewee relies heavily on <a href="https://docs.python.org/3/library/threading.html#thread-local-data" target="_blank">Python's `threading.local`</a>, and it doesn't have a direct way to override it or let you handle connections/sessions directly (as is done in the SQLAlchemy tutorial). |
|||
|
|||
And `threading.local` is not compatible with the new async features of modern Python. |
|||
|
|||
!!! note "Technical Details" |
|||
`threading.local` is used to have a "magic" variable that has a different value for each thread. |
|||
|
|||
This was useful in older frameworks designed to have one single thread per request, no more, no less. |
|||
|
|||
Using this, each request would have its own database connection/session, which is the actual final goal. |
|||
|
|||
But FastAPI, using the new async features, could handle more than one request on the same thread. And at the same time, for a single request, it could run multiple things in different threads (in a threadpool), depending on if you use `async def` or normal `def`. This is what gives all the performance improvements to FastAPI. |
|||
|
|||
But Python 3.7 and above provide a more advanced alternative to `threading.local`, that can also be used in the places where `threading.local` would be used, but is compatible with the new async features. |
|||
|
|||
We are going to use that. It's called <a href="https://docs.python.org/3/library/contextvars.html" target="_blank">`contextvars`</a>. |
|||
|
|||
We are going to override the internal parts of Peewee that use `threading.local` and replace them with `contextvars`, with the corresponding updates. |
|||
|
|||
This might seem a bit complex (and it actually is), you don't really need to completely understand how it works to use it. |
|||
|
|||
We will create a `PeeweeConnectionState`: |
|||
|
|||
```Python hl_lines="8 9 10 11 12 13 14 15 16 17 18 19 20 21" |
|||
{!./src/sql_databases_peewee/sql_app/database.py!} |
|||
``` |
|||
|
|||
This class inherits from a special internal class used by Peewee. |
|||
|
|||
It has all the logic to make Peewee use `contextvars` instead of `threading.local`. |
|||
|
|||
`contextvars` works a bit differently than `threading.local`. But the rest of Peewee's internal code assumes that this class works with `threading.local`. |
|||
|
|||
So, we need to do some extra tricks to make it work as if it was just using `threading.local`. The `__init__`, `__setattr__`, and `__getattr__` implement all the required tricks for this to be used by Peewee without knowing that it is now compatible with FastAPI. |
|||
|
|||
!!! tip |
|||
This will just make Peewee behave correctly when used with FastAPI. Not randomly opening or closing connections that are being used, creating errors, etc. |
|||
|
|||
But it doesn't give Peewee async super-powers. You should still use normal `def` functions and not `async def`. |
|||
|
|||
### Use the custom `PeeweeConnectionState` class |
|||
|
|||
Now, overwrite the `._state` internal attribute in the Peewee database `db` object using the new `PeeweeConnectionState`: |
|||
|
|||
```Python hl_lines="26" |
|||
{!./src/sql_databases_peewee/sql_app/database.py!} |
|||
``` |
|||
|
|||
!!! tip |
|||
Make sure you overwrite `db._state` *after* creating `db`. |
|||
|
|||
!!! tip |
|||
You would do the same for any other Peewee database, including `PostgresqlDatabase`, `MySQLDatabase`, etc. |
|||
|
|||
## Create the database models |
|||
|
|||
Let's now see the file `sql_app/models.py`. |
|||
|
|||
### Create Peewee models for our data |
|||
|
|||
Now create the Peewee models (classes) for `User` and `Item`. |
|||
|
|||
This is the same you would do if you followed the Peewee tutorial and updated the models to have the same data as in the SQLAlchemy tutorial. |
|||
|
|||
!!! tip |
|||
Peewee also uses the term "**model**" to refer to these classes and instances that interact with the database. |
|||
|
|||
But Pydantic also uses the term "**model**" to refer to something different, the data validation, conversion, and documentation classes and instances. |
|||
|
|||
Import `db` from `database` (the file `database.py` from above) and use it here. |
|||
|
|||
```Python hl_lines="3 6 7 8 9 10 11 12 15 16 17 18 19 20 21" |
|||
{!./src/sql_databases_peewee/sql_app/models.py!} |
|||
``` |
|||
|
|||
!!! tip |
|||
Peewee creates several magic attributes. |
|||
|
|||
It will automatically add an `id` attribute as an integer to be the primary key. |
|||
|
|||
It will chose the name of the tables based on the class names. |
|||
|
|||
For the `Item`, it will create an attribute `owner_id` with the integer ID of the `User`. But we don't declare it anywhere. |
|||
|
|||
## Create the Pydantic models |
|||
|
|||
Now let's check the file `sql_app/schemas.py`. |
|||
|
|||
!!! tip |
|||
To avoid confusion between the Peewee *models* and the Pydantic *models*, we will have the file `models.py` with the Peewee models, and the file `schemas.py` with the Pydantic models. |
|||
|
|||
These Pydantic models define more or less a "schema" (a valid data shape). |
|||
|
|||
So this will help us avoiding confusion while using both. |
|||
|
|||
### Create the Pydantic *models* / schemas |
|||
|
|||
Create all the same Pydantic models as in the SQLAlchemy tutorial: |
|||
|
|||
```Python hl_lines="16 17 18 21 22 25 26 27 28 29 30 34 35 38 39 42 43 44 45 46 47 48" |
|||
{!./src/sql_databases_peewee/sql_app/schemas.py!} |
|||
``` |
|||
|
|||
!!! tip |
|||
Here we are creating the models with an `id`. |
|||
|
|||
We didn't explicitly specify an `id` attribute in the Peewee models, but Peewee adds one automatically. |
|||
|
|||
We are also adding the magic `owner_id` attribute to `Item`. |
|||
|
|||
### Create a `PeeweeGetterDict` for the Pydantic *models* / schemas |
|||
|
|||
When you access a relationship in a Peewee object, like in `some_user.items`, Peewee doesn't provide a `list` of `Item`. |
|||
|
|||
It provides a special custom object of class `ModelSelect`. |
|||
|
|||
It's possible to create a `list` of its items with `list(some_user.items)`. |
|||
|
|||
But the object itself is not a `list`. And it's also not an actual Python <a href="https://docs.python.org/3/glossary.html#term-generator" target="_blank">generator</a>. Because of this, Pydantic doesn't know by default how to convert it to a `list` of Pydantic *models* / schemas. |
|||
|
|||
But recent versions of Pydantic allow providing a custom class that inherits from `pydantic.utils.GetterDict`, to provide the functionality used when using the `orm_mode = True` to retrieve the values for ORM model attributes. |
|||
|
|||
We are going to create a custom `PeeweeGetterDict` class and use it in all the same Pydantic *models* / schemas that use `orm_mode`: |
|||
|
|||
```Python hl_lines="3 8 9 10 11 12 13 31 49" |
|||
{!./src/sql_databases_peewee/sql_app/schemas.py!} |
|||
``` |
|||
|
|||
Here we are checking if the attribute that is being accessed (e.g. `.items` in `some_user.items`) is an instance of `peewee.ModelSelect`. |
|||
|
|||
And if that's the case, just return a `list` with it. |
|||
|
|||
And then we use it in the Pydantic *models* / schemas that use `orm_mode = True`, with the configuration variable `getter_dict = PeeweeGetterDict`. |
|||
|
|||
!!! tip |
|||
We only need to create one `PeeweeGetterDict` class, and we can use it in all the Pydantic *models* / schemas. |
|||
|
|||
## CRUD utils |
|||
|
|||
Now let's see the file `sql_app/crud.py`. |
|||
|
|||
### Create all the CRUD utils |
|||
|
|||
Create all the same CRUD utils as in the SQLAlchemy tutorial, all the code is very similar: |
|||
|
|||
```Python hl_lines="1 4 5 8 9 12 13 16 17 18 19 20 23 24 27 28 29 30" |
|||
{!./src/sql_databases_peewee/sql_app/crud.py!} |
|||
``` |
|||
|
|||
There are some differences with the code for the SQLAlchemy tutorial. |
|||
|
|||
We don't pass a `db` attribute around. Instead we use the models directly. This is because the `db` object is a global object, that includes all the connection logic. That's why we had to do all the `contextvars` updates above. |
|||
|
|||
Aso, when returning several objects, like in `get_users`, we directly call `list`, like in: |
|||
|
|||
```Python |
|||
list(models.User.select()) |
|||
``` |
|||
|
|||
This is for the same reason that we had to create a custom `PeeweeGetterDict`. But by returning something that is already a `list` instead of the `peewee.ModelSelect` the `response_model` in the path operation with `List[models.User]` (that we'll see later) will work correctly. |
|||
|
|||
## Main **FastAPI** app |
|||
|
|||
And now in the file `sql_app/main.py` let's integrate and use all the other parts we created before. |
|||
|
|||
### Create the database tables |
|||
|
|||
In a very simplistic way create the database tables: |
|||
|
|||
```Python hl_lines="8 9 10" |
|||
{!./src/sql_databases_peewee/sql_app/main.py!} |
|||
``` |
|||
|
|||
### Create a dependency |
|||
|
|||
Create a dependency that will connect the database right at the beginning of a request and disconnect it at the end: |
|||
|
|||
```Python hl_lines="15 16 17 18 19 20 21 22" |
|||
{!./src/sql_databases_peewee/sql_app/main.py!} |
|||
``` |
|||
|
|||
Here we have an empty `yield` because we are actually not using the database object directly. |
|||
|
|||
It is connecting to the database and storing the connection data in an internal variable that is independent for each request (using the `contextvars` tricks from above). |
|||
|
|||
And then, in each *path operation function* that needs to access the database we add it as a dependency. |
|||
|
|||
But we are not using the value given by this dependency (it actually doesn't give any value, as it has an empty `yield`). So, we don't add it to the *path operation function* but to the *path operation decorator* in the `dependencies` parameter: |
|||
|
|||
```Python hl_lines="25 33 40 52 58 65" |
|||
{!./src/sql_databases_peewee/sql_app/main.py!} |
|||
``` |
|||
|
|||
### Create your **FastAPI** *path operations* |
|||
|
|||
Now, finally, here's the standard **FastAPI** *path operations* code. |
|||
|
|||
```Python hl_lines="25 26 27 28 29 30 33 34 35 36 39 40 41 42 43 44 45 46 49 50 51 52 53 54 55 58 59 60 61 64 65 66 67 68 69 70" |
|||
{!./src/sql_databases_peewee/sql_app/main.py!} |
|||
``` |
|||
|
|||
### About `def` vs `async def` |
|||
|
|||
The same as with SQLAlchemy, we are not doing something like: |
|||
|
|||
```Python |
|||
user = await models.User.select().first() |
|||
``` |
|||
|
|||
...but instead we are using: |
|||
|
|||
```Python |
|||
user = models.User.select().first() |
|||
``` |
|||
|
|||
So, again, we should declare the *path operation functions* and the dependency without `async def`, just with a normal `def`, as: |
|||
|
|||
```Python hl_lines="2" |
|||
# Something goes here |
|||
def read_users(skip: int = 0, limit: int = 100): |
|||
# Something goes here |
|||
``` |
|||
|
|||
## Testing Peewee with async |
|||
|
|||
This example includes an extra *path operation* that simulates a long processing request with `time.sleep(15)`. |
|||
|
|||
It will have the database connection open at the beginning and will just wait 15 seconds before replying back. |
|||
|
|||
This will easily let you test that your app with Peewee and FastAPI is behaving correctly with all the stuff about threads. |
|||
|
|||
If you want to check how Peewee would break your app if used without modification, go the the `sql_app/database.py` file and comment the line: |
|||
|
|||
```Python |
|||
# db._state = PeeweeConnectionState() |
|||
``` |
|||
|
|||
Then run your app with Uvicorn: |
|||
|
|||
```bash |
|||
uvicorn sql_app.main:app --reload |
|||
``` |
|||
|
|||
Open your browser at <a href="http://127.0.0.1:8000/docs" target="_blank">http://127.0.0.1:8000/docs</a> and create a couple of users. |
|||
|
|||
Then open 10 tabs at <a href="http://127.0.0.1:8000/docs#/default/read_slow_users_slowusers__get" target="_blank">http://127.0.0.1:8000/docs#/default/read_slow_users_slowusers__get</a> at the same time. |
|||
|
|||
Go to the *path operation* "Get `/slowusers/`" in all of the tabs. Use the "Try it out" button and execute the request in each tab, one right after the other. |
|||
|
|||
The tabs will wait for a bit and then some of them will show `Internal Server Error`. |
|||
|
|||
### What happens |
|||
|
|||
The first tab will make your app create a connection to the database and wait for 15 seconds before replying back and closing the connection. |
|||
|
|||
Then one of the other tabs will try to open a database connection, but as one of those requests for the other tabs will probably be handled in the same thread as the first one, it will have the same database connection that is already open, and Peewee will throw an error and you will see it in the terminal, and the response will have an `Internal Server Error`. |
|||
|
|||
This will probably happen for more than one of those tabs. |
|||
|
|||
If you had multiple clients talking to your app exactly at the same time, this is what could happen. |
|||
|
|||
And as your app starts to handle more and more clients at the same time, the waiting time in a single requests needs to be shorter and shorter to trigger the error. |
|||
|
|||
### Fix Peewee with FastAPI |
|||
|
|||
Now go back to the file `sql_app/database.py`, and uncomment the line: |
|||
|
|||
```Python |
|||
db._state = PeeweeConnectionState() |
|||
``` |
|||
|
|||
Terminate your running app and start it again. |
|||
|
|||
Repeat the same process with the 10 tabs. This time all of them will wait and you will get all the results without errors. |
|||
|
|||
...You fixed it! |
|||
|
|||
## Review all the files |
|||
|
|||
Remember you should have a directory named `my_super_project` that contains a sub-directory called `sql_app`. |
|||
|
|||
`sql_app` should have the following files: |
|||
|
|||
* `sql_app/__init__.py`: is an empty file. |
|||
|
|||
* `sql_app/database.py`: |
|||
|
|||
```Python hl_lines="" |
|||
{!./src/sql_databases_peewee/sql_app/database.py!} |
|||
``` |
|||
|
|||
* `sql_app/models.py`: |
|||
|
|||
```Python hl_lines="" |
|||
{!./src/sql_databases_peewee/sql_app/models.py!} |
|||
``` |
|||
|
|||
* `sql_app/schemas.py`: |
|||
|
|||
```Python hl_lines="" |
|||
{!./src/sql_databases_peewee/sql_app/schemas.py!} |
|||
``` |
|||
|
|||
* `sql_app/crud.py`: |
|||
|
|||
```Python hl_lines="" |
|||
{!./src/sql_databases_peewee/sql_app/crud.py!} |
|||
``` |
|||
|
|||
* `sql_app/main.py`: |
|||
|
|||
```Python hl_lines="" |
|||
{!./src/sql_databases_peewee/sql_app/main.py!} |
|||
``` |
|||
|
|||
## Technical Details |
|||
|
|||
If you want to go deeper into the technical details related to Peewee with FastAPI, you can <a href="https://github.com/coleifer/peewee/pull/2072" target="_blank">read more about it here</a>. |
@ -0,0 +1,430 @@ |
|||
import time |
|||
from pathlib import Path |
|||
from unittest.mock import MagicMock |
|||
|
|||
import pytest |
|||
from starlette.testclient import TestClient |
|||
|
|||
from ...utils import skip_py36 |
|||
|
|||
openapi_schema = { |
|||
"openapi": "3.0.2", |
|||
"info": {"title": "Fast API", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/users/": { |
|||
"get": { |
|||
"summary": "Read Users", |
|||
"operationId": "read_users_users__get", |
|||
"parameters": [ |
|||
{ |
|||
"required": False, |
|||
"schema": {"title": "Skip", "type": "integer", "default": 0}, |
|||
"name": "skip", |
|||
"in": "query", |
|||
}, |
|||
{ |
|||
"required": False, |
|||
"schema": {"title": "Limit", "type": "integer", "default": 100}, |
|||
"name": "limit", |
|||
"in": "query", |
|||
}, |
|||
], |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"title": "Response Read Users Users Get", |
|||
"type": "array", |
|||
"items": {"$ref": "#/components/schemas/User"}, |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create User", |
|||
"operationId": "create_user_users__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/UserCreate"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/User"} |
|||
} |
|||
}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
"/users/{user_id}": { |
|||
"get": { |
|||
"summary": "Read User", |
|||
"operationId": "read_user_users__user_id__get", |
|||
"parameters": [ |
|||
{ |
|||
"required": True, |
|||
"schema": {"title": "User Id", "type": "integer"}, |
|||
"name": "user_id", |
|||
"in": "path", |
|||
} |
|||
], |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/User"} |
|||
} |
|||
}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"/users/{user_id}/items/": { |
|||
"post": { |
|||
"summary": "Create Item For User", |
|||
"operationId": "create_item_for_user_users__user_id__items__post", |
|||
"parameters": [ |
|||
{ |
|||
"required": True, |
|||
"schema": {"title": "User Id", "type": "integer"}, |
|||
"name": "user_id", |
|||
"in": "path", |
|||
} |
|||
], |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/ItemCreate"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item"} |
|||
} |
|||
}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"parameters": [ |
|||
{ |
|||
"required": False, |
|||
"schema": {"title": "Skip", "type": "integer", "default": 0}, |
|||
"name": "skip", |
|||
"in": "query", |
|||
}, |
|||
{ |
|||
"required": False, |
|||
"schema": {"title": "Limit", "type": "integer", "default": 100}, |
|||
"name": "limit", |
|||
"in": "query", |
|||
}, |
|||
], |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"title": "Response Read Items Items Get", |
|||
"type": "array", |
|||
"items": {"$ref": "#/components/schemas/Item"}, |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"/slowusers/": { |
|||
"get": { |
|||
"summary": "Read Slow Users", |
|||
"operationId": "read_slow_users_slowusers__get", |
|||
"parameters": [ |
|||
{ |
|||
"required": False, |
|||
"schema": {"title": "Skip", "type": "integer", "default": 0}, |
|||
"name": "skip", |
|||
"in": "query", |
|||
}, |
|||
{ |
|||
"required": False, |
|||
"schema": {"title": "Limit", "type": "integer", "default": 100}, |
|||
"name": "limit", |
|||
"in": "query", |
|||
}, |
|||
], |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"title": "Response Read Slow Users Slowusers Get", |
|||
"type": "array", |
|||
"items": {"$ref": "#/components/schemas/User"}, |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"title": "HTTPValidationError", |
|||
"type": "object", |
|||
"properties": { |
|||
"detail": { |
|||
"title": "Detail", |
|||
"type": "array", |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
} |
|||
}, |
|||
}, |
|||
"Item": { |
|||
"title": "Item", |
|||
"required": ["title", "id", "owner_id"], |
|||
"type": "object", |
|||
"properties": { |
|||
"title": {"title": "Title", "type": "string"}, |
|||
"description": {"title": "Description", "type": "string"}, |
|||
"id": {"title": "Id", "type": "integer"}, |
|||
"owner_id": {"title": "Owner Id", "type": "integer"}, |
|||
}, |
|||
}, |
|||
"ItemCreate": { |
|||
"title": "ItemCreate", |
|||
"required": ["title"], |
|||
"type": "object", |
|||
"properties": { |
|||
"title": {"title": "Title", "type": "string"}, |
|||
"description": {"title": "Description", "type": "string"}, |
|||
}, |
|||
}, |
|||
"User": { |
|||
"title": "User", |
|||
"required": ["email", "id", "is_active"], |
|||
"type": "object", |
|||
"properties": { |
|||
"email": {"title": "Email", "type": "string"}, |
|||
"id": {"title": "Id", "type": "integer"}, |
|||
"is_active": {"title": "Is Active", "type": "boolean"}, |
|||
"items": { |
|||
"title": "Items", |
|||
"type": "array", |
|||
"items": {"$ref": "#/components/schemas/Item"}, |
|||
"default": [], |
|||
}, |
|||
}, |
|||
}, |
|||
"UserCreate": { |
|||
"title": "UserCreate", |
|||
"required": ["email", "password"], |
|||
"type": "object", |
|||
"properties": { |
|||
"email": {"title": "Email", "type": "string"}, |
|||
"password": {"title": "Password", "type": "string"}, |
|||
}, |
|||
}, |
|||
"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"}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
@pytest.fixture(scope="module") |
|||
def client(): |
|||
# Import while creating the client to create the DB after starting the test session |
|||
from sql_databases_peewee.sql_app.main import app |
|||
|
|||
test_db = Path("./test.db") |
|||
with TestClient(app) as c: |
|||
yield c |
|||
test_db.unlink() |
|||
|
|||
|
|||
@skip_py36 |
|||
def test_openapi_schema(client): |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200 |
|||
assert response.json() == openapi_schema |
|||
|
|||
|
|||
@skip_py36 |
|||
def test_create_user(client): |
|||
test_user = {"email": "[email protected]", "password": "secret"} |
|||
response = client.post("/users/", json=test_user) |
|||
assert response.status_code == 200 |
|||
data = response.json() |
|||
assert test_user["email"] == data["email"] |
|||
assert "id" in data |
|||
response = client.post("/users/", json=test_user) |
|||
assert response.status_code == 400 |
|||
|
|||
|
|||
@skip_py36 |
|||
def test_get_user(client): |
|||
response = client.get("/users/1") |
|||
assert response.status_code == 200 |
|||
data = response.json() |
|||
assert "email" in data |
|||
assert "id" in data |
|||
|
|||
|
|||
@skip_py36 |
|||
def test_inexistent_user(client): |
|||
response = client.get("/users/999") |
|||
assert response.status_code == 404 |
|||
|
|||
|
|||
@skip_py36 |
|||
def test_get_users(client): |
|||
response = client.get("/users/") |
|||
assert response.status_code == 200 |
|||
data = response.json() |
|||
assert "email" in data[0] |
|||
assert "id" in data[0] |
|||
|
|||
|
|||
time.sleep = MagicMock() |
|||
|
|||
|
|||
@skip_py36 |
|||
def test_get_slowusers(client): |
|||
response = client.get("/slowusers/") |
|||
assert response.status_code == 200 |
|||
data = response.json() |
|||
assert "email" in data[0] |
|||
assert "id" in data[0] |
|||
|
|||
|
|||
@skip_py36 |
|||
def test_create_item(client): |
|||
item = {"title": "Foo", "description": "Something that fights"} |
|||
response = client.post("/users/1/items/", json=item) |
|||
assert response.status_code == 200 |
|||
item_data = response.json() |
|||
assert item["title"] == item_data["title"] |
|||
assert item["description"] == item_data["description"] |
|||
assert "id" in item_data |
|||
assert "owner_id" in item_data |
|||
response = client.get("/users/1") |
|||
assert response.status_code == 200 |
|||
user_data = response.json() |
|||
item_to_check = [it for it in user_data["items"] if it["id"] == item_data["id"]][0] |
|||
assert item_to_check["title"] == item["title"] |
|||
assert item_to_check["description"] == item["description"] |
|||
response = client.get("/users/1") |
|||
assert response.status_code == 200 |
|||
user_data = response.json() |
|||
item_to_check = [it for it in user_data["items"] if it["id"] == item_data["id"]][0] |
|||
assert item_to_check["title"] == item["title"] |
|||
assert item_to_check["description"] == item["description"] |
|||
|
|||
|
|||
@skip_py36 |
|||
def test_read_items(client): |
|||
response = client.get("/items/") |
|||
assert response.status_code == 200 |
|||
data = response.json() |
|||
assert data |
|||
first_item = data[0] |
|||
assert "title" in first_item |
|||
assert "description" in first_item |
@ -0,0 +1,5 @@ |
|||
import sys |
|||
|
|||
import pytest |
|||
|
|||
skip_py36 = pytest.mark.skipif(sys.version_info < (3, 7), reason="skip python3.6") |
Loading…
Reference in new issue