Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

async-injection

pcafstockf9.5kMIT2.0.1TypeScript support: included

A robust lightweight dependency injection library for TypeScript.

ioc, di, async, dependency, injection, dependency injection, inversion of control, injector, container, typescript

readme

Async-Injection

CI Actions Publish Actions npm version License: MIT OSS Lifecycle

A robust lightweight dependency injection library for TypeScript.

About

Async-Injection is a small IoC container with support for both synchronous and asynchronous dependency injection, as well as isolated and/or hierarchical scopes.

Installation

You can get the latest release using npm:

$ npm install async-injection --save

To enhance flexibility, Async-Injection does not dictate which Reflect API implementation you use. However, you will need to explicitly choose and load one.
You can use:

The Reflect polyfill import should only be added once, and before Async-Injection is used:

// entry.ts
import "reflect-metadata";
// Your code here...

Please note that this library supports a wide variety of runtimes and is distributed as both esm and cjs modules, side by side.

Basic Usage (synchronous)

Here we 'get' a new transaction handling object, that itself, relies on a shared service:

@Injectable()
class SharedService {
    constructor(@Inject('LogLevel') @Optional('warn') private logLevel: string) { }
}

@Injectable()
class TransactionHandler {
    constructor(svc: SharedService) { }
}

// Create a simple container (we will bind providers into it).
const container = new Container();

// A single instance will be created and shared by everyone.
container.bindClass(SharedService).asSingleton();

// A new instance will be created each time one is requested.
container.bindClass(TransactionHandler);

// If we omit this line, the logLevel of SharedService will be initialized to 'warn'
container.bindConstant('LogLevel', 'info');

// In our request processing code (which would be an anti-pattern)...
// Instantiate a new transaction handler (it will be injected with the shared service).
const tx = container.get(TransactionHandler);

NOTE:
The examples in this ReadMe are contrived to quickly communicate concepts and usage.
Your real world project should of course follow best practices like separation of concerns, having a composition root, and should avoid anti-patterns like service locator.

Scopes

Scopes can be created using multiple Containers, and/or a hierarchy of Containers.

IoC Modules

Why reinvent the wheel? TypeScript is great!
Implement the "module" you want and just import it:

my-http-ioc-module.ts

import {myContainer} from './app';
import {Logger, HttpClient} from './services';
import {HttpClientGotWrapper} from './impl';

myContainer.bind(Logger).asSingleton();
myContainer.bind(HttpClient, HttpClientGotWrapper);

Asynchronous Support

For simplicity, it is recommended that you use traditional synchronous injection for any class where that is possible.
But when that's just to much work, you can "blend" synchronous and asynchronous injection. To support "blending", we introduce three new methods on the Container which will be explained below.

Asynchronous Usage

Perhaps in the example above, our SharedService is useless until it has established a database connection.
Of course such a simple scenario could easily be handled in user-land code, but as application complexity grows, this becomes more tedious and difficult to maintain.
Let's modify the example as follows:

@Injectable()
class SharedService {
    constructor() { }
    connect(): Promise<void> { ... }
}

const container = new Container();

// Bind a factory function that awaits until it can fully create a SharedService.
container.bindAsyncFactory(SharedService, async () => {
    let svc = new SharedService();
    return await svc.connect();
}).asSingleton();

// A new transient instance will be created each time one is requested.
container.bindClass(TransactionHandler);    

// Wait for all bound asynchronous factory functions to complete.
// This step is optional.  You could omit and use Container.resolve instead (see alternative below).
await container.resolveSingletons(true);
// We are now connected to the database

// In our request processing code...
const tx = container.get(TransactionHandler);

As an alternative, we could remove the call to Container.resolveSingletons, and in our request processing code, simply call Container.resolve.

const tx = await container.resolve(TransactionHandler);

Important - Container.resolve vs Container.get

Blending synchronous and asynchronous injection adds complexity to your application.
The key to successful blending is to think of the object you are requesting, not as an object, but as a tree of objects with your object at the top.
Keep in mind that you may have transient objects which need to be created each time, as well as existing singleton objects in your dependency tree.
If you know ahead of time that every object which you depend on is immediately (synchronously) available, or if they are asynchronous singletons which have already been resolved (via Container.resolveSingletons, or a previous call to Container.resolve), then no need to wait, you can just Container.get the tree.
Otherwise you need to await the full resolution of the tree with await Container.resolve.

