Skip to content

Commit e944283

Browse files
authored
Merge pull request #23 from tiny-nestjs/dev
release: `v0.2.1`
2 parents a1e6ac5 + 11ceb67 commit e944283

37 files changed

+250
-282
lines changed

.eslintrc.js

-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
module.exports = {
22
parser: '@typescript-eslint/parser',
33
parserOptions: {
4-
project: 'tsconfig.json',
54
tsconfigRootDir: __dirname,
65
sourceType: 'module',
76
},

.gitignore

+4
Original file line numberDiff line numberDiff line change
@@ -398,3 +398,7 @@ Temporary Items
398398
dist
399399
.webpack
400400
.serverless/**/*.zip
401+
402+
# Local Test
403+
tsconfig.build.json
404+
src/

.husky/pre-commit

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
#!/usr/bin/env sh
2+
. "$(dirname -- "$0")/_/husky.sh"
3+
4+
npx lint-staged

.prettierrc

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
{
22
"singleQuote": true,
3-
"trailingComma": "all"
3+
"trailingComma": "all",
4+
"printWidth": 120,
5+
"tabWidth": 2
46
}

README.md

+52-46
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.0.1-blue" alt="npm version">
6+
<img src="https://img.shields.io/badge/npm-v0.2.1-blue" alt="npm version">
77
<img src="https://img.shields.io/badge/license-MIT-green" alt="License">
88
</div>
99

@@ -16,8 +16,9 @@ With this library, you can inject dependencies into classes without the need for
1616

1717
## Features
1818

19-
- `@AutoInjectable()` decorator allows classes to be automatically injectable for DI.
2019
- `@ComponentScan()` decorator enables automatic scanning and injection of classes within a module.
20+
- `@AutoInjectable()` decorator allows classes to be automatically injectable for DI.
21+
- `@AutoController()` decorator automatically registers controllers.
2122

2223
## Installation
2324

@@ -29,62 +30,67 @@ npm install @tiny-nestjs/auto-injectable
2930

3031
**1. `@ComponentScan()`**
3132

32-
Use `@ComponentScan()` decorator to enable automatic scanning and injection of classes within a module:
33-
34-
```typescript
35-
import { Module } from '@nestjs/common';
36-
import { AppService } from './app.service';
37-
import { AppController } from './app.controller';
38-
import { ComponentScan } from '@nestjs/auto-injectable';
39-
40-
@ComponentScan()
41-
@Module({
42-
imports: [],
43-
controllers: [AppController],
44-
providers: [AppService],
45-
})
46-
export class AppModule {}
47-
```
48-
49-
By applying the `@ComponentScan()` decorator to the AppModule class, Nest will automatically scan for classes and
50-
inject necessary dependencies.
33+
```ts
34+
import { Module } from '@nestjs/common';
35+
import { ComponentScan } from '@tiny-nestjs/auto-injectable';
36+
37+
@ComponentScan()
38+
@Module({
39+
imports: [],
40+
controllers: [],
41+
providers: [],
42+
})
43+
export class AppModule {
44+
}
45+
```
46+
47+
By applying the `@ComponentScan()` decorator to the AppModule class, Nest will automatically scan for classes and
48+
inject necessary dependencies.
5149

5250
**2. `@AutoInjectable()`**
5351

54-
Use `@AutoInjectable()` decorator to make a class injectable for DI:
52+
```ts
53+
import { AutoInjectable } from '@tiny-nestjs/auto-injectable';
5554

56-
```typescript
57-
import { AutoInjectable } from '@nestjs/auto-injectable';
58-
59-
@AutoInjectable()
60-
export class CatService {
61-
// ...
62-
}
63-
```
55+
@AutoInjectable()
56+
export class CatService {
57+
// ...
58+
}
59+
```
6460

65-
In this case, by applying the @AutoInjectable() decorator to the CatService class, the class has become injectable, allowing it to be injected into other modules without the need for module definitions.
61+
In this case, by applying the `@AutoInjectable()` decorator to the CatService class, the class has become injectable,
62+
allowing it to be injected into other modules without the need for module definitions.
6663

67-
**3. Inject**
64+
**3. `@AutoController()` and dependency injection**
65+
66+
```ts
67+
import { AutoController } from '@tiny-nestjs/auto-injectable';
68+
69+
@AutoController()
70+
export class CatController {
71+
constructor(private readonly catService: CatService) {
72+
}
73+
74+
@Get('cats')
75+
getCats() {
76+
return this.catService.findAll();
77+
}
78+
}
79+
```
6880

69-
```typescript
70-
@Controller()
71-
export class AppController {
72-
constructor(private readonly catService: CatService) {}
73-
74-
@Get('cats')
75-
getCats() {
76-
return this.catService.findAll();
77-
}
78-
}
79-
```
81+
The class with the `@AutoInjectable()` decorator has been successfully injected and `/cats` api can be accessed by
82+
applying `@AutoController()` on `CatController` service.
8083

81-
The class with the @AutoInjectable() decorator has been successfully injected.
84+
| You can see actual [project example](https://github.com/tiny-nestjs/auto-injectable-example) here. |
85+
|----------------------------------------------------------------------------------------------------|
8286

8387
## Contribution
8488

85-
To contribute to this library, fork the GitHub repository, make your changes, and create a pull request. Your
89+
To contribute to this library, fork the [GitHub repository](https://github.com/tiny-nestjs/auto-injectable), make your
90+
changes, and create a pull request. Your
8691
contributions are highly appreciated. If you find any improvements or bugs, please open an issue.
8792

8893
## License
8994

90-
`@tiny-nestjs/auto-injectable` is distributed under the MIT license.
95+
`@tiny-nestjs/auto-injectable` is distributed under
96+
the [MIT license](https://github.com/tiny-nestjs/auto-injectable/blob/main/LICENSE).

jest.config.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
module.exports = {
22
moduleFileExtensions: ['js', 'json', 'ts'],
3-
rootDir: 'src',
3+
rootDir: './',
44
testRegex: '.*\\.spec\\.ts$',
55
transform: {
66
'^.+\\.(t|j)s$': 'ts-jest',

lib/auto.module.ts

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import { DynamicModule, Module } from '@nestjs/common';
2+
import { Importer } from './core';
3+
4+
@Module({})
5+
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+
}

lib/core/importer.spec.ts

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
import { Importer } from './importer';
2+
3+
describe('Importer', () => {
4+
const paths = ['./test/fixture.ts'];
5+
6+
it('should get path of patterns', async () => {
7+
const result = await Importer.load(paths);
8+
console.log(result);
9+
});
10+
11+
it('should import file', async () => {
12+
const result = await Importer.load(paths);
13+
console.log(result);
14+
});
15+
});

lib/core/importer.ts

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
import { glob } from 'glob';
2+
import { resolve } from 'path';
3+
import { AUTO_CONTROLLER_WATERMARK, AUTO_INJECTABLE_WATERMARK } from '../interfaces';
4+
import 'reflect-metadata';
5+
6+
type ClassType = new (...args: any[]) => any;
7+
8+
interface AutoClasses {
9+
providers: ClassType[];
10+
controllers: ClassType[];
11+
}
12+
13+
export class Importer {
14+
private static instance: Importer | null = null;
15+
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)));
27+
return foundClasses.reduce(
28+
(merged, found) => ({
29+
providers: [...merged.providers, ...found.providers],
30+
controllers: [...merged.controllers, ...found.controllers],
31+
}),
32+
{ providers: [], controllers: [] } as AutoClasses,
33+
);
34+
}
35+
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[];
39+
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;
45+
},
46+
{ providers: [], controllers: [] },
47+
);
48+
}
49+
50+
private async matchGlob(patterns: string[]) {
51+
const globs = patterns.map((pattern) => glob(resolve(process.cwd(), pattern)));
52+
return (await Promise.all(globs)).flat();
53+
}
54+
}
File renamed without changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
import { Controller, ControllerOptions } from '@nestjs/common';
2+
import { AUTO_CONTROLLER_WATERMARK } from '../interfaces';
3+
4+
export function AutoController(prefixOrOptions?: string | string[] | ControllerOptions) {
5+
return (target: object) => {
6+
Reflect.defineMetadata(AUTO_CONTROLLER_WATERMARK, true, target);
7+
8+
/**
9+
* Listing of if statements due to `@Controller` overloading.
10+
*/
11+
if (typeof prefixOrOptions === 'undefined') {
12+
return Controller()(target as new (...args: any[]) => any);
13+
}
14+
if (typeof prefixOrOptions === 'string' || Array.isArray(prefixOrOptions)) {
15+
return Controller(prefixOrOptions)(target as new (...args: any[]) => any);
16+
}
17+
if (typeof prefixOrOptions === 'object') {
18+
return Controller(prefixOrOptions)(target as new (...args: any[]) => any);
19+
}
20+
};
21+
}

