Skip to content

youngspe/checked-inject

Repository files navigation

checked-inject

Documentation

npm install --save checked-inject

checked-inject is a TypeScript dependency injection library that verifies all dependencies are met at compile time.

class NameKey extends TypeKey<string>() { private _: any }
class IdKey extends TypeKey<number>() { private _: any }

class User {
  name: string
  id: number
  constructor(name: string, id: number) {
    this.name = name; this.id = id
  }

  static inject = Inject.construct(this, NameKey, IdKey)
}

class App {
  user: User
  constructor(user: User) {
    this.user = user
  }
}

const UserModule = Module(ct => ct
  .provideInstance(NameKey, 'Alice')
  .provideInstance(IdKey, 123)
)

const AppModule = Module(UserModule, ct => ct
  .provide(App, { user: User }, ({ user }) => new App(user))
)

AppModule.inject({ app: App }, ({ app }) => {
  console.log(`Welcome, ${app.user.name}`)
})

Injection

Containers

A Container is used to provide injections and request resources, solving the dependency graph at compile time and runtime. An empty container is obtained through Container.create()

Providing Resources

You can build a container by creating it, then providing resources in a method chain. It's important to use the last container returned from the method chain so the type system is aware of the full dependency graph.

In the following methods, key is either a TypeKey or a class.

provideInstance(key, value) specifies a value to directly return when the given key is requested.

provide(key, [scope], dep, init) supplies a function that will be called when the given key is requested. dep specifies the dependencies of the resource you're providing. The dependencies are passed to the function init, which returns the resources' value. scope is an optional parameter. It can either be a Scope or an arbitrarily-nested list of Scopes. When provided, the provided resource is bound to the given Scope(s).

provider(key, [scope], computedKey) supplies a ComputedKey that resolves to the resource to be returns when key is requested.

const container = Container.create()
  .provideInstance(NameKey, 'Alice')
  .provideInstance(IdKey, 123)
  .provide(User,
    { name: NameKey, id: IdKey }, ({ name, id }) => new User(name, id))
  .provide(Foo, Inject.map(IdKey, id => new Foo(id)))

Requesting Resources

// 'request' returns the resource identified by the given DependencyKey
let user: User = container.request(User)
// 'inject' calls the given function with the resource as an argument
container.inject(User, user => console.log(user.name, user.id))

Compile-Time Checks

This library is centered around verifying all dependencies are met within the type system.

If you request a resource and any of its transitive dependencies are unavailable, there will be a compile error ending with something like: Type 'void' is not assignable to type 'typeof Foo | typeof Bar'. where Foo and Bar are resources that are needed but not provided.

Scopes, Child Containers, and Subcomponents

A child container is a container that inherits all of its parent's provided resources. It can be created with Container#createChild.

A container can be assigned one or more Scopes, and a provided resource can be associated with one or more Scopes. When a resource is scoped, its value is stored in the nearest container in the ancestor chain that either:

  • Has a provider for the resource
  • Contains one or more of the requested scopes

Because of this rule, if you provide a scoped resource to a child container of the one assigned the scope, the stored value will not be visible to consumers of the parent container or its sibling containers, and it will be safe to depend on resources also provided to the child container:

In the following example, it's safe for the 'Baz' to reference 'Bar' even though 'Bar' isn't in the parent container or bound to 'MyScope because it's provided in the child container rather than the parent:

const parent = Container.create()
  .addScope(MyScope)
  .provide(Foo, {}, () => new Foo())

const child = parent.createChild()
  .provide(Bar, { foo: Foo }, ({ foo }) => new Bar(foo))
  .provide(Baz, MyScope, { bar: Bar }, ({ bar }) => new Baz(bar))

const baz1 = child.request(Baz)
const baz2 = child.request(Baz)

console.log(baz1 === baz2) // prints 'true'

This example will fail to compile with: Type 'void' is not assignable to type 'typeof Bar'. because Bar is not found in MyScope from the container where Baz is provided:

const parent = Container.create()
  .addScope(MyScope)
  .provide(Foo, {}, () => new Foo())
  .provide(Baz, MyScope, { bar: Bar }, ({ bar }) => new Baz(bar))