@PostConstruct Support

It is not always possible to fully initialize your object in the class constructor. This (albeit contrived) demo shows that the Employee class is not yet initialized when the Person subclass tries to call the overridden state method.

class Person {
    public constructor() { this.describe(); }
    protected state() { return "relaxing"; }
    public describe() { console.log("Hi I'm '" + this.state() + "'"); }
}
class Employee extends Person {
    constructor(private manager: boolean) {    super(); }
    protected state() { return this.manager ? "busy" : "producing"; }
}
// This will print: 
//  "Hi I'm 'producing", even though the author probably expected 
//  "Hi I'm busy", because they passed true for the 'manager' parameter.
new Employee(true); 

Can we refactor code to work around this? Sure. You may have to submit a couple of PR's, re-write legacy code that has no unit tests, trash encapsulation, skip a few nights sleep, etc. But why?
A PostConstruct annotation ensure's your initialization method is working on a fully constructed version of your object. Even better, since constructors cannot be asynchronous, PostConstruct gives you an easy way to asynchronously prepare an object before it's put into service.

@PostConstruct Usage

Post construction methods can be either synchronous or asynchronous.

class A {
    public constructor() { }

    // Called before the object is placed into the container (or is returned from get/resolve)
    @PostConstruct()
    public init(): void { ... } 
}
class D {
    public constructor() { }

    // Will not be placed into the container (or returned) until the Promise has been resolved.
    @PostConstruct()
    public init(): Promise<void> { ... }    
}

@PostConstruct Guidelines:

  • Ensure your post construction method signature properly declares it's return type.
    WARNING! An unspecified return type signature where the type is implied by return new Promise(...) is not sufficient! You must explicitly declare the return type.
  • Container.get will throw an exception if you try to retrieve a class with @PostConstruct on a method that returns a Promise, but which does not declare it's return type to be Promise.
  • The library will not invoke @PostConstruct on an object returned from a factory. It is the factory's responsibility to construct and initialize before returning.
  • You will likely want a Container.resolveSingletons(true) call between your last Container.bindXXX() call and any Container.get call.

API Overview

Async-Injection tries to follow the common API patterns found in most other DI implementations. Please refer to the examples above or the linked elements below for specific syntax.

  • The Container class implements a Binder interface, which allows you to bind a Constant, Factory, AsyncFactory, or Class value to an InjectableId (aka key) within a Container.
  • The Container also implements an Injector interface which allows you to synchronously get or asynchronously resolve anything that has been bound.
  • When binding a Factory, AsyncFactory or Class to an InjectableId, you can chain the result of the call to specify the binding as a Singleton, and/or configure an Error Handler.
  • Containers may be nested by passing a parent Container to the constructor of a child Container.
  • To bind a Class into the Container, you must add the @Injectable decorator (aka annotation) to your class (see examples above).
  • You may optionally add a @PostConstruct decorator to a method of your class to perform synchronous or asynchronous initialization of a new instance.
  • By default, Async-Injection will examine the parameters of a class constructor and do it's best to match those to a bound InjectableId.
  • You may use the @Inject decorator to explicitly declare which InjectableId should be used (generally required for a Constant binding as in the logLevel example above).
  • The @Optional decorator allows you to specify a default value for a class constructor parameter in the event that no matching InjectableId can be found.
  • The Container's resolveSingletons method may be used to wait for any bound (a)synchronous Singletons to finish initialization before continuing execution of your application.

Acknowledgements

Thanks to all the contributors at InversifyJS. It is a powerful, clean, flexible, inspiring design.

Thanks to everyone at NestJS for giving us Asynchronous providers.

Thanks to Darcy Rayner for describing a DI implementation so simply and clearly.

Thanks to Carlos Delgado for the idea of a "QuerablePromise" which allowed us to blend asynchronous DI with the simplicity of synchronous DI.

MIT License

Copyright (c) 2020-2023 Frank Stock

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

changelog

2.0.1 / 2024-09-11

  • Fix package.json exports #18.
    • Thanks to @IvanLi-CN for finding this, and the solution.

