FastAPI example
Project Setup
This project folder structure is like the below.
modelresolvers-fastapi-example
├── main.py
├── models
│ └── user.py
├── Pipfile
├── Pipfile.lock
└── resolvers
└── user.py
modelresolvers
requires Python3.10
we create virtual env by specify Python version
Install dependencies
Activate virtual environment
Define User model
Create user.py
under models
folder
Import strawberry
module to define GraphQL type
Create User model
Since the model is a strawberry GraphQL type it must be decorated with strawberry.type
decorator.
id
field is optional because we want to set it automatically instead of getting it from the client.
The entire file code
Resolvers
Create user.py
under resolvers
folder
Import modelresolvers
Import the User model that we just created
Create ModelResolvers instance for user model resolvers
Since we are simulating a real-world example we just define a user list
users = [
User(id=1, firstname="John", lastname="Doe"),
User(id=2, firstname="Jane", lastname="Doe"),
]
Querying a single user
@user_resolvers.query(name="user")
def user(id: int) -> User | None:
return next((user for user in users if id == user.id), None)
To make a function a query we decorate it with the query decorator of ModelResolvers instance.
The name of the query decorator parameter is optional. The default is the function name that
is decorated which in this case is user. The function parameters and return parameter type
should be specified by type hints otherwise the strawberry raises error. The user function
searches the user based on the id it is not found it then returns None
get_users
query returns all users available in the users
list. Pay close attention to the function name and the name
argument of the query decorator. They are different. get_users
query will be displayed as users
in the GraphQL client UI.
Another ModelResolvers decorator is mutation.
@user_resolvers.mutation(name="add_user")
def add_user(firstname: str, lastname: str) -> User:
user = User(id=len(users) + 1, firstname=firstname, lastname=lastname)
users.append(user)
return user
firstname and lastname pass by GraphQL client. The id field is set on the backend side. It is calculated as users list length plus 1. Then the created user returns to the client.
The entire file code
from modelresolvers import ModelResolvers
from models.user import User
user_resolvers = ModelResolvers()
users = [
User(id=1, firstname="John", lastname="Doe"),
User(id=2, firstname="Jane", lastname="Doe"),
]
@user_resolvers.query(name="user")
def user(id: int) -> User | None:
return next((user for user in users if id == user.id), None)
@user_resolvers.query(name="users")
def get_users() -> list[User]:
return users
@user_resolvers.mutation(name="add_user")
def add_user(firstname: str, lastname: str) -> User:
user = User(id=len(users) + 1, firstname=firstname, lastname=lastname)
users.append(user)
return user
Main file
Create main.py
file
Import Schema module from modelresolvers package
Import FastAPI module from fastapi package
Import GraphQLRouter module from strawberry package
Import resolvers instance you have just created
When you create an modelresolvers Schema it will also create a strawberry schema as modelresolvers is based on strawberry At this point, we need to strawberry schema rather than modelresolvers Schema. We get strawberry schema from modelresolvers schema
Create GraphQL router
Create fastapi app and add GraphQL router to the app
The entire file code
from modelresolvers import Schema
from fastapi import FastAPI
from strawberry.fastapi import GraphQLRouter
from resolvers.user import user_resolvers
schema = Schema(models_resolvers=[user_resolvers]).strawberry_schema
graphql_app = GraphQLRouter(schema)
app = FastAPI()
app.include_router(graphql_app, prefix="/graphql")
Run the application
Go to http://127.0.0.1:8000/graphql
and run queries and mutations