Skip to content

Bundles

Bundles

As the number of classes and objects grow you may find that you will create more complex dependency graphs. You will have a groups of components that you will want to register together. For example if you want to have a sqlalchemy Session you will likely want a sqlalchemy Engine. This is where bundles come in to play.

At it's minimal a bundle is a just a Callable[[Container], None] that can be used to set up related registrations on the container. The simplest type of bundle is a simple function that takes a container as a argument.

class ClientDependency:
    def get_int(self):
        return 10

class Client:
    def __init__(self, dep: ClientDependency):
        self.dep = dep

    def get_number(self):
        return self.dep.get_int()

def client_bundle(c: Container):
    c.register(ClientDependency)
    c.register(Client)

container.apply_bundle(client_bundle)

client = container.resolve(Client)

client.get_number() # returns 10

Helpers for bundles

There are a set of base Bundle classes that can be used to set up bundles - BaseBundle: Basic callable - OnlyRunOncePerInstanceBundle: Only run the bundle once in the container instance - OnlyRunOncePerClassBundle: Only run this bundle class once in a container

You can find these classes in the clean_ioc.bundles module.

These bundle classes are also useful if you want to pass instances of dependencies with your bundle.

@dataclass
class ClientConfig:
    url: str

class Client:
    def __init__(self, config: ClientConfig):
        self.base_url = config.url

    def get_user(self):
        # Do some requests stuff here
        pass



class ClientBundle(OnlyRunOncePerClassBundle):

    def __init__(self, config: ClientConfig):
        self.config = config

    def apply(self, c: Container):
        c.register(ClientConfig, instance=self.config)
        c.register(Client)



client_config = ClientConfig(
    url = "https://example.com"
)

container.apply_bundle(ClientBundle(config=client_config)) # Gets applied
container.apply_bundle(ClientBundle(config=client_config)) # Does not get applied

client = container.resolve(Client)

user = client.get_user()