@golevelup/nestjs-discovery
Version:
A Badass NestJS module for querying your app's controllers, providers and handlers
116 lines (82 loc) • 4.49 kB
Markdown
# Discovery
This module provides access to the `DiscoveryService` which can be used to query the various modules, providers, controllers and handlers that make up your NestJS application.
<div style="display: flex; gap: 10px;">
<a href="https://www.npmjs.com/package/@golevelup/nestjs-discovery"><img src="https://img.shields.io/npm/v/@golevelup/nestjs-discovery.svg?style=flat" alt="version" /></a>
<a href="https://www.npmjs.com/package/@golevelup/nestjs-discovery"><img alt="downloads" src="https://img.shields.io/npm/dt/@golevelup/nestjs-discovery.svg?style=flat"></a>
<img alt="license" src="https://img.shields.io/npm/l/@golevelup/nestjs-discovery.svg">
</div>
## Getting Started
::: code-group
```bash [npm]
npm install ---save /nestjs-discovery
```
```bash [yarn]
yarn add /nestjs-discovery
```
```bash [pnpm]
pnpm add /nestjs-discovery
```
:::
## Import
Import and add `DiscoveryModule` to the `imports` section of the module you wish to implement Discovery features in. It's common to inject it directly into consuming Module's contructor so that it can be used during the `onModuleInit` lifecycle hook at application startup.
```typescript
import { DiscoveryModule } from '@golevelup/nestjs-discovery';
import { Module } from '@nestjs/common';
({
imports: [DiscoveryModule],
})
export class ExampleModule implements OnModuleInit {
constructor(private readonly discover: DiscoveryService) {}
public async onModuleInit() {
// const providers = await this.discover.providersWithMetaAtKey<number>('metaKey')
}
}
```
## Discover
The `DiscoveryService` exposes several different querying patterns for your app's components that are [well documented with comments](src/discovery.service.ts). This will also provide intellisense for querying in a TypeScript compatible IDE.
In the case of querying for `providers` or `controllers`, the service returns the following interfaces:
```typescript
export interface DiscoveredModule {
name: string;
instance: {};
injectType?: Type<{}>;
dependencyType: Type<{}>;
}
export interface DiscoveredClass extends DiscoveredModule {
parentModule: DiscoveredModule;
}
```
This gives access to the (singleton) `instance` of the matching provider or controller created by the NestJS Dependency Injection container.
The `injectType` can contain the constructor function of the provider token if it is provided as an class. In the case of custom providers, this value will either contain the type of the factory function that created the dependency, or undefined if a value was directly provided with `useValue`.
The `dependencyType` is a shortcut to retrieve the constructor function of the actual provided dependency itself. For providers/controllers this will simply be the decorated class but for dyanmic providers it will return the constructor function of whatever dependency was actually returned from `useValue` or `useFactory`.
It also provides the string based name for convenience. A `DiscoveredClass` contains a `parentModule` which provides the same set of information for the `` class that the dependency was discovered in.
When querying for methods on `providers` or `controllers` the following interface is returned:
```typescript
export interface DiscoveredMethod {
handler: (...args: any[]) => any;
methodName: string;
parentClass: DiscoveredClass;
}
```
This gives access to the `handler` which is the actual class method implementation as well as the ability to navigate back up the dependency tree with the attached `parentClass`.
When specifically querying for components in the context of looking for decorator metadata, the `...WithMetaAtKey<T>` service methods return the types above along with the metadata that was discovered.
```typescript
export interface DiscoveredMethodWithMeta<T> {
discoveredMethod: DiscoveredMethod;
meta: T;
}
export interface DiscoveredClassWithMeta<T> {
discoveredClass: DiscoveredClass;
meta: T;
}
```
## Example
Assuming you were using a custom decorator in your application that attached metadata at a key called `exampleKey`:
```typescript
const ExampleDecorator = (meta: string) => SetMetadata('exampleKey', meta);
```
Find all controller methods that have been decorated with `` and retrieve the value they set for meta:
```typescript
const exampleMethodsMeta =
await this.discover.controllerMethodsWithMetaAtKey<string>('exampleKey');
```