Security News
Node.js EOL Versions CVE Dubbed the "Worst CVE of the Year" by Security Experts
Critics call the Node.js EOL CVE a misuse of the system, sparking debate over CVE standards and the growing noise in vulnerability databases.
typed-inject
Advanced tools
Type safe dependency injection for TypeScript
A tiny, 100% type safe dependency injection framework for TypeScript. You can inject classes, interfaces or primitives. If your project compiles, you know for sure your dependencies are resolved at runtime and have their declared types.
If you are new to 'Dependency Injection'/'Inversion of control', please read up on it in this blog article about it
If you want to know more about how typed-inject works, please read my blog article about it
Install typed-inject locally within your project folder, like so:
npm i typed-inject
Or with yarn:
yarn add typed-inject
Note: this package uses advanced TypeScript features. Only TS 3.0 and above is supported!
An example:
import { rootInjector, tokens } from 'typed-inject';
interface Logger {
info(message: string): void;
}
const logger: Logger = {
info(message: string) {
console.log(message);
}
};
class HttpClient {
constructor(private log: Logger) { }
public static inject = tokens('logger');
}
class MyService {
constructor(private http: HttpClient, private log: Logger) { }
public static inject = tokens('httpClient', 'logger');
}
const appInjector = rootInjector
.provideValue('logger', logger)
.provideClass('httpClient', HttpClient);
const myService = appInjector.injectClass(MyService);
// Dependencies for MyService validated and injected
In this example:
logger
is injected into a new instance of HttpClient
by value.HttpClient
and the logger
are injected into a new instance of MyService
.Dependencies are resolved using the static inject
property on their classes. They must match the names given to the dependencies when configuring the injector with provideXXX
methods.
Expect compiler errors when you mess up the order of tokens or forget it completely.
import { rootInjector, tokens } from 'typed-inject';
// Same logger as before
class HttpClient {
constructor(private log: Logger) { }
// ERROR! Property 'inject' is missing in type 'typeof HttpClient' but required
}
class MyService {
constructor(private http: HttpClient, private log: Logger) { }
public static inject = tokens('logger', 'httpClient');
// ERROR! Types of parameters 'http' and 'args_0' are incompatible
}
const appInjector = rootInjector
.provideValue('logger', logger)
.provideClass('httpClient', HttpClient);
const myService = appInjector.injectClass(MyService);
The error messages are a bit cryptic at times, but it sure is better than running into them at runtime.
JavaScript and TypeScript development already has a great dependency injection solution with InversifyJS. However, InversifyJS comes with 2 caveats.
InversifyJS works with a nice API using decorators. Decorators is in Stage 2 of ecma script proposal at the moment of writing this, so will most likely land in ESNext. However, it also is opinionated in that it requires you to use reflect-metadata, which is supposed to be an ecma script proposal, but isn't yet (at the moment of writing this). It might take years for reflect-metadata to land in Ecma script, if it ever does.
InversifyJS is also not type-safe. There is no check to see of the injected type is actually injectable or that the corresponding type adheres to the expected type.
Type safe dependency injection works by combining awesome TypeScript features. Some of those features are:
Please read my blog article on Medium if you want to know how this works.
The Injector
interface is responsible for injecting classes of functions. However, typed-inject
only comes with one implementation: the rootInjector
. It does not provide any dependencies (expect for magic tokens).
In order to do anything useful with the rootInjector
, you'll need to create child injectors. This what you do with the provideXXX
methods.
import { rootInjector, tokens } from 'typed-inject';
function barFactory(foo: number){ return foo + 1};
barFactory.inject = tokens('foo');
class Baz {
constructor(bar: number){ console.log(`bar is: ${bar}`)};
static inject = tokens('bar');
}
const childInjector = rootInjector
.provideValue('foo', 42)
.provideFactory('bar', barFactory)
.provideClass('baz', Baz);
In the example above, a child injector is created. It can provide values for the tokens 'foo'
, 'bar'
and 'baz'
. You can create as many child injectors as you want.
The rootInjector
always remains stateless. So don't worry about reusing it in your tests or reusing it for different parts of your application. However,
any ChildInjector is stateful. For example, it can cache the injected value or keep track of stuff to dispose
You can determine the lifecycle of dependencies with the third Scope
parameter of provideFactory
and provideClass
methods.
function loggerFactory(target: Function | null){
return getLogger((target && target.name) || 'UNKNOWN');
}
loggerFactory.inject('target');
class Foo {
constructor(public log: Logger) { log.info('Foo created'); }
static inject = tokens('log');
}
const fooProvider = injector
.provideFactory('log', loggerFactory, Scope.Transient)
.provideClass('foo', Foo, Scope.Singleton);
const foo = fooProvider.resolve('foo');
const fooCopy = fooProvider.resolve('foo');
const log = fooProvider.resolve('log');
console.log(foo === fooCopy); // => true
console.log(log === foo.log); // => false
A scope has 2 possible values.
Scope.Singleton
(default value)Scope.Singleton
to enable caching. Every time the dependency needs to be provided by the injector, the same instance is returned. Other injectors will still create their own instances, so it's only a Singleton
for the specific injector (and child injectors created from it). In other words,
the instance will be scoped to the Injector
Scope.Transient
Scope.Transient
to completely disable cashing. You'll always get fresh instances.Memory in JavaScript is garbage collected, so usually we don't care about cleaning up after ourselves. However, there might be a need to explicit clean up. For example removing a temp folder, or killing a child process.
As typed-inject
is responsible for creating (providing) your dependencies, it only makes sense it is also responsible for the disposing of them.
Any Injector
has a dispose
method. If you call it, the injector in turn will call dispose
on any instance that was ever created from it (if it has one).
import { rootInjector } from 'typed-inject';
class Foo {
constructor() { console.log('Foo created'); }
dispose(){ console.log('Foo disposed');}
}
const fooProvider = rootInjector.provideClass('foo', Foo);
fooProvider.resolve('foo'); // => "Foo created"
fooProvider.dispose(); // => "Foo disposed"
fooProvider.resolve('foo'); // Error: Injector already disposed
To help you implementing the dispose
method correctly, typed-inject
exports the Disposable
interface for convenience:
import { Disposable } from 'typed-inject';
class Foo implements Disposable {
dispose(){ }
}
Using dispose
on an injector will automatically dispose it's parent injectors as well:
import { rootInjector } from 'typed-inject';
class Foo { }
class Bar { }
const fooProvider = rootInjector.provideClass('foo', Foo);
const barProvider = fooProvider.provideClass('bar', Bar);
barProvider.dispose(); // => fooProvider is also disposed!
fooProvider.resolve('foo'); // => Error: Injector already disposed
Disposing of provided values is done in order of parent first. So they are disposed in the order of respective providedXXX
calls.
Any instance created with injectClass
or injectFactory
will not be disposed when dispose
is called. You were responsible for creating it, so you are also responsible for the disposing of it. In the same vain, anything provided as a value with providedValue
will also not be disposed when dispose
is called on it's injector.
Any Injector
instance can always inject the following tokens:
Token name | Token value | Description |
---|---|---|
INJECTOR_TOKEN | '$injector' | Injects the current injector |
TARGET_TOKEN | '$target' | The class or function in which the current values is injected, or undefined if resolved directly |
An example:
import { rootInjector, Injector, tokens, TARGET_TOKEN, INJECTOR_TOKEN } from 'typed-inject';
class Foo {
constructor(injector: Injector<{}>, target: Function | undefined) {}
static inject = tokens(INJECTOR_TOKEN, TARGET_TOKEN);
}
const foo = rootInjector.inject(Foo);
Note: some generic parameters are omitted for clarity.
Injector<TContext>
The Injector<TContext>
is the core interface of typed-inject. It provides the ability to inject your class or function with injectClass
and injectFunction
respectively. You can create new child injectors from it using the provideXXX
methods.
The TContext
generic arguments is a lookup type. The keys in this type are the tokens that can be injected, the values are the exact types of those tokens. For example, if TContext extends { foo: string, bar: number }
, you can let a token 'foo'
be injected of type string
, and a token 'bar'
of type number
.
Typed inject comes with only one implementation. The rootInjector
. It implements Injector<{}>
interface, meaning that it does not provide any tokens (except for magic tokens). Import it with import { rootInjector } from 'typed-inject'
. From the rootInjector
, you can create child injectors. See creating child injectors for more information.
Don't worry about reusing the rootInjector
in your application. It is stateless and read-only, so safe for concurrent use.
injector.injectClass(injectable: InjectableClass)
This method creates a new instance of class injectable
and returns it.
When there are any problems in the dependency graph, it gives a compiler error.
class Foo {
constructor(bar: number) { }
static inject = tokens('bar');
}
const foo /*: Foo*/ = injector.injectClass(Foo);
injector.injectFunction(fn: InjectableFunction)
This methods injects the function with requested tokens and returns the return value of the function. When there are any problems in the dependency graph, it gives a compiler error.
function foo(bar: number) {
return bar + 1;
}
foo.inject = tokens('bar');
const baz /*: number*/ = injector.injectFunction(Foo);
injector.resolve(token: Token): CorrespondingType<TContext, Token>
The resolve
method lets you resolve tokens by hand.
const foo = injector.resolve('foo');
// Equivalent to:
function retrieveFoo(foo: number){
return foo;
}
retrieveFoo.inject = tokens('foo');
const foo2 = injector.injectFunction(retrieveFoo);
injector.provideValue(token: Token, value: R): Injector<ChildContext<TContext, Token, R>>
Create a child injector that can provide value value
for token 'token'
. The new child injector can resolve all tokens the parent injector can as well as 'token'
.
const fooInjector = injector.provideValue('foo', 42);
injector.provideFactory(token: Token, factory: InjectableFunction<TContext>, scope = Scope.Singleton): Injector<ChildContext<TContext, Token, R>>
Create a child injector that can provide a value using factory
for token 'token'
. The new child injector can resolve all tokens the parent injector can, as well as the new 'token'
.
With scope
you can decide whether the value must be cached after the factory is invoked once. Use Scope.Singleton
to enable caching (default), or Scope.Transient
to disable caching.
const fooInjector = injector.provideFactory('foo', () => 42);
function loggerFactory(target: Function | undefined) {
return new Logger((target && target.name) || '');
}
loggerFactory.inject = tokens(TARGET_TOKEN);
const fooBarInjector = fooInjector.provideFactory('logger', loggerFactory, Scope.Transient)
injector.provideFactory(token: Token, Class: InjectableClass<TContext>, scope = Scope.Singleton): Injector<ChildContext<TContext, Token, R>>
Create a child injector that can provide a value using instances of Class
for token 'token'
. The new child injector can resolve all tokens the parent injector can, as well as the new 'token'
.
Scope is also supported here, for more info, see provideFactory
.
injector.dispose()
Use dispose
to explicitly dispose the injector
. It will in turn call dispose
on it's parent injector as well as calling dispose
on any dependency created by the injector (if it exists) using provideClass
or provideFactory
(not provideValue
or injectXXX
).
After a child injector is disposed, you cannot us it any more. Any attempt to use it will result in a Injector already disposed
error.
The rootInjector
will never be disposed.
Scope
The Scope
enum indicates the scope of a provided injectable (class or factory). Possible values: Scope.Transient
(new injection per resolve) or Scope.Singleton
(inject once, and reuse values). It generally defaults to Singleton
.
tokens
The tokens
function is a simple helper method that makes sure that an inject
array is filled with a tuple type filled with literal strings.
const inject = tokens('foo', 'bar');
// Equivalent to:
const inject: ['foo', 'bar'] = ['foo', 'bar'].
Note: hopefully TypeScript will introduce explicit tuple syntax, so this helper method can be removed
InjectableClass<TContext, R, Tokens extends InjectionToken<TContext>[]>
The InjectableClass
interface is used to identify the (static) interface of classes that can be injected. It is defined as follows:
{
new(...args: CorrespondingTypes<TContext, Tokens>): R;
readonly inject: Tokens;
}
In other words, it makes sure that the inject
tokens is corresponding with the constructor types.
InjectableFunction<TContext, R, Tokens extends InjectionToken<TContext>[]>
Comparable to InjectableClass
, but for (non-constructor) functions.
Disposable
You can implement the Disposable
interface in your dependencies. It looks like this:
interface Disposable {
dispose(): void;
}
With this, you can let the Injector
call your dispose method.
Note: This is just a convenience interface. Due to TypeScripts structural typing system typed-inject
calls your dispose
method without you having to explicitly implement it.
This entire framework would not be possible without the awesome guys working on TypeScript. Guys like Ryan, Anders and the rest of the team: a heartfelt thanks! 💖
Inspiration for the API with static inject
method comes from years long AngularJS development. Special thanks to the Angular team.
FAQs
Type safe dependency injection framework for TypeScript
The npm package typed-inject receives a total of 60,843 weekly downloads. As such, typed-inject popularity was classified as popular.
We found that typed-inject demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 2 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Critics call the Node.js EOL CVE a misuse of the system, sparking debate over CVE standards and the growing noise in vulnerability databases.
Security News
cURL and Go security teams are publicly rejecting CVSS as flawed for assessing vulnerabilities and are calling for more accurate, context-aware approaches.
Security News
Bun 1.2 enhances its JavaScript runtime with 90% Node.js compatibility, built-in S3 and Postgres support, HTML Imports, and faster, cloud-first performance.