2.0.0 / 2024-08-06

  • Move typedefs up from lib/{cjs|esm} to just lib.
  • Remove binder.ts exports from index.ts

1.6.0 / 2024-05-23

  • Change internal meta-data constants from symbols to namespaced strings. This enables different compilation units to share a common Container. Thanks to @dmtaub for finding this and proposing a solution.
  • Update github build matrix to drop node 14 and add node 20. This does not impact the library or its targets (only the build matrix github uses to run tests).

1.5.5 / 2024-01-18

  • Backwards compatibly binder api updates.
  • Update patch level dependencies.

1.5.4 / 2023-11-15

  • Improved handling of class constructor parameter type recognition.
  • Update dependencies.

1.5.3 / 2023-06-05

  • Address esm issues #10 and #15
  • Fix missing type overload on Binder interface.
  • Update dev dependencies.

1.5.2 / 2023-01-18

  • No code changes.
  • Updates to docs for consistency across projects.

1.5.1 / 2023-01-18

  • Update dev dependencies.
  • Update github workflows.
  • Update badges in main ReadMe.
  • Add support directory.

1.5.0 / 2022-04-16

  • Add experimental ability to clone a Container (see Container.clone JSDoc comments for details).
  • Fix error handling callback to pass instantiated object when construction succeeds but post construction fails.
  • Reformat code project wide (based on IntelliJ formatting options).

1.4.0 / 2022-02-16

  • Add Angular style InjectionToken class as a variant of InjectableId to support implicit typing of constants and interfaces.
  • Minor update to Binder.resolveSingletons to make it chainable (aka return the Container/Binder instance).
  • Minor updates to ReadMe.

1.3.0 / 2021-11-27

  • Support Container driven release of Singleton allocated resources (see Container.releaseSingletons).
  • Update devDependencies.
  • Minor updates to ReadMe.

1.2.7 / 2021-08-02

  • Revert type declaration for AbstractConstructor which was broken during eslint integration.
  • Update eslint related dev-dependencies.

1.2.6 / 2021-07-14

  • Merge PR #9 ESLINT integration + Improvements.
  • Update devDependencies.
  • Resolved a couple of eslint warnings.
  • tsc no longer removes comments in generated code. This can cause problems with post-processing tools such as istanbul. If file size is of concern to you, you should probably be minifying anyway.

1.2.5 / 2021-06-28

  • No actual source code changes.
  • Added Reflect type from reflect-metadata in order to remove the @ts-ignore comments.
  • Improved tsconfig.json structure for IDE compatibility.
    ** Thanks to @tripodsgames for those contributions.
  • Update tsc devDependency from 4.3.3 to 4.3.4.
  • Update the ChangeLog to properly reflect recent GitHub releases.

1.2.4 / 2021-06-17

  • Build esm into esm dir (not mjs).
  • No actual source code changes.

1.2.3 / 2021-06-11

  • cjs and esm distributions.
  • Build now generates both cjs and esm distributions.
  • tslib (where used) is now inlined instead of imported.
  • No other code changes.

1.2.0 / 2021-06-08

  • New Feature: Allow alternate polyfill for Reflect API
    WARNING: This is a a breaking change release.
    The API and code have not changed, but you will need to explicitly import a polyfill into your own code in order to use this release (see the ReadMe).
    Previously Async-Injection relied on reflect-metadata (which is still supported), but this release also allows for the use of alternative implementations such as:
    core-js (core-js/es7/reflect) reflection Thank you to @tripodsgames for this contribution.

1.1.0 / 2021-05-07

  • Add post construction handling feature to Binder.bindClass. This is for scenarios where it is not feasible to add the @PostConstruct decorator to the target class.
  • Updated tslib
  • Updated jasmine devDependency.

1.0.8 / 2020-06-08

  • Add ability to walk up the parent container hierarchy to methods Injector.isIdKnown and Container.removeBinding
  • Update tslib
  • Update ts-node and nyc devDependencies.

1.0.7 / 2019-04-27

  • Fix issue #1
  • Update ts-node and source-map-support devDependencies
  • Add tslint and Changelog
  • Update ReadMe with badges