Skip to main content

Library for annotation-based dependency injection

Project description

PyPI Version Supported Python Versions Build Status Coverage report

andi tells which kwargs should to be passed to a callable, based on callable arguments’ type annotations.

andi is useful as a building block for frameworks, or as a library which helps to implement dependency injection (thus the name - ANnotation-based Dependency Injection).

License is BSD 3-clause.

Installation

pip install andi

andi requires Python >= 3.5.3.

Idea

andi does a simple thing, but it requires some explanation why this thing is useful.

You’re building a framework. This framework has code which calls some user-defined function (callback). Callback receives arguments foo and bar:

def my_framework(callback):
    # ... compute foo and bar somehow
    result = callback(foo=foo, bar=bar)
    # ...

Then you decide that you want the framework to be flexible, and support callbacks which take

  • both foo and bar,

  • only foo,

  • only bar,

  • nothing.

If a callback only takes foo, it may be unnecessary to compute bar.

In addition to that, you realize that there can be environments or implementations where foo is available for the framework, but bar isn’t, but you still want to reuse the callbacks which work without bar, and disable (or error) those who need bar.

So, the logic is the following:

  1. Framework defines which inputs are available, or can be possibly computed (e.g. foo and bar).

  2. Callback declares which inputs it receives (e.g. bar).

  3. Framework inspects the callback, finds arguments the callback needs.

  4. Optional: if there are some arguments which callback needs, but framework doesn’t provide, an error is raised (or callback is disabled).

  5. Framework computes argument values (bar in this case).

  6. Framework calls the callback.

Depending on implementation, steps 1-5 may happen iteratevely - e.g. middlewares may be populating different parts of callback kwargs. In this case step (4 - raising an error) can be skipped.

andi is a library which helps to support this workflow.

Usage

andi usage looks like this:

import andi

class Foo:
    pass

class Bar:
    pass

class Baz:
    pass


# use type annotations to declare which inputs a callback wants
def my_callback1(foo: Foo):
    pass


def my_callback2(bar: Bar, foo: Foo):
    pass


def my_framework(callback):
    kwargs_to_provide = andi.to_provide(callable,
                                        can_provide={Foo, Bar, None})
    # for my_callback: kwargs_to_provide == {'foo': Foo}

    # Create all the dependencies - implementation is framework-specific,
    # and can be organized in different ways. Code below is an example.
    kwargs = {}
    for name, cls in kwargs_to_provide.items():
        if cls is Foo:
            kwargs[name] = Foo()
        elif cls is Bar:
            kwargs[name] = fetch_bar()
        elif cls is None:
            kwargs[name] = None
        else:
            raise Exception("Unexpected type")  # shouldn't really happen

    # everything is ready, call the callback
    result = callback(**kwargs)
    # ...

my_framework(my_callback1)  # Foo instance is passed to my_callback1
my_framework(my_callback2)  # Bar and Foo instances are passed to my_callback2

If a callback wants some input which framework can’t provide, then some arguments are going to be missing in kwargs, and Python can raise TypeError, as usual. It is possible to check it explicitly, to avoid doing unnecessary work creating values for other arguments:

arguments = andi.inspect(callable)
kwargs_to_provide = andi.to_provide(arguments,
                                    can_provide={Foo, Bar, None})
cant_provide = arguments.keys() - kwargs_to_provide.keys()
if cant_provide:
    raise Exception("Can't provide arguments: %s" % cant_provide)

andi support typing.Union. If an argument is annotated as Union[Foo, Bar], it means “both Foo and Bar objects are fine, but callable prefers Foo”:

def callback4(x: Union[Baz, Bar, Foo]):
    pass

# Bar is preferred to Foo, and Baz is not available, so my_framework passes
# Bar instance to ``x`` argument (``x = fetch_bar()``)
my_framework(callback4)

andi also supports typing.Optional types. If an argument is annotated as optional, it means Union[<other types>, None]. So usually framework specifies that None is OK, and provides it; None has the least priority:

def callback4(foo: Optional[Foo], baz: Optional[Baz]):
    pass

# foo=Foo(), baz=None is passed, because my_framework
# supports Foo, but not Baz
my_framework(callback4)

andi only checks type-annotated arguments; arguments without annotations are ignored.

Constructor Dependency Injection

It is common for frameworks to ask users to define classes with a certain interface, not just callbacks. andi can be used like this:

class UserClass:
    def __init__(self, foo: Foo):
        self.foo = foo
    # ...

class MyFramework:
    # ...
    def create_instance(self, user_cls):
        kwargs_to_provide = andi.to_provide(user_cls.__init__,
                                            can_provide={Foo, Bar})
        # ... fill kwargs, based on ``kwargs_to_provide``
        return user_cls(**kwargs)

obj = framework.create_instance(UserClass)

Pattern is the following:

  1. ask user classes to declare all dependencies in __init__ method,

  2. then framework creates instances of these classes, passing all the required dependencies.