const child = parent.createChild()
  .provide(Bar, { foo: Foo }, ({ foo }) => new Bar(foo))

const baz = child.request(Baz)

If you request a resource and any Scope it's bound to is not assigned to this container or its ancestors, there will be a compile error ending with something like: Type 'void' is not assignable to type 'typeof MyScope'. where MyScope is a Scope a dependency is bound to but is not available.

Singleton

Singleton is a built-in scope that's automatically assigned to a new Container. Use Singleton when you only want to create one instance of a resource. If you want an instance to have a shorter lifetime, you can create a custom Scope.

Custom Scopes

To ensure each Scope has its own distinct type, a Scope is declared as a class extending Scope() with at least one private member:

class MyScope extends Scope() { private _: any }

Subcomponents

A Subcomponent transforms a list of arguments into a new child container.

The Inject.subcomponent method takes a lambda that initializes a child container given a list of arguments. It returns a ComputedKey that resolves to a Subcomponent.

class UserScope extends Scope() { private _: any }
const UserComponent = Inject.subcomponent((ct, name: string, id: number ) => ct
  // UserScope allows us to cache dependencies that are only valid when a name
  // and id are available
  .addScope(UserScope)
  .provideInstance(NameKey, name)
  .provideInstance(IdKey, id)
)

const parent = Container.create()
  // Binding the 'User' resource to 'UserScope' allows us to cache a single
  // intance and return the same one every time 'User' is requested.
  // 'Singleton' would be insufficient because 'NameKey' and 'IdKey' are
  // provided in a child container, so they are unavailable in the Singleton
  // scope.
  .provide(User, UserScope, Inject.construct(NameKey, IdKey))

// This will cause a compile error because `UserScope' is unavailable:
// const user1 = parent.request(User)

// The following three lines are equivalent:
const child1 = parent.request(UserComponent)('Alice', 123)
const child2 = parent.request(UserComponent.Build('Alice', 123))
const child3 = parent.build(UserComponent, 'Alice', 123)

// This is okay because child1 is in 'UserScope', and both 'NameKey' and 'IdKey'
// are provided:
const user2 = child1.request(User)

Modules

You can group providers into Modules using the Module() function. This function takes a lambda that accepts a Container.Builder to which you can provide resources just like a Container. Container#apply(module) applies all providers in the Module to the container. Modules also have a container() method that returns a new container with the Module applied to it. Additionally, Modules have request[Async] and inject[Async] methods that internally creates a Container and calls the respective Container method.

const UserModule = Module(ct => ct
  .provideInstance(NameKey, 'Alice')
  .provideInstance(IdKey, 123)
  .provide(User, Inject.construct(User, NameKey, IdKey))
)

// These two lines are equivalent:
const container1 = Container.create().apply(UserModule)
const container2 = UserModule.container()

// You also call 'request[Async]' or 'inject[Async]' without directly creating
// a 'Container':
UserModule.inject(User, user => {
  console.log(user.name, user.id)
})

Modules can be combined into a single module:

const AppModule = Module(UserModule, DataModule, FooModule)

AppModule.inject(App, app => {
  // ...
})

Asynchronous Resources

Not all resources are available immediately. Using Container#provideAsync, you can provide a Promise that will eventually resolve to the provided resource. Any resource depending on an asynchronous resource is itself resolved asynchronously.

Asynchronous resources can be resolved with Container#requestAsync or Container#injectAsync. These methods are analogous to their non-async counterparts. The only differences are:

  • The async versions can resolve asynchronous resources
  • They return a Promise rather than directly returning the value
  • injectAsync's function can be an async function
let container = Container.create()
  .provideInstance(NameKey, 'Alice')
  .provideAsync(IdKey, {}, async () => 123)
  .provide(User, Inject.construct(User, NameKey, IdKey))

// This is okay since 'NameKey' is available synchronously:
let name1: string = container.request(NameKey)

// This will fail since 'IdKey' is an asynchronous resource:
// let id1: number = container.request(IdKey)

// This will fail since 'User' depends on 'IdKey', which is asynchronous:
// let user1: User = container.request(User)