src/lib/decorators/auto-injectable.decorator.spec.ts lib/decorators/auto-injectable.decorator.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
import { AUTO_INJECTABLE_WATERMARK } from '../interfaces';
21
import { AutoInjectable } from './auto-injectable.decorator';
2+
import { AUTO_INJECTABLE_WATERMARK } from '../interfaces';
33

44
describe('AutoInjectable decorator', () => {
55
@AutoInjectable()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
import { AUTO_INJECTABLE_WATERMARK } from '../interfaces';
2+
import { Injectable } from '@nestjs/common';
3+
import { InjectableOptions } from '@nestjs/common/decorators/core/injectable.decorator';
4+
5+
export function AutoInjectable(options?: InjectableOptions) {
6+
return (target: object) => {
7+
Reflect.defineMetadata(AUTO_INJECTABLE_WATERMARK, true, target);
8+
9+
/**
10+
* `@Injectable` decorator only has one type.
11+
*/
12+
return Injectable(options)(target as new (...args: any[]) => any);
13+
};
14+
}

src/lib/decorators/component-scan.decorator.ts lib/decorators/component-scan.decorator.ts

+5-8
Original file line numberDiff line numberDiff line change
@@ -2,27 +2,24 @@ import { AutoModule } from '../auto.module';
22
import { Module } from '@nestjs/common';
33
import * as path from 'path';
44

5-
export function ComponentScan(
6-
paths: string[] = [getRootPath()],
7-
): ClassDecorator {
5+
export function ComponentScan(paths: string[] = [getRootPath()]): ClassDecorator {
86
return function (target: any) {
97
const originalMetadata = {
108
imports: Reflect.getMetadata(MODULE_OPTIONS.IMPORTS, target) || [],
11-
controllers:
12-
Reflect.getMetadata(MODULE_OPTIONS.CONTROLLERS, target) || [],
9+
controllers: Reflect.getMetadata(MODULE_OPTIONS.CONTROLLERS, target) || [],
1310
providers: Reflect.getMetadata(MODULE_OPTIONS.PROVIDERS, target) || [],
1411
};
1512

16-
const options = paths.map((path) => `${path}/**/*.js`);
13+
const patterns = paths.map((path) => `${path}/**/*.js`);
1714
Module({
1815
...originalMetadata,
19-
imports: [...originalMetadata.imports, AutoModule.forRootAsync(options)],
16+
imports: [...originalMetadata.imports, AutoModule.forRootAsync(patterns)],
2017
})(target);
2118
};
2219
}
2320

2421
function getRootPath(): string {
25-
return path.join(path.resolve(require.main?.filename), '..') || 'dist';
22+
return path.join(path.resolve(require.main?.filename ?? ''), '..') || 'dist';
2623
}
2724

2825
const MODULE_OPTIONS = {
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,3 @@
1+
export * from './auto-controller.decorator';
12
export * from './auto-injectable.decorator';
23
export * from './component-scan.decorator';

src/lib/index.ts lib/index.ts

File renamed without changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export const AUTO_CONTROLLER_WATERMARK = Symbol('AUTO_CONTROLLER_WATERMARK');
Original file line numberDiff line numberDiff line change
@@ -1 +1,2 @@
11
export * from './auto-injectable.constant';
2+
export * from './auto-controller.constant';

0 commit comments

Comments
 (0)