Skip to main content

Constructor injection for Python

Project description

diapyr

Constructor injection for Python

Usage examples

Dependency injection

>>> from diapyr import DI, types
>>>
>>> class Config: pass
>>>
>>> class Database:
...
...     @types(Config)
...     def __init__(self, config):
...         self.conninfo = config.conninfo
...
>>> class Server:
...
...     @types(Config, Database)
...     def __init__(self, config, database):
...         self.port = config.port
...         self.database = database
...
...     def serve_forever(self):
...         print(self.port)
...
>>> config = Config()
>>> config.conninfo = 'dbname=test user=postgres'
>>> config.port = 8000
>>> with DI() as di:
...     di.add(config)
...     di.add(Database)
...     di.add(Server)
...     di(Server).serve_forever()
8000

Java-style inner class

>>> from diapyr.util import innerclass
>>>
>>> class X:
...
...     @innerclass
...     class Y(object):
...
...         def bar(self):
...             return self.foo # Y can access all fields of the enclosing instance of X.
...
...     def __init__(self, foo):
...         self.foo = foo
...
>>> x = X(100)
>>> y = x.Y()
>>> y.bar()
100

Overview

  • Automatic wiring between application objects
  • Declare what collaborator types you want in your constructor using an @types decorator, they correspond with params
    • You can decorate a (factory) function in the same way, in which case specify a return type using the this kwarg, or if left out the return type is the function itself
    • Surround a type in square brackets if you want a list of all matching objects, normally diapyr provides a unique match
  • Add such classes/factories to a DI instance
    • You can also add objects, for example an application config object
  • Request a type from the DI instance and diapyr will attempt to make it for you, along with the rest of the object graph
  • Instances are cached in the DI object
    • On exit from with clause, dispose is called on any created instances that have it

Motivation

  • Manual wiring is messy and tedious especially when an app gets big
  • Constructor injection avoids spaghetti code
  • It's trivial to depend on an object anywhere in the graph as long as you don't create a reference cycle e.g. most objects will depend on the same config instance
    • No need to resort to globals, which come with a big risk of leaking state between unit tests
  • Unit testing is easy when an object only interacts with what was passed into its constructor

Convention

  • When depending on a config object the constructor should first extract what it needs and assign those values to fields
  • Collaborators should also be assigned to fields, and ideally the constructor won't do anything else

Advanced

  • Parameter defaults are honoured, this can be used to depend on a module's log object in real life but pass in a mock in unit tests
  • Decorating an instance method (without this kwarg) will make it behave as an additional constructor
    • Take advantage of name mangling (start with double underscore e.g. __init) to avoid having to call super
  • Decorating an instance method with this kwarg will make it behave as a factory function
    • Adding a class to DI will implicity add all such methods it has as factories
  • You can play fast and loose with types, diapyr doesn't care whether a factoried object satisfies the declared type

API

diapyr

DI

Convenience import.

types

Convenience import.

diapyr.diapyr

types
def types(*deptypes, **kwargs)

Declare that the decorated function or method expects args of the given types. Use square brackets to request all instances of a type as a list. Use this kwarg to declare the type of result returned by a factory.

DI Objects

class DI()

add
def add(obj)

Register the given class, factory or instance.

all
def all(type)

Return all objects of the given type, instantiating them and collaborators if necessary.

__call__
def __call__(clazz)

Return unique object of the given type, instantiating it and its collaborators if necessary.

__exit__
def __exit__(*exc_info)

Discard all instances created by this container, calling dispose if they have it.

diapyr.start

diapyr.util

innerclass
def innerclass(cls)

An instance of the decorated class may access its enclosing instance via self.

singleton
def singleton(t)

The decorated class is replaced with a no-arg instance.

invokeall
def invokeall(callables)

Invoke every callable, even if one or more of them fail. This is mostly useful for synchronising with futures. If all succeeded return their return values as a list, otherwise raise all exceptions thrown as a chain.

bfs
def bfs(keys)

Breadth-first search starting with the given iterable of keys, intended to be used as a decorator. If a function is decorated it should take an info object and key, and yield subsequent keys. If a class is decorated, a new instance of it is used as info object: The class should have a newdepth method that will be called before each depth, and a process method that takes a key and yields subsequent keys as in the function case. The info object is kept updated with the list of currentkeys, current depth and the set of donekeys. Note that the first currentkeys (depth 0) is exactly the passed in keys iterable, subsequent currentkeys will be non-empty lists. The process function is only invoked for keys that have not yet been processed, i.e. unique keys. When finished the decorated function/class is replaced with the last state of the info object.

setuphacks

getsetupkwargs
def getsetupkwargs(setuppath, fields)

Extract the kwargs passed to setup at the given path (typically some setup.py) that have names in the given fields.

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

diapyr-33.tar.gz (11.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

diapyr-33-py2.py3-none-any.whl (12.5 kB view details)

Uploaded Python 2Python 3

File details

Details for the file diapyr-33.tar.gz.

File metadata

  • Download URL: diapyr-33.tar.gz
  • Upload date:
  • Size: 11.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.1 CPython/3.10.12

File hashes

Hashes for diapyr-33.tar.gz
Algorithm Hash digest
SHA256 556e24b030094a96cabcc5f5b2e612b5f83c22d3afaf30c8880a118705707aa2
MD5 e4ae1d7e19ee082b279241a57d974143
BLAKE2b-256 f6218799f3199cd6552b6f01f99a036ad842883f35f3ba421a1152de34c2e617

See more details on using hashes here.

File details

Details for the file diapyr-33-py2.py3-none-any.whl.

File metadata

  • Download URL: diapyr-33-py2.py3-none-any.whl
  • Upload date:
  • Size: 12.5 kB
  • Tags: Python 2, Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.1 CPython/3.10.12

File hashes

Hashes for diapyr-33-py2.py3-none-any.whl
Algorithm Hash digest
SHA256 9222c11ee87af68f6d31b2bbef004e3acd762063dbd391f247ab5349ff13d3a0
MD5 7d48e10c43400f900219be390791d2d5
BLAKE2b-256 d07fb8330fa82e96c6294df5ef93866a5ee0971831713b11c9e38f03ae468189

See more details on using hashes here.

Supported by

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