Skip to content

Commit 35bcb1b

Browse files
authored
Merge pull request #37 from tiny-nestjs/dev
release: `v0.3.1`
2 parents 242ddaa + 7d08ce4 commit 35bcb1b

14 files changed

+284
-129
lines changed

.eslintrc.js

+2-4
Original file line numberDiff line numberDiff line change
@@ -5,17 +5,15 @@ module.exports = {
55
sourceType: 'module',
66
},
77
plugins: ['@typescript-eslint/eslint-plugin'],
8-
extends: [
9-
'plugin:@typescript-eslint/recommended',
10-
'plugin:prettier/recommended',
11-
],
8+
extends: ['plugin:@typescript-eslint/recommended', 'plugin:prettier/recommended'],
129
root: true,
1310
env: {
1411
node: true,
1512
jest: true,
1613
},
1714
ignorePatterns: ['.eslintrc.js'],
1815
rules: {
16+
'@typescript-eslint/ban-types': 'off',
1917
'@typescript-eslint/ban-ts-comment': 'off',
2018
'@typescript-eslint/interface-name-prefix': 'off',
2119
'@typescript-eslint/explicit-function-return-type': 'off',

README.md

+89-9
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
</p>
44

55
<div align="center">
6-
<img src="https://img.shields.io/badge/npm-v0.3.0-blue" alt="npm version">
6+
<img src="https://img.shields.io/badge/npm-v0.3.1-blue" alt="npm version">
77
<img src="https://img.shields.io/badge/license-MIT-green" alt="License">
88
</div>
99

@@ -17,9 +17,10 @@ With this library, you can inject dependencies into classes without the need for
1717

1818
## Features
1919

20-
- `@ComponentScan()` decorator enables automatic scanning and injection of classes within a module.
21-
- `@AutoInjectable()` decorator allows classes to be automatically injectable for DI.
22-
- `@AutoController()` decorator automatically registers controllers.
20+
- `@ComponentScan()` enables automatic scanning and injection of classes within a module.
21+
- `@AutoInjectable()` allows classes to be automatically injectable for DI.
22+
- `@AutoController()` automatically registers controllers.
23+
- `@AutoAlias()` defines an alias for the @AutoInjectable() class.
2324

2425
## Installation
2526

@@ -33,7 +34,7 @@ yarn add @tiny-nestjs/auto-injectable
3334

3435
## Usage
3536

36-
**1. `@ComponentScan()`**
37+
**1. `@ComponentScan()` basic usage**
3738

3839
```ts
3940
import { Module } from '@nestjs/common';
@@ -49,7 +50,7 @@ export class AppModule {
4950
}
5051
```
5152

52-
By applying the `@ComponentScan()` decorator to the AppModule class, Nest will automatically scan for classes and
53+
By applying the `@ComponentScan()` decorator to the `AppModule` class, Nest will automatically scan for classes and
5354
inject necessary dependencies.
5455

5556
**2. `@AutoInjectable()`**
@@ -63,8 +64,9 @@ export class CatService {
6364
}
6465
```
6566

66-
In this case, by applying the `@AutoInjectable()` decorator to the CatService class, the class has become injectable,
67-
allowing it to be injected into other modules without the need for module definitions.
67+
In this case, by applying the `@AutoInjectable()` decorator to the `CatService` class, the class has become injectable,
68+
allowing it to be injected into other modules without the need for module definitions. (The parameter
69+
of `@AutoInjectable()` is the same as `@Injectable()`)
6870

6971
**3. `@AutoController()` and dependency injection**
7072

@@ -84,11 +86,89 @@ export class CatController {
8486
```
8587

8688
The class with the `@AutoInjectable()` decorator has been successfully injected and `/cats` api can be accessed by
87-
applying `@AutoController()` on `CatController` service.
89+
applying `@AutoController()` on `CatController` service. (The parameter of `@AutoController()` is the same
90+
as `@Controller()`)
8891

8992
| You can see actual [project example](https://github.com/tiny-nestjs/auto-injectable-example) here. |
9093
|----------------------------------------------------------------------------------------------------|
9194

95+
<br>
96+
97+
---
98+
99+
<br>
100+
101+
- _Below are advanced usage techniques of the library. In most cases, utilizing the methods above will suffice._
102+
103+
<br>
104+
105+
**4. `@AutoAlias()`**
106+
107+
```ts
108+
import { AutoAlias } from '@tiny-nestjs/auto-injectable';
109+
import { AutoInjectable } from '@tiny-nestjs/auto-injectable';
110+
111+
@AutoAlias('kitty')
112+
@AutoInjectable()
113+
export class CatService {
114+
// ...
115+
}
116+
```
117+
118+
```ts
119+
import { Inject } from '@nestjs/common';
120+
import { AutoController } from '@tiny-nestjs/auto-injectable';
121+
122+
@AutoController()
123+
export class CatController {
124+
constructor(@Inject('kitty') private readonly catService: CatService) {
125+
}
126+
}
127+
```
128+
129+
`@AutoAlias()` is a decorator used to specify an alias. In the constructor of the `CatService` class, `@Inject('kitty')`
130+
is used to configure the injection of a `CatService` instance with the alias 'kitty'.
131+
As the library is fully compatible with the `Nest` core, you can use `Nest`'s built-in `@Inject()` decorator.
132+
133+
**5. Define DI scope with the `@ComponentScan()`**
134+
135+
```ts
136+
import { Module } from '@nestjs/common';
137+
import { ComponentScan } from '@tiny-nestjs/auto-injectable';
138+
139+
@ComponentScan()
140+
@Module({})
141+
export class AnimalModule {
142+
}
143+
```
144+
145+
The library recommends using `@ComponentScan()` in the AppModule. However, to enable seamless DI within the desired
146+
scope, you can also specify `@ComponentScan()` in other modules.
147+
148+
**6. `@ComponentScan()` parameters**
149+
150+
```bash
151+
# normal case
152+
- /cat
153+
- cat.module.ts
154+
- ...
155+
```
156+
157+
```bash
158+
# special case
159+
- /animal
160+
- /cat
161+
- /module
162+
- cat.module.ts
163+
- /service
164+
- cat.service.ts
165+
```
166+
167+
In most cases, the module is positioned at the top-level directory of its domain. However, in some cases, you can
168+
specify the exact directory path as an array parameter, such as `@ComponentScan(['animal/cat'])`.
169+
170+
<br>
171+
92172
## Contribution
93173

94174
To contribute to this library, fork the [GitHub repository](https://github.com/tiny-nestjs/auto-injectable), make your

lib/auto.module.ts

+2-13
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,10 @@
11
import { DynamicModule, Module } from '@nestjs/common';
2-
import { Importer, ImporterSync } from './core';
2+
import { Importer } from './core';
33

44
@Module({})
55
export class AutoModule {
6-
static async forRootAsync(patterns: string[]): Promise<DynamicModule> {
7-
const classes = await Importer.load(patterns);
8-
9-
return {
10-
module: AutoModule,
11-
controllers: [...classes.controllers],
12-
providers: [...classes.providers],
13-
exports: [...classes.providers],
14-
};
15-
}
16-
176
static forRoot(patterns: string[]): DynamicModule {
18-
const classes = ImporterSync.load(patterns);
7+
const classes = Importer.load(patterns);
198

209
return {
2110
module: AutoModule,

lib/core/import-sync.ts

-59
This file was deleted.

lib/core/importer.ts

+69-30
Original file line numberDiff line numberDiff line change
@@ -1,54 +1,93 @@
1-
import { glob } from 'glob';
1+
/* eslint-disable @typescript-eslint/no-var-requires */
2+
import { globSync } from 'glob';
23
import { resolve } from 'path';
3-
import { AUTO_CONTROLLER_WATERMARK, AUTO_INJECTABLE_WATERMARK } from '../interfaces';
4-
import 'reflect-metadata';
4+
import {
5+
AUTO_ALIAS_WATERMARK,
6+
AUTO_CONTROLLER_WATERMARK,
7+
AUTO_INJECTABLE_WATERMARK,
8+
COMPONENT_SCAN_WATERMARK,
9+
} from '../interfaces';
10+
import { Logger, Provider } from '@nestjs/common';
11+
import { Type } from '@nestjs/common/interfaces/type.interface';
12+
import { DynamicModule } from '@nestjs/common/interfaces/modules/dynamic-module.interface';
13+
import { ForwardReference } from '@nestjs/common/interfaces/modules/forward-reference.interface';
14+
import { Abstract } from '@nestjs/common/interfaces/abstract.interface';
15+
import { locate } from 'func-loc';
516

617
type ClassType = new (...args: any[]) => any;
718

819
interface AutoClasses {
9-
providers: ClassType[];
10-
controllers: ClassType[];
20+
providers: Provider[];
21+
controllers: Type[];
22+
exports: Array<
23+
DynamicModule | Promise<DynamicModule> | string | symbol | Provider | ForwardReference | Abstract<any> | Function
24+
>;
1125
}
1226

1327
export class Importer {
14-
private static instance: Importer;
28+
private rootPath = '';
1529

16-
static getInstance(): Importer {
17-
if (!Importer.instance) {
18-
Importer.instance = new Importer();
19-
}
20-
return Importer.instance;
21-
}
22-
23-
static async load(patterns: string[]): Promise<AutoClasses> {
24-
const importer = Importer.getInstance();
25-
const pathNames = await importer.matchGlob(patterns);
26-
const foundClasses = await Promise.all(pathNames.map((pathName) => importer.scan(pathName)));
30+
static load(patterns: string[]): AutoClasses {
31+
const importer = new Importer();
32+
const pathNames = importer.matchGlob(patterns);
33+
const foundClasses = pathNames.map((pathName) => importer.scan(pathName));
2734
return foundClasses.reduce(
2835
(merged, found) => ({
2936
providers: [...merged.providers, ...found.providers],
3037
controllers: [...merged.controllers, ...found.controllers],
38+
exports: [...merged.exports, ...found.providers],
3139
}),
32-
{ providers: [], controllers: [] } as AutoClasses,
40+
{ providers: [], controllers: [], exports: [] },
3341
);
3442
}
3543

36-
private async scan(pathName: string): Promise<AutoClasses> {
37-
const exports: Record<string, unknown> = await import(pathName);
38-
const autoClasses = Object.values(exports).filter((value) => typeof value === 'function') as ClassType[];
44+
private scan(pathName: string): AutoClasses {
45+
const exports: Record<string, unknown> = require(pathName);
46+
return (Object.values(exports) as ClassType[]).reduce(
47+
(classes: AutoClasses, value: ClassType) => {
48+
if (typeof value === 'function') {
49+
Reflect.hasMetadata(COMPONENT_SCAN_WATERMARK, value) && this.catchOverlappedScanScope(value, pathName);
3950

40-
return autoClasses.reduce(
41-
(result: AutoClasses, value: ClassType) => {
42-
Reflect.hasMetadata(AUTO_INJECTABLE_WATERMARK, value) && result.providers.push(value);
43-
Reflect.hasMetadata(AUTO_CONTROLLER_WATERMARK, value) && result.controllers.push(value);
44-
return result;
51+
if (Reflect.hasMetadata(AUTO_ALIAS_WATERMARK, value)) {
52+
classes.providers.push({
53+
provide: Reflect.getMetadata(AUTO_ALIAS_WATERMARK, value),
54+
useClass: value,
55+
});
56+
} else if (Reflect.hasMetadata(AUTO_INJECTABLE_WATERMARK, value)) {
57+
classes.providers.push(value);
58+
} else if (Reflect.hasMetadata(AUTO_CONTROLLER_WATERMARK, value)) {
59+
classes.controllers.push(value);
60+
}
61+
}
62+
return classes;
4563
},
46-
{ providers: [], controllers: [] },
64+
{ providers: [], controllers: [], exports: [] },
65+
);
66+
}
67+
68+
private matchGlob(patterns: string[]) {
69+
const globs = patterns.map((pattern) =>
70+
globSync(resolve(require.main?.path || process.cwd(), pattern), {
71+
ignore: ['**/node_modules/**'],
72+
}),
4773
);
74+
return globs.flat();
4875
}
4976

50-
private async matchGlob(patterns: string[]) {
51-
const globs = patterns.map((pattern) => glob(resolve(process.cwd(), pattern)));
52-
return (await Promise.all(globs)).flat();
77+
/**
78+
* This code is intentionally structured to execute the callback function registered with `.then()`
79+
* and proceed through the event loop only after the asynchronous task of the `locate` function is completed.
80+
* Designed to handle potential errors after the scan is completed.
81+
*/
82+
private catchOverlappedScanScope(value: { new (...args: any[]): any; name: string }, pathName: string) {
83+
locate(value as any).then(({ path }: { path: string }) => {
84+
if (!this.rootPath) this.rootPath = pathName;
85+
if (this.rootPath !== path) {
86+
new Logger('ExceptionHandler').error(
87+
`ComponentScan() module scope cannot be overlapped.\n\nPotential causes:\n- A overlapped dependecy between modules.\n- Please check the module in '${this.rootPath}' and '${path}'\n\nScope [${value.name}]`,
88+
);
89+
process.exit(1);
90+
}
91+
});
5392
}
5493
}

lib/core/index.ts

-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1 @@
11
export * from './importer';
2-
export * from './import-sync';
+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
import { AUTO_ALIAS_WATERMARK } from '../interfaces';
2+
3+
export function AutoAlias(alias: string) {
4+
return (target: object) => {
5+
Reflect.defineMetadata(AUTO_ALIAS_WATERMARK, alias, target);
6+
};
7+
}

0 commit comments

Comments
 (0)