Instead of __init__ you can also use a classmethod.

Recursive dependencies

andi can be used on different levels in a framework. For example, framework supports callbacks which receive instances of some BaseClass subclasses:

class UserClass(framework.BaseClass):
    def __init__(self, foo: Foo):
        self.foo = foo

def callback(user: UserClass):
    # ...

class MyFramework:
    # ...
    def create_instance(self, user_cls):
        kwargs_to_provide = andi.to_provide(user_cls.__init__,
                                            can_provide={Foo, Bar})
        # ... fill kwargs, based on ``kwargs_to_provide``, i.e.
        # create Foo and Bar objects somehow
        return user_cls(**kwargs)

    def call_callback(self, callback):
        kwargs_to_provide = andi.to_provide(
            callback,
            can_provide=self.is_allowed_callback_argument
        )
        kwargs = {}
        for name, user_cls in kwargs_to_provide.items():
            kwargs[name] = self.create_instance(user_cls)
        return callback(**kwargs)

    def is_allowed_callback_argument(self, cls):
        return issubclass(cls, framework.BaseClass)

In this example callback needs a dependency (UserClass object), and UserClass object on itself has a dependency (Foo). So andi is used to find out these dependencies, and then framework creates Foo object first, then UserClass object, and then finally calls the callback.

Implementation can be recursive as well, e.g. Foo may need some dependencies as well.

Why type annotations?

andi uses type annotations to declare dependencies (inputs). It has several advantages, and some limitations as well.

Advantages:

  1. Built-in language feature.

  2. You’re not lying when specifying a type - these annotations still work as usual type annotations.

  3. In many projects you’d annotate arguments anyways, so andi support is “for free”.

Limitations:

  1. Callable can’t have two arguments of the same type.

  2. This feature could possibly conflict with regular type annotation usages.

If your callable has two arguments of the same type, consider making them different types. For example, a callable may receive url and html of a web page:

def parse(html: str, url: str):
    # ...

To make it play well with andi, you may define separate types for url and for html:

class HTML(str):
    pass

class URL(str):
    pass

def parse(html: HTML, url: URL):
    # ...

This is more boilerplate though.

You can also refactor parse to have a single argument:

@dataclass
class Response:
    url: str
    html: str

def parse(response: Response):
    # ...

Why doesn’t andi handle creation of objects?

Currently andi just inspects callable and chooses best concrete types a framework needs to create and pass to a callable, without prescribing how to create them. This makes andi useful in various contexts - e.g.

  • creation of some objects may require asynchronous funnctions, and it may depend on libraries used (asyncio, twisted, etc.)

  • in streaming architectures (e.g. based on Kafka) inspection may happen on one machine, while creation of objects may happen on different nodes in a distributed system, and then actually running a callable may happen on yet another machine.

It is hard to design API with enough flexibility for all such use cases. That said, andi may provide more helpers in future, once patterns emerge, even if they’re useful only in certain contexts.

Contributing

Use tox to run tests with different Python versions:

tox

The command above also runs type checks; we use mypy.

Changes

TBA

Initial release.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

andi-0.1.tar.gz (9.7 kB view details)

Uploaded Source

Built Distribution

andi-0.1-py3-none-any.whl (7.5 kB view details)

Uploaded Python 3

File details

Details for the file andi-0.1.tar.gz.

File metadata

  • Download URL: andi-0.1.tar.gz
  • Upload date:
  • Size: 9.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/1.13.0 pkginfo/1.5.0.1 requests/2.22.0 setuptools/41.2.0 requests-toolbelt/0.9.1 tqdm/4.35.0 CPython/3.7.1

File hashes

Hashes for andi-0.1.tar.gz
Algorithm Hash digest
SHA256 b78e51b238c0c99b3ce5ac12f90c3bdb652cc2b6bdc40c8c3dc9c8f296b77f3b
MD5 05b527572312f5b863ffbe2415062e9e
BLAKE2b-256 68c5b862a8890cb01620cbec939ade48b7c96fa38354854eeb63d16d07a39edb

See more details on using hashes here.

Provenance

File details

Details for the file andi-0.1-py3-none-any.whl.

File metadata

  • Download URL: andi-0.1-py3-none-any.whl
  • Upload date:
  • Size: 7.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/1.13.0 pkginfo/1.5.0.1 requests/2.22.0 setuptools/41.2.0 requests-toolbelt/0.9.1 tqdm/4.35.0 CPython/3.7.1

File hashes

Hashes for andi-0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 6f5435a510ca487b683783934ac1db67ba75d5e682c2eb426946ef19b2de2e8c
MD5 3cda79495f7401996460781dc0e0d903
BLAKE2b-256 c01200abded0948a6245b9e2d549a74800da3fa8fc3250884c3fbc91107f967b

See more details on using hashes here.

Provenance

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page