// The following are all allowed:
let name2: Promise<string> = container.requestAsync(NameKey)
let id2: Promise<number> = container.requestAsync(IdKey)
let user2: Promise<User> = container.requestAsync(User)

To synchronously resolve a Promise for the resource rather than asynchronously waiting for the resource to complete, use Inject.async or (TypeKey|ComputedKey|typeof Injectable)#Async():

let user1: Promise<User> = container.request(Inject.async(User))
// This works if 'User' extends 'Injectable':
let user2: Promise<User> = container.request(User.Async())

If you request an asynchronous resource synchronously, there will be a compile error ending with something like: Type 'void' is not assignable to type 'NotSync<typeof Foo>'. where Foo is an asynchronous resource.

DependencyKey

A DependencyKey identifies a resource that can be injected. It may be a dependency of another resource or requested directly from a container.

There are multiple kinds of DependencyKeys:

Class

An InjectableClass<T> is any class object that optionally has a static scope: extends ScopeList and/or a static inject: extends ComputedKey <T> property.

Examples

class User {
  name: string
  id: number
  constructor(name: string, id: number) {
    this.name = name; this.id = id
  }
}
class UserManager {
  private userApiClient: UserApiClient
  constructor(userApiClient: UserApiClient) {
    this.userApiClient
  }
  // If 'UserManager' is not explicitly provided to the container, the value of
  // 'inject' will be used to resolve it instead.
  static inject = Inject.construct(this, UserApiClient)
}
abstract class UserApiClient {
  abstract getUser(id: number): Promise<User>

  // Instances of 'UserApiClient' will be stored and reused for all containers
  // marked with the scope called 'UserScope'
  static scope: UserScope
}
class ApiConfig {
  appId: string
  apiKey: string

  // All root containers are automatically marked with 'Singleton', so a single
  // instance of 'ApiConfig' will be created and reused within the container
  // it's provided to.
  static scope: Singleton
}

Injectable

Though optional, classes can extend Injectable, which does the following:

  • Prevent static scope from being assigned a value that does not extend ScopeList
  • Prevent static inject from being assigned a value that does not extend ComputedKey <T>
  • Add static operator methods equivalent to those on TypeKey and ComputedKey and analogous to the methods on Inject like Lazy(), Provider(), Async(), Optional()
class User extends Injectable {
  name: string
  id: number
  constructor(name: string, id: number) {
    super(); this.name = name; this.id = id
  }
}

// Later, we can use operators like 'Optional()' when requesting 'User':

const MyModule = Module(ct => ct
  .provide(Foo, { user: User.Optional() }, ({ user }) => new Foo(user))
)

TypeKey

A TypeKey specifies a resource not tied to a specific class object--like a named dependency.

To ensure each TypeKey has its own distinct type, a TypeKey<T> is declared as a class extending TypeKey<T>() with at least one private member:

class NameKey extends TypeKey<string>() { private _: any }
class IdKey extends TypeKey<number>() { private _: any }

// You can set a 'ComputedKey' like `Inject.map(...)' as a default provider.
// If 'CurrentUser' is not explicitly provided to a container, the default
// provider will be used to resolve it.
class CurrentUserKey extends TypeKey({
  default: Inject.map([NameKey, IdKey], ([name, id]) => new User(name, id)),
}) { private _: any }

A TypeKey may also have a static scope: extends ScopeList that binds its resource to one or more scopes:

class MyKey extends TypeKey<string>() {
  private _: any
  static scope = Singleton
}

Structured Keys:

DependencyKeys can also be structured into arrays or objects of the form DependencyKey[] or { [k: string]: DependencyKey } like so:

const [name, id, user] = container.request([NameKey, IdKey.Provider(), User])
const { name, id, user } = container.request({
  name: NameKey,
  id: IdKey.Provider(),
  user: User,
})

If you want you can even nest structured keys:

const { userInfo: { name, id } }  = container.request({
  userInfo: { name: NameKey, id: IdKey },
})

ComputedKey

A ComputedKey transforms a dependency into another type. Implementations are found in the Inject namespace.

These methods can operate over any DependencyKey, even structured keys:

const f = container.request(Inject.provider({
  name: NameKey,
  id: IdKey,
}))

const { name, id } = f()

Some common ComputedKeys include the following:

Lazy, Provider

Inject.lazy(src) and Inject.provider(src) both resolve to a function returning the target type of src. lazy caches the result after the first call, whereas provider resolves the resource every time it's called. TypeKeys, ComputedKeys, and class objects that extends Injectable have analogous methods called Lazy() and Provider().

Because the resulting function returns synchronously, these methods can only be used on dependencies that can be resolved synchronously. To resolve asynchronously, use Inject.async(MyKey).Lazy()/Provider() or (if MyKey is a TypeKey or class object that extends Injectable) MyKey.Async().Lazy()/Provider().

This will yield a value of type () => Promise<Target<MyKey>>.

Async

Inject.async(src) resolves the given dependency as a Promise. This allows you to request dependencies that cannot be resolved synchronously without having to use requestAsync or injectAsync. Instead the dependent resource can await the promises independently.

See also Async()

class UserInfo {
  name: string
  private _id: Promise<number>
  getIdAsync() { return _id }

  constructor(name: string, id: Promise<number>) {
    this.name = name; this._id = id
  }
}

const ct = Container.create()
//.provide(UserService, ...)
  .provideAsync(IdKey,
    { service: UserService}, ({ service }) => service.getIdAsync())
  .provide(UserInfo, Inject.construct(UserInfo, NameKey, IdKey.Async()))

// This would be a compile error because IdKey is not provided synchronously:
// const id = container.request(IdKey)

// This is okay because Async() provides the value synchronously as a promise:
let id = await container.request(IdKey.Async())
// It's equivalent to this:
id = await container.requestAsync()

// This is okay even though it requires IdKey because it uses Async():
const userInfo = container.request(UserInfo)

// userInfo is created but we still have to await IdKey's promise to get the id:
id = await userInfo.getIdAsync()

Map

Inject.map(dep, transform) resolves to a transformation over another dependency.

See also Map(transform).

const user: User = container.request(Inject.map({
  name: NameKey,
  id: IdKey,
}), ({ name, id}) => new User(name, id))

This is useful for default injections:

class User {
  name: string
  id: number
  constructor(name: string, id: number) {
    this.name = name; this.id = id
  }

  static inject = Inject.map({
    name: NameKey,
    id: IdKey,
  }, ({ name, id }) => new User(name, id))
}

class UserKey extends TypeKey({
  default: Inject.map({
    name: NameKey,
    id: IdKey,
  }, ({ name, id }) => new User(name, id))
}) { private _: any }

Construct

Inject.construct(ctor, ...deps) resolves to the instantiation of the given class contructor given the following dependencies. It's equivalent to Inject.map(ctor, [...deps], ([...args]) => new ctor(...args)).

const user: User = container.request(Inject.construct(User, NameKey, IdKey))

This is useful for default injections:

class User {
  name: string
  id: number
  constructor(name: string, id: number) {
    this.name = name; this.id = id
  }

  static inject = Inject.construct(this, NameKey, IdKey)
}

class UserKey extends TypeKey({
  default: Inject.construct(User, NameKey, IdKey)
}) { private _: any }

or dependency providers:

const container = Container.create()
  .provide(User, Inject.construct(User, NameKey, IdKey))

Target Types

Target<K>, where K extends DependencyKey, indicates what type the K resolves to when requested from a container:

KindKey Target Type

TypeKey <string>

NameKey string

TypeKey <number>

IdKey number

InjectableClass <User>

User User

ComputedKey

One of:

NameKey.Provider()
Inject.provider(NameKey)

() => string

ComputedKey

One of:

IdKey.Map(id => id.toString())
Inject.map(IdKey, id => id.toString())

string

ComputedKey

One of:

Inject.async(User).Lazy()
Inject.lazy(Inject.async(User))
// If 'User' extends 'Injectable':
User.Async().Lazy()

() => Promise<User>

Object key
{
  name: NameKey,
  id: IdKey.Provider(),
  user: User,
}
{
  name: string,
  id: () => number,
  user: User,
}
Array key
[NameKey, IdKey.Provider(), User]
[string, () => number, User]

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published