# Lazy-loading provider

6.81.0+

By default, modules are eagerly loaded, which means that as soon as the application loads, so do all the modules, whether or not they are immediately necessary. While this is fine for most applications, it may become a bottleneck for apps running in the serverless environment, where the startup latency ("cold start") is crucial.

Lazy loading can help decrease bootstrap time by loading only modules required by the specific serverless function invocation. In addition, you could also load other modules asynchronously once the serverless function is "warm" to speed-up the bootstrap time for subsequent calls even further (deferred modules registration).

# Getting started

To load a provider on-demand, Ts.ED provide decorators LazyInject and OptionalLazyInject . Here is an example with a PlatformExceptions :

import {Injectable, LazyInject} from "@tsed/di";
import type {PlatformExceptions} from "@tsed/platform-exceptions";

@Injectable()
class MyInjectable {
  @LazyInject("PlatformException", () => import("@tsed/platform-exceptions"))
  private platformExceptions: Promise<PlatformExceptions>;

  async use() {
    try {
      /// do something
    } catch (er) {
      const exceptions = await this.platformExceptions;
      platformExceptions.catch(er, {});
    }
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

The LazyInject decorator will load the node_module and invoke automatically the PlatformException exported class, only when the decorated property will be used by your code.

TIP

Lazy-loaded provider is cached on the first call. That means, each consecutive call will be very fast and will return a cached instance, instead of loading the provider again.

Create you own lazy injectable doesn't require special things, just declare a module or an injectable service is enough:

import {Module} from "@tsed/di";

@Module({
  // works also with @Injectable
  imports: [] // Use the imports field if you have services to build
})
export class MyModule {
  $onInit() {
    // The hook will be called once the module is loaded
  }
}
1
2
3
4
5
6
7
8
9
10
11

Then use it:

import {Injectable, LazyInject} from "@tsed/di";
import type {MyModule} from "../MyModule.ts";

@Injectable()
class MyInjectable {
  @LazyInject("MyModule", () => import("../MyModule.ts"))
  private myModule: Promise<MyModule>;

  async use() {
    const myModule = await this.myModule;

    myModule.doSomething();
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14

WARNING

If you use Webpack, make sure to update your tsconfig.json file - setting compilerOptions.module to "esnext" and adding compilerOptions.moduleResolution property with "node" as a value:

{
  "compilerOptions": {
    "module": "esnext",
    "moduleResolution": "node"
  }
}
1
2
3
4
5
6

# Lazy loading programmatically

Ts.ED provide also a way to lazy load a provider programmatically. You just need to inject the InjectorService in service:

import {Injectable, Inject, InjectorService} from "@tsed/di";

@Injectable()
class MyService {
  @Inject()
  protected injector: InjectorService;

  async load() {
    const {MyModule} = await import("../lazy/my-module.ts");
    const myModule = await this.injector.lazyInvoke(MyModule);

    myModule.doSomething();
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14

# Limitation

Some providers cannot be lazy loaded:

  • Controllers,
  • Middlewares,
  • All providers that need to run a specific hook (excepted $onInit hook).

Last Updated: 9/9/2024, 7:14:58 AM

Other topics