@inject decorator or inject_all = True for AsyncBase, refs #878

This commit is contained in:
Simon Willison 2021-11-16 14:01:16 -08:00
commit 22f41f7983
2 changed files with 22 additions and 2 deletions

View file

@ -8,13 +8,23 @@ except ImportError:
from . import vendored_graphlib as graphlib
def inject(fn):
fn._inject = True
return fn
class AsyncMeta(type):
def __new__(cls, name, bases, attrs):
# Decorate any items that are 'async def' methods
_registry = {}
new_attrs = {"_registry": _registry}
inject_all = attrs.get("inject_all")
for key, value in attrs.items():
if inspect.iscoroutinefunction(value) and not value.__name__ == "resolve":
if (
inspect.iscoroutinefunction(value)
and not value.__name__ == "resolve"
and (inject_all or getattr(value, "_inject", None))
):
new_attrs[key] = make_method(value)
_registry[key] = new_attrs[key]
else:

View file

@ -1,5 +1,5 @@
import asyncio
from datasette.utils.asyncdi import AsyncBase
from datasette.utils.asyncdi import AsyncBase, inject
import pytest
from random import random
@ -8,15 +8,22 @@ class Simple(AsyncBase):
def __init__(self):
self.log = []
@inject
async def two(self):
self.log.append("two")
@inject
async def one(self, two):
self.log.append("one")
return self.log
async def not_inject(self, one, two):
return one + two
class Complex(AsyncBase):
inject_all = True
def __init__(self):
self.log = []
@ -40,6 +47,8 @@ class Complex(AsyncBase):
class WithParameters(AsyncBase):
inject_all = True
async def go(self, calc1, calc2, param1):
return param1 + calc1 + calc2
@ -53,6 +62,7 @@ class WithParameters(AsyncBase):
@pytest.mark.asyncio
async def test_simple():
assert await Simple().one() == ["two", "one"]
assert await Simple().not_inject(6, 7) == 13
@pytest.mark.asyncio