2018-10-18 13:45:18 +00:00
|
|
|
|
2021-02-05 19:13:39 +00:00
|
|
|
!!! Warning
|
|
|
|
|
|
|
|
GraphQL support in Starlette is **deprecated** as of version 0.15 and will
|
2021-06-19 17:02:53 +00:00
|
|
|
be removed in a future release. It is also incompatible with Python 3.10+.
|
|
|
|
Please consider using a third-party library to provide GraphQL support. This
|
|
|
|
is usually done by mounting a GraphQL ASGI application.
|
|
|
|
See [#619](https://github.com/encode/starlette/issues/619).
|
2021-02-05 19:13:39 +00:00
|
|
|
Some example libraries are:
|
|
|
|
|
|
|
|
* [Ariadne](https://ariadnegraphql.org/docs/asgi)
|
|
|
|
* [`tartiflette-asgi`](https://tartiflette.github.io/tartiflette-asgi/)
|
|
|
|
* [Strawberry](https://strawberry.rocks/docs/integrations/asgi)
|
|
|
|
* [`starlette-graphene3`](https://github.com/ciscorn/starlette-graphene3)
|
|
|
|
|
2018-10-18 13:45:18 +00:00
|
|
|
Starlette includes optional support for GraphQL, using the `graphene` library.
|
|
|
|
|
|
|
|
Here's an example of integrating the support into your application.
|
|
|
|
|
|
|
|
```python
|
|
|
|
from starlette.applications import Starlette
|
2019-11-13 12:25:18 +00:00
|
|
|
from starlette.routing import Route
|
2018-11-01 12:52:16 +00:00
|
|
|
from starlette.graphql import GraphQLApp
|
2018-10-18 13:45:18 +00:00
|
|
|
import graphene
|
|
|
|
|
|
|
|
|
|
|
|
class Query(graphene.ObjectType):
|
|
|
|
hello = graphene.String(name=graphene.String(default_value="stranger"))
|
|
|
|
|
|
|
|
def resolve_hello(self, info, name):
|
|
|
|
return "Hello " + name
|
|
|
|
|
2019-11-13 12:25:18 +00:00
|
|
|
routes = [
|
|
|
|
Route('/', GraphQLApp(schema=graphene.Schema(query=Query)))
|
|
|
|
]
|
2018-10-18 13:45:18 +00:00
|
|
|
|
2020-02-04 14:14:53 +00:00
|
|
|
app = Starlette(routes=routes)
|
2018-10-18 13:45:18 +00:00
|
|
|
```
|
|
|
|
|
2018-10-18 15:58:49 +00:00
|
|
|
If you load up the page in a browser, you'll be served the GraphiQL tool,
|
|
|
|
which you can use to interact with your GraphQL API.
|
|
|
|
|
2019-02-19 13:14:42 +00:00
|
|
|
|
2018-10-18 15:58:49 +00:00
|
|
|
![GraphiQL](img/graphiql.png)
|
|
|
|
|
2018-12-04 16:45:44 +00:00
|
|
|
## Accessing request information
|
|
|
|
|
|
|
|
The current request is available in the context.
|
|
|
|
|
|
|
|
```python
|
|
|
|
class Query(graphene.ObjectType):
|
|
|
|
user_agent = graphene.String()
|
|
|
|
|
|
|
|
def resolve_user_agent(self, info):
|
|
|
|
"""
|
|
|
|
Return the User-Agent of the incoming request.
|
|
|
|
"""
|
|
|
|
request = info.context["request"]
|
|
|
|
return request.headers.get("User-Agent", "<unknown>")
|
|
|
|
```
|
|
|
|
|
2018-12-14 14:56:31 +00:00
|
|
|
## Adding background tasks
|
|
|
|
|
|
|
|
You can add background tasks to run once the response has been sent.
|
|
|
|
|
|
|
|
```python
|
|
|
|
class Query(graphene.ObjectType):
|
|
|
|
user_agent = graphene.String()
|
|
|
|
|
|
|
|
def resolve_user_agent(self, info):
|
|
|
|
"""
|
|
|
|
Return the User-Agent of the incoming request.
|
|
|
|
"""
|
|
|
|
user_agent = request.headers.get("User-Agent", "<unknown>")
|
|
|
|
background = info.context["background"]
|
|
|
|
background.add_task(log_user_agent, user_agent=user_agent)
|
|
|
|
return user_agent
|
|
|
|
|
|
|
|
async def log_user_agent(user_agent):
|
|
|
|
...
|
|
|
|
```
|
|
|
|
|
2018-10-18 13:45:18 +00:00
|
|
|
## Sync or Async executors
|
|
|
|
|
|
|
|
If you're working with a standard ORM, then just use regular function calls for
|
|
|
|
your "resolve" methods, and Starlette will manage running the GraphQL query within a
|
2021-02-02 11:13:59 +00:00
|
|
|
separate thread.
|
2018-10-18 13:45:18 +00:00
|
|
|
|
2021-02-02 11:13:59 +00:00
|
|
|
If you want to use an asynchronous ORM, then use "async resolve" methods, and
|
2018-10-18 15:58:49 +00:00
|
|
|
make sure to setup Graphene's AsyncioExecutor using the `executor` argument.
|
2018-10-18 13:45:18 +00:00
|
|
|
|
|
|
|
```python
|
|
|
|
from graphql.execution.executors.asyncio import AsyncioExecutor
|
|
|
|
from starlette.applications import Starlette
|
2019-05-12 09:14:14 +00:00
|
|
|
from starlette.graphql import GraphQLApp
|
2019-11-13 12:25:18 +00:00
|
|
|
from starlette.routing import Route
|
2018-10-18 13:45:18 +00:00
|
|
|
import graphene
|
|
|
|
|
|
|
|
|
|
|
|
class Query(graphene.ObjectType):
|
|
|
|
hello = graphene.String(name=graphene.String(default_value="stranger"))
|
|
|
|
|
|
|
|
async def resolve_hello(self, info, name):
|
|
|
|
# We can make asynchronous network calls here.
|
|
|
|
return "Hello " + name
|
|
|
|
|
2019-11-13 12:25:18 +00:00
|
|
|
routes = [
|
|
|
|
# We're using `executor_class=AsyncioExecutor` here.
|
|
|
|
Route('/', GraphQLApp(
|
|
|
|
schema=graphene.Schema(query=Query),
|
|
|
|
executor_class=AsyncioExecutor
|
|
|
|
))
|
|
|
|
]
|
2018-10-18 13:45:18 +00:00
|
|
|
|
2019-11-13 12:25:18 +00:00
|
|
|
app = Starlette(routes=routes)
|
2018-10-18 13:45:18 +00:00
|
|
|
```
|