Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/taskiq-python/aiohttp-deps
Dependency injection for AioHTTP
https://github.com/taskiq-python/aiohttp-deps
aiohttp aiohttp-server dependency-injection fastapi openapi3 swagger
Last synced: 2 months ago
JSON representation
Dependency injection for AioHTTP
- Host: GitHub
- URL: https://github.com/taskiq-python/aiohttp-deps
- Owner: taskiq-python
- License: mit
- Created: 2023-04-21T00:08:54.000Z (over 1 year ago)
- Default Branch: master
- Last Pushed: 2024-10-01T08:13:19.000Z (3 months ago)
- Last Synced: 2024-10-29T14:22:14.161Z (2 months ago)
- Topics: aiohttp, aiohttp-server, dependency-injection, fastapi, openapi3, swagger
- Language: Python
- Homepage:
- Size: 297 KB
- Stars: 9
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/aiohttp-deps?style=for-the-badge)](https://pypi.org/project/aiohttp-deps/)
[![PyPI](https://img.shields.io/pypi/v/aiohttp-deps?style=for-the-badge)](https://pypi.org/project/aiohttp-deps/)
[![PyPI - Downloads](https://img.shields.io/pypi/dm/aiohttp-deps?style=for-the-badge)](https://pypistats.org/packages/aiohttp-deps)# AioHTTP deps
This project was initially created to show the abillities of [taskiq-dependencies](https://github.com/taskiq-python/taskiq-dependencies) project, which is used by [taskiq](https://github.com/taskiq-python/taskiq) to provide you with the best experience of sending distributed tasks.
This project adds [FastAPI](https://github.com/tiangolo/fastapi)-like dependency injection to your [AioHTTP](https://github.com/aio-libs/aiohttp) application and swagger documentation based on types.
To start using dependency injection, just initialize the injector.
```python
from aiohttp import web
from aiohttp_deps import init as deps_initapp = web.Application()
app.on_startup.append(deps_init)
web.run_app(app)
```
If you use mypy, then we have a custom router with propper types.
```python
from aiohttp import web
from aiohttp_deps import init as deps_init
from aiohttp_deps import Routerrouter = Router()
@router.get("/")
async def handler():
return web.json_response({})app = web.Application()
app.router.add_routes(router)
app.on_startup.append(deps_init)
web.run_app(app)
```
Also, you can nest routers with prefixes,
```python
api_router = Router()memes_router = Router()
main_router = Router()
main_router.add_routes(api_router, prefix="/api")
main_router.add_routes(memes_router, prefix="/memes")
```## Swagger
If you use dependencies in you handlers, we can easily generate swagger for you.
We have some limitations:
1. We don't support resolving type aliases if hint is a string.
If you define variable like this: `myvar = int | None` and then in handler
you'd create annotation like this: `param: "str | myvar"` it will fail.
You need to unquote type hint in order to get it work.We will try to fix these limitations later.
To enable swagger, just add it to your startup.
```python
from aiohttp_deps import init, setup_swaggerapp = web.Application()
app.on_startup.extend([init, setup_swagger()])
```### Responses
You can define schema for responses using dataclasses or
pydantic models. This would not affect handlers in any way,
it's only for documentation purposes, if you want to actually
validate values your handler returns, please write your own wrapper.```python
from dataclasses import dataclassfrom aiohttp import web
from pydantic import BaseModelfrom aiohttp_deps import Router, openapi_response
router = Router()
@dataclass
class Success:
data: strclass Unauthorized(BaseModel):
why: str@router.get("/")
@openapi_response(200, Success, content_type="application/xml")
@openapi_response(200, Success)
@openapi_response(401, Unauthorized, description="When token is not correct")
async def handler() -> web.Response:
...
```This example illustrates how much you can do with this decorator. You
can have multiple content-types for a single status, or you can have different
possble statuses. This function is pretty simple and if you want to make
your own decorator for your responses, it won't be hard.## Default dependencies
By default this library provides only two injectables. `web.Request` and `web.Application`.
```python
async def handler(app: web.Application = Depends()): ...
async def handler2(request: web.Request = Depends()): ...
```
It's super useful, because you can use these dependencies in
any other dependency. Here's a more complex example of how you can use this library.```python
from aiohttp_deps import Router, Depends
from aiohttp import webrouter = Router()
async def get_db_session(app: web.Application = Depends()):
async with app[web.AppKey("db")] as sess:
yield sessclass MyDAO:
def __init__(self, session=Depends(get_db_session)):
self.session = sessionasync def get_objects(self) -> list[object]:
return await self.session.execute("SELECT 1")@router.get("/")
async def handler(db_session: MyDAO = Depends()):
objs = await db_session.get_objects()
return web.json_response({"objects": objs})```
If you do something like this, you would never think about initializing your DAO. You can just inject it and that's it.
# Built-in dependencies
This library also provides you with some default dependencies that can help you in building the best web-service.
## Json
To parse json, create a pydantic model and add a dependency to your handler.
```python
from aiohttp import web
from pydantic import BaseModel
from aiohttp_deps import Router, Json, Dependsrouter = Router()
class UserInfo(BaseModel):
name: str@router.post("/users")
async def new_data(user: UserInfo = Depends(Json())):
return web.json_response({"user": user.model_dump()})```
This dependency automatically validates data and send
errors if the data doesn't orrelate with schema or body is not a valid json.If you want to make this data optional, just mark it as optional.
```python
@router.post("/users")
async def new_data(user: Optional[UserInfo] = Depends(Json())):
if user is None:
return web.json_response({"user": None})
return web.json_response({"user": user.model_dump()})```
## Headers
You can get and validate headers using `Header` dependency.
Let's try to build simple example for authorization.
```python
from aiohttp_deps import Router, Header, Depends
from aiohttp import webrouter = Router()
def decode_token(authorization: str = Depends(Header())) -> str:
if authorization == "secret":
# Let's pretend that here we
# decode our token.
return authorization
raise web.HTTPUnauthorized()@router.get("/secret_data")
async def new_data(token: str = Depends(decode_token)) -> web.Response:
return web.json_response({"secret": "not a secret"})```
As you can see, header name to parse is equal to the
name of a parameter that introduces Header dependency.If you want to use some name that is not allowed in python, or just want to have different names, you can use alias. Like this:
```python
def decode_token(auth: str = Depends(Header(alias="Authorization"))) -> str:
```Headers can also be parsed to types. If you want a header to be parsed as int, just add the typehint.
```python
def decode_token(meme_id: int = Depends(Header())) -> str:
```If you want to get list of values of one header, use parameter `multiple=True`.
```python
def decode_token(meme_id: list[int] = Depends(Header(multiple=True))) -> str:```
And, of course, you can provide this dependency with default value if the value from user cannot be parsed for some reason.
```python
def decode_token(meme_id: str = Depends(Header(default="not-a-secret"))) -> str:
```## Queries
You can depend on `Query` to get and parse query parameters.
```python
from aiohttp_deps import Router, Query, Depends
from aiohttp import webrouter = Router()
@router.get("/shop")
async def shop(item_id: str = Depends(Query())) -> web.Response:
return web.json_response({"id": item_id})```
the name of the parameter is the same as the name of function parameter.
The Query dependency is acually the same as the Header dependency, so everything about the `Header` dependency also applies to `Query`.
## Views
If you use views as handlers, please use View class from `aiohttp_deps`, otherwise the magic won't work.
```python
from aiohttp_deps import Router, View, Depends
from aiohttp import webrouter = Router()
@router.view("/view")
class MyView(View):
async def get(self, app: web.Application = Depends()):
return web.json_response({"app": str(app)})```
## Forms
Now you can easiy get and validate form data from your request.
To make the magic happen, please add `arbitrary_types_allowed` to the config of your model.```python
import pydantic
from aiohttp_deps import Router, Depends, Form
from aiohttp import webrouter = Router()
class MyForm(pydantic.BaseModel):
id: int
file: web.FileFieldmodel_config = pydantic.ConfigDict(arbitrary_types_allowed=True)
@router.post("/")
async def handler(my_form: MyForm = Depends(Form())):
with open("my_file", "wb") as f:
f.write(my_form.file.file.read())
return web.json_response({"id": my_form.id})```
## Path
If you have path variables, you can also inject them in your handler.
```python
from aiohttp_deps import Router, Path, Depends
from aiohttp import webrouter = Router()
@router.get("/view/{var}")
async def my_handler(var: str = Depends(Path())):
return web.json_response({"var": var})```
## ExtraOpenAPI
This dependency is used to add additional swagger fields to the endpoint's swagger
that is using this dependency. It might be even indirect dependency.You can check how this thing can be used in our [examples/swagger_auth.py](https://github.com/taskiq-python/aiohttp-deps/tree/master/examples/swagger_auth.py).
## Overriding dependencies
Sometimes for tests you don't want to calculate actual functions
and you want to pass another functions instead.To do so, you can add "dependency_overrides" or "values_overrides" to the aplication's state.
These values should be dicts. The keys for these values can be found in `aiohttp_deps.keys` module.Here's an example.
```python
def original_dep() -> int:
return 1class MyView(View):
async def get(self, num: int = Depends(original_dep)):
"""Nothing."""
return web.json_response({"request": num})
```Imagine you have a handler that depends on some function,
but instead of `1` you want to have `2` in your tests.To do it, jsut add `dependency_overrides` somewhere,
where you create your application. And make sure that keys
of that dict are actual function that are being replaced.```python
from aiohttp_deps import VALUES_OVERRIDES_KEYmy_app[VALUES_OVERRIDES_KEY] = {original_dep: 2}
```But `values_overrides` only overrides returned values. If you want to
override functions, you have to use `dependency_overrides`. Here's an example:```python
from aiohttp_deps import DEPENDENCY_OVERRIDES_KEYdef replacing_function() -> int:
return 2my_app[DEPENDENCY_OVERRIDES_KEY] = {original_dep: replacing_function}
```The cool point about `dependency_overrides`, is that it recalculates graph and
you can use dependencies in function that replaces the original.