Skip to main content

..a tiny but powerful DI library for Python.

Project description

hazrakah on PyPI hazrakah on readthedocs

hazrakah (הזרקה) is a tiny but powerful DI library for Python.

This README is only a high-level introduction to hazrakah. For more detailed documentation, please view the official docs at https://hazrakah.readthedocs.io.

Features

  • Supports Transient, Scoped, Singleton and Instance registrations.
  • Registration targets can be a concrete type or a factory.
  • Scopes are hierarchical, and scoped registrations are isolated to each scope.
  • Container is mutable by default, but can be frozen on-demand.

Installation

You can install hazrakah from PyPI through usual means, such as pip:

   pip install hazrakah

Usage

To use hazrakah simply create a Container instance and create one or more type registrations. Afterward, the container can be used to resolve instances for the types you have registered:

    from hazrakah import Container

    # assume you have three classes, Fizz, Buzz, and FizzBuzz,
    # and also assume you have Protocols (interfaces) for each.

    container = Container()
    scoped_container = container.create_scope()

    # TRANSIENT == a new instance of `Foo` is created
    # for every resolve of `IFoo`.
    container.register_transient(IFizz, Fizz)
    #
    fizz1 = container.resolve(IFizz)
    fizz2 = container.resolve(IFizz)
    assert fizz1 is not fizz2, 'transients reg, every resolve is a new instance.'


    # SCOPED == within a given "scope" a single instance is returned
    # for every resolve of `IBuzz`. new scopes == new instances.
    container.register_scoped(IBuzz, Buzz)
    #
    buzz1 = container.resolve(IBuzz)
    buzz2 = container.resolve(IBuzz)
    buzz3 = scoped_container.resolve(IBuzz)
    assert buzz1 is buzz2, 'scoped resolves are singleton-like WITHIN a container scope.'
    assert buzz1 is not buzz3, 'scoped resolves are NOT shared between container scopes, new scope? new singleton.'

    # INSTANCE == the provided instance is returned
    # for every resolve of `FizzBuzz`.
    container.register_instance(FizzBuzz, FizzBuzz())
    #
    fizzbuzz1 = container.resolve(FizzBuzz)
    fizzbuzz2 = scoped_container.resolve(FizzBuzz)
    assert fizzbuzz1 is fizzbuzz2, 'instance resolves always yield the provided instance'
    # NOTE: "scopes" resolve hierarchically any "non-scoped" type registration,
    #       this is why `scoped_container`` resolved the same instance as `container`.
    #       if this were a scoped registration a NEW instance would have been created.
    #       this scoping logic is true for INSTANCE, SINGLETON, and TRANSIENT regs.

    # SINGLETON == a SINGLE instance will be created
    # for ALL resolves of `IFizzBuzz`.
    container.register_singleton(IFizzBuzz, lambda c: c.resolve(FizzBuzz))
    fizzbuzz3 = container.resolve(IFizzBuzz)
    fizzbuzz4 = container.resolve(IFizzBuzz)
    assert fizzbuzz3 is fizzbuzz4, 'singleton resolves always yield a single instance.'

    # for completeness, concrete types can self-regsiter without a target spec.
    container.register_transient(Fizz)
    fizz3 = container.resolve(Fizz)
    assert fizz3 is not None

    # and last, but not least, containers can be frozen,
    # making them immutable (also freezing any scopes
    # created after being frozen.)  once frozen, they
    # cannot be unfrozen.
    try:
        container.freeze()
        container.register_instance(Fizz, Fizz())
    except RegistrationError:
        pass
    else:
        assert False, 'Frozen containers should be immutable.'

    try:
        scoped_container.register_instance(Fizz, Fizz())
    except:
        assert False, 'Scopes created BEFORE freezing are NOT frozen.'

    try:
        scope2 = container.create_scope()
        scope2.register_instance(Fizz, Fizz())
    except RegistrationError:
        pass
    else:
        assert False, 'Scopes created AFETR freezing are also frozen.'

    try:
        setattr(container, '__frozen', False)
    except AttributeError:
        pass
    else:
        assert False, 'Attempts to modify Container directly will fail.'

Contact

You can reach me on Discord or open an Issue on Github.

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

hazrakah-0.0.4.tar.gz (8.8 kB view details)

Uploaded Source

Built Distribution

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

hazrakah-0.0.4-py3-none-any.whl (7.8 kB view details)

Uploaded Python 3

File details

Details for the file hazrakah-0.0.4.tar.gz.

File metadata

  • Download URL: hazrakah-0.0.4.tar.gz
  • Upload date:
  • Size: 8.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.11.15+

File hashes

Hashes for hazrakah-0.0.4.tar.gz
Algorithm Hash digest
SHA256 658c38242cfcb2fb7162e7cd61c5b4f3100986e6dc7f1a18abfee56bf25cc9c1
MD5 a0de3a2038ab676ee121034db624d271
BLAKE2b-256 a0cc0046f7b82e3181269a6b758b907f90ef48f016e4e5d0e6e439d8c17ac657

See more details on using hashes here.

File details

Details for the file hazrakah-0.0.4-py3-none-any.whl.

File metadata

  • Download URL: hazrakah-0.0.4-py3-none-any.whl
  • Upload date:
  • Size: 7.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.11.15+

File hashes

Hashes for hazrakah-0.0.4-py3-none-any.whl
Algorithm Hash digest
SHA256 b86d8d58575edaaf967898f402df9d8ca4edf1819dda30e7ff5ccb941d4e2c17
MD5 7351ca63a97eb3b2b13c59c1e1c9a03a
BLAKE2b-256 e02e2d14ed3dfd461761319522356d7f35f8a0cb600a70b8119bb4876a470086

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