Schema basics
GraphQL servers use a schema to describe the shape of the data. The schema defines a hierarchy of types with fields that are populated from data stores. The schema also specifies exactly which queries and mutations are available for clients to execute.
This guide describes the basic building blocks of a schema and how to use Strawberry to create one.
Schema definition language (SDL)
There are two approaches for creating the schema for a GraphQL server. One is called “schema-first” and the other is called “code-first”. Strawberry only supports code-first schemas. Before diving into code-first, let’s first explain what the Schema definition language is.
Schema first works using the Schema Definition Language of GraphQL, which is included in the GraphQL spec.
Here’s an example of schema defined using the SDL:
type Book { title: String! author: Author!}
type Author { name: String! books: [Book!]!}
The schema defines all the types and relationships between them. With this we enable client developers to see exactly what data is available and request a specific subset of that data.
The !
sign specifies that a field is non-nullable.
Notice that the schema doesn’t specify how to get the data. That comes later when defining the resolvers.
Code first approach
As mentioned Strawberry uses a code first approach. The previous schema would look like this in Strawberry
import typingimport strawberry
@strawberry.typeclass Book: title: str author: "Author"
@strawberry.typeclass Author: name: str books: typing.List["Book"]
As you can see the code maps almost one to one with the schema, thanks to python’s type hints feature.
Notice that here we are also not specifying how to fetch data, that will be explained in the resolvers section.
Supported types
GraphQL supports a few different types:
- Scalar types
- Object types
- The Query type
- The Mutation type
- Input types
Scalar types
Scalar types are similar to Python primitive types. Here’s the list of the default scalar types in GraphQL:
- Int, a signed 32-bit integer, maps to python’s int
- Float, a signed double-precision floating-point value, maps to python’s float
- String, maps to python’s str
- Boolean, true or false, maps to python’s bool
- ID, a unique identifier that usually used to refetch an object or as the key
for a cache. Serialized as string and available as
strawberry.ID(“value”)
UUID
, a UUID value serialized as a string
Strawberry also includes support for date, time and datetime objects, they are not officially included with the GraphQL spec, but they are usually needed in most servers. They are serialized as ISO-8601.
These primitives work for the majority of use cases, but you can also specify your own scalar types.
Object types
Most of the types you define in a GraphQL schema are object types. An object type contains a collection of fields, each of which can be either a scalar type or another object type.
Object types can refer to each other, as we had in our schema earlier:
import typingimport strawberry
@strawberry.typeclass Book: title: str author: "Author"
@strawberry.typeclass Author: name: str books: typing.List[Book]
Providing data to fields
In the above schema, a Book
has an author
field and an Author
has a books
field, yet we do not know how our data can be mapped to fulfil the structure of
the promised schema.
To achieve this, we introduce the concept of the resolver that provides some data to a field through a function.
Continuing with this example of books and authors, resolvers can be defined to provides values to the fields:
def get_author_for_book(root) -> "Author": return Author(name="Michael Crichton")
@strawberry.typeclass Book: title: str author: "Author" = strawberry.field(resolver=get_author_for_book)
def get_books_for_author(root): return [Book(title="Jurassic Park")]
@strawberry.typeclass Author: name: str books: typing.List[Book] = strawberry.field(resolver=get_books_for_author)
def get_authors(root) -> typing.List[Author]: return [Author(name="Michael Crichton")]
@strawberry.typeclass Query: authors: typing.List[Author] = strawberry.field(resolver=get_authors) books: typing.List[Book] = strawberry.field(resolver=get_books_for_author)
These functions provide the strawberry.field
with the ability to render data
to the GraphQL query upon request and are the backbone of all GraphQL APIs.
This example is trivial since the resolved data is entirely static. However, when building more complex APIs, these resolvers can be written to map data from databases, e.g. making SQL queries using SQLAlchemy, and other APIs, e.g. making HTTP requests using aiohttp.
For more information and detail on the different ways to write resolvers, see the resolvers section.
The Query type
The Query
type defines exactly which GraphQL queries (i.e., read operations)
clients can execute against your data. It resembles an object type, but its name
is always Query
.
Each field of the Query
type defines the name and return type of a different
supported query. The Query
type for our example schema might resemble the
following:
@strawberry.typeclass Query: books: typing.List[Book] authors: typing.List[Author]
This Query type defines two available queries: books and authors. Each query returns a list of the corresponding type.
With a REST-based API, books and authors would probably be returned by different endpoints (e.g., /api/books and /api/authors). The flexibility of GraphQL enables clients to query both resources with a single request.
Structuring a query
When your clients build queries to execute against your data graph, those queries match the shape of the object types you define in your schema.
Based on our example schema so far, a client could execute the following query, which requests both a list of all book titles and a list of all author names:
query { books { title }
authors { name }}
Our server would then respond to the query with results that match the query's structure, like so:
{ "data": { "books": [{ "title": "Jurassic Park" }], "authors": [{ "name": "Michael Crichton" }] }}
Although it might be useful in some cases to fetch these two separate lists, a client would probably prefer to fetch a single list of books, where each book's author is included in the result.
Because our schema's Book type has an author field of type Author, a client could structure their query like so:
query { books { title author { name } }}
And once again, our server would respond with results that match the query's structure:
{ "data": { "books": [ { "title": "Jurassic Park", "author": { "name": "Michael Crichton" } } ] }}
The Mutation type
The Mutation
type is similar in structure and purpose to the Query type.
Whereas the Query type defines your data's supported read operations, the
Mutation
type defines supported write operations.
Each field of the Mutation
type defines the signature and return type of a
different mutation. The Mutation
type for our example schema might resemble
the following:
@strawberry.typeclass Mutation: @strawberry.field def add_book(self, title: str, author: str) -> Book: ...
This Mutation type defines a single available mutation, addBook
. The mutation
accepts two arguments (title and author) and returns a newly created Book
object. As you'd expect, this Book object conforms to the structure that we
defined in our schema.
Strawberry converts fields names from snake case to camel case automatically.
Structuring a mutation
Like queries, mutations match the structure of your schema's type definitions. The following mutation creates a new Book and requests certain fields of the created object as a return value:
mutation { addBook(title: "Fox in Socks", author: "Dr. Seuss") { title author { name } }}
As with queries, our server would respond to this mutation with a result that matches the mutation's structure, like so:
{ "data": { "addBook": { "title": "Fox in Socks", "author": { "name": "Dr. Seuss" } } }}
Input types
Input types are special object types that allow you to pass objects as arguments to queries and mutations (as opposed to passing only scalar types). Input types help keep operation signatures clean.
Consider our previous mutation to add a book:
@strawberry.typeclass Mutation: @strawberry.field def add_book(self, title: str, author: str) -> Book: ...
Instead of accepting two arguments, this mutation could accept a single input type that includes all of these fields. This comes in extra handy if we decide to accept an additional argument in the future, such as a publication date.
An input type's definition is similar to an object type's, but it uses the input keyword:
@strawberry.inputclass AddBookInput: title: str author: str
@strawberry.typeclass Mutation: @strawberry.field def add_book(self, book: AddBookInput) -> Book: ...
Not only does this facilitate passing the AddBookInput type around within our schema, it also provides a basis for annotating fields with descriptions that are automatically exposed by GraphQL-enabled tools:
@strawberry.inputclass AddBookInput: title: str = strawberry.field(description="The title of the book") author: str = strawberry.field(description="The name of the author")
Input types can sometimes be useful when multiple operations require the exact same set of information, but you should reuse them sparingly. Operations might eventually diverge in their sets of required arguments.
More
If you want to learn more about schema design make sure you follow the documentation provided by Apollo.