Skip to content

Commit 2f94f4f

Browse files
authored
Feat/reworked dependency injection (#24)
1 parent d24f6e2 commit 2f94f4f

19 files changed

+262
-172
lines changed

app/src/__tests__/base-app-module.helper.spec.ts

+5-3
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,11 @@ import {
55
registerSingletonDynamicModule,
66
envFilePathList,
77
yalcBaseAppModuleMetadataFactory,
8+
YalcDefaultAppModule,
89
} from '../base-app-module.helper.js';
910
import { LifeCycleHandler } from '../life-cycle-handler.service.js';
1011
import { DynamicModule, Module } from '@nestjs/common';
11-
import { IBaseAppOptions } from '../base-app.interface.js';
12+
import { IYalcBaseAppOptions } from '../base-app.interface.js';
1213
import { createMock } from '@golevelup/ts-jest';
1314
import { AppContextService } from '../app-context.service.js';
1415

@@ -17,13 +18,14 @@ let appContextService: AppContextService = createMock<AppContextService>({
1718
});
1819

1920
class DummyDynamicModule extends YalcBaseAppModule {
20-
static forRoot(options?: IBaseAppOptions): DynamicModule {
21+
static forRoot(options?: IYalcBaseAppOptions): DynamicModule {
2122
return this.assignDynamicProperties(
2223
yalcBaseAppModuleMetadataFactory(this, 'appAlias', {
2324
...this.assignDynamicProperties({}),
2425
configFactory: () => ({}),
2526
logger: true,
2627
isSingleton: true,
28+
imports: [YalcDefaultAppModule.forRoot('appAlias', [])],
2729
...options,
2830
}),
2931
options,
@@ -123,7 +125,7 @@ describe('base-app', () => {
123125

124126
describe('registerSingletonDynamicModule', () => {
125127
class TestModule extends DummyDynamicModule {
126-
static forRoot(options?: IBaseAppOptions | undefined): DynamicModule {
128+
static forRoot(options?: IYalcBaseAppOptions | undefined): DynamicModule {
127129
return this._forRoot('test', options);
128130
}
129131
}

app/src/__tests__/tsconfig.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
{
22
"extends": "../../../tsconfig.test.json"
3-
}
3+
}

app/src/app-bootstrap-base.helper.ts

+31-4
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,36 @@
1-
import { INestApplicationContext, LoggerService, Type } from '@nestjs/common';
1+
import {
2+
DynamicModule,
3+
INestApplicationContext,
4+
LoggerService,
5+
Type,
6+
} from '@nestjs/common';
27
import { ConfigService } from '@nestjs/config';
38
// import { GqlExceptionFilter } from '@nestjs/graphql';
49
import { NestFastifyApplication } from '@nestjs/platform-fastify';
510
import type { IServiceConf } from './conf.type.js';
611
import { SYSTEM_LOGGER_SERVICE } from './def.const.js';
12+
import { YalcDefaultAppModule } from './base-app-module.helper.js';
13+
import { IGlobalOptions } from './app-bootstrap.helper.js';
714

815
export abstract class BaseAppBootstrap<
916
TAppType extends NestFastifyApplication | INestApplicationContext,
10-
TGlobalOptions = unknown,
17+
TGlobalOptions = IGlobalOptions,
1118
> {
1219
protected app?: TAppType;
1320
protected loggerService!: LoggerService;
21+
protected module: Type<any> | DynamicModule;
1422

1523
constructor(
1624
protected appAlias: string,
17-
protected readonly module: Type<any>,
18-
) {}
25+
protected readonly appModule: Type<any>,
26+
options?: { globalsOptions?: IGlobalOptions },
27+
) {
28+
this.module = YalcDefaultAppModule.forRoot(
29+
this.appAlias,
30+
[appModule, ...(options?.globalsOptions?.extraImports ?? [])],
31+
options?.globalsOptions,
32+
);
33+
}
1934

2035
setApp(app: TAppType) {
2136
this.app = app;
@@ -36,6 +51,18 @@ export abstract class BaseAppBootstrap<
3651
return this.app;
3752
}
3853

54+
/**
55+
*
56+
* @returns The main module of the business logic (the one that is passed in the constructor)
57+
*/
58+
getAppModule() {
59+
return this.appModule;
60+
}
61+
62+
/**
63+
*
64+
* @returns The global module that is used to bootstrap the app (YalcDefaultAppModule)
65+
*/
3966
getModule() {
4067
return this.module;
4168
}

app/src/app-bootstrap-standalone.helper.ts

+4-3
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,15 @@ import { FastifyInstance } from 'fastify';
55
import { envIsTrue } from '@nestjs-yalc/utils/env.helper.js';
66
import clc from 'cli-color';
77
import { BaseAppBootstrap } from './app-bootstrap-base.helper.js';
8+
import { IGlobalOptions } from './app-bootstrap.helper.js';
89

910
export class StandaloneAppBootstrap extends BaseAppBootstrap<INestApplicationContext> {
10-
constructor(appAlias: string, readonly module: any) {
11+
constructor(appAlias: string, module: any) {
1112
super(appAlias, module);
1213
}
1314

1415
async initApp(options?: {
15-
globalsOptions?: unknown;
16+
globalsOptions?: IGlobalOptions;
1617
fastifyInstance?: FastifyInstance;
1718
}) {
1819
await this.createApp({
@@ -33,7 +34,7 @@ export class StandaloneAppBootstrap extends BaseAppBootstrap<INestApplicationCon
3334
}
3435

3536
async createApp(_options?: {
36-
globalsOptions?: unknown;
37+
globalsOptions?: IGlobalOptions;
3738
fastifyInstance?: FastifyInstance;
3839
}) {
3940
let app: INestApplicationContext;

app/src/app-bootstrap.helper.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { SystemExceptionFilter } from '@nestjs-yalc/errors/filters/index.js';
22
import {
33
BadRequestException,
4+
DynamicModule,
45
ExceptionFilter,
56
ValidationPipe,
67
ValidationPipeOptions,
@@ -28,13 +29,15 @@ export interface IGlobalOptions {
2829
abortOnError?: boolean;
2930
enableSwagger?: boolean;
3031
validationPipeOptions?: ValidationPipeOptions;
32+
appAlias: string;
33+
extraImports?: NonNullable<DynamicModule['imports']>;
3134
}
3235

3336
export class AppBootstrap extends BaseAppBootstrap<NestFastifyApplication> {
3437
private fastifyInstance?: FastifyInstance;
3538
protected isSwaggerEnabled: boolean = false;
3639

37-
constructor(appAlias: string, readonly module: any) {
40+
constructor(appAlias: string, module: any) {
3841
super(appAlias, module);
3942
}
4043

app/src/base-app-module.helper.ts

+36-73
Original file line numberDiff line numberDiff line change
@@ -21,9 +21,10 @@ import {
2121
} from './app-config.service.js';
2222
import { AppContextModule } from './app-context.module.js';
2323
import { NODE_ENV } from '@nestjs-yalc/types/global.enum.js';
24-
import { ConfigModule, registerAs } from '@nestjs/config';
24+
import { ConfigModule, ConfigService, registerAs } from '@nestjs/config';
2525
import Joi from 'joi';
2626
import { MODULE_OPTIONS_TOKEN } from '@nestjs/common/cache/cache.module-definition.js';
27+
import { IGlobalOptions } from './app-bootstrap.helper.js';
2728

2829
const singletonDynamicModules = new Map<any, any>();
2930

@@ -68,11 +69,15 @@ export function envFilePathList(dirname: string = '.') {
6869
return envFilePath;
6970
}
7071

71-
function _metadataFactory(
72+
/**
73+
* Used for applications with controller/resolver support
74+
* you should override it
75+
*/
76+
export function yalcBaseAppModuleMetadataFactory(
7277
module: any,
7378
appAlias: string,
7479
options?: Omit<IYalcBaseAppOptions, 'module'>,
75-
) {
80+
): IYalcBaseStaticModule {
7681
const _options = {
7782
// default values
7883
isSingleton: false,
@@ -100,9 +105,7 @@ function _metadataFactory(
100105
];
101106

102107
if (options?.logger) {
103-
_providers.push(
104-
LoggerServiceFactory(appAlias, APP_LOGGER_SERVICE, appAlias),
105-
);
108+
_providers.push(LoggerServiceFactory(APP_LOGGER_SERVICE, appAlias));
106109
}
107110

108111
const hasConfig = _options.extraConfigs || _options.configFactory;
@@ -196,7 +199,7 @@ function _metadataFactory(
196199

197200
const _controllers: DynamicModule['controllers'] = [];
198201

199-
if (controllers && _options.isStandalone !== true) {
202+
if (controllers) {
200203
_controllers.push(...controllers);
201204
}
202205

@@ -212,24 +215,6 @@ function _metadataFactory(
212215
return config;
213216
}
214217

215-
/**
216-
* Used for applications with controller/resolver support
217-
* you should override it
218-
*/
219-
export function yalcBaseAppModuleMetadataFactory(
220-
module: any,
221-
appAlias: string,
222-
options?: Omit<IYalcBaseAppOptions, 'module'>,
223-
): IYalcBaseStaticModule {
224-
const config = _metadataFactory(module, appAlias, options);
225-
226-
if (!options || !options.skipDefaultApp) {
227-
config.imports.push(YalcDefaultAppModule.forRoot(appAlias, options));
228-
}
229-
230-
return config;
231-
}
232-
233218
/**
234219
* Util class that can be extended to create a NestJS application
235220
*/
@@ -294,11 +279,13 @@ export class YalcBaseAppModule {
294279
export class YalcDefaultAppModule {
295280
static forRoot(
296281
appAlias: string,
297-
options?: Omit<IYalcBaseAppOptions, 'module'>,
282+
imports: NonNullable<DynamicModule['imports']>,
283+
options?: IGlobalOptions,
298284
) {
299-
const imports: NonNullable<DynamicModule['imports']> = [
285+
const _imports: NonNullable<DynamicModule['imports']> = [
300286
EventEmitterModule.forRoot(),
301287
AppContextModule,
288+
...imports,
302289
];
303290

304291
const providers: NonNullable<DynamicModule['providers']> = [
@@ -312,59 +299,35 @@ export class YalcDefaultAppModule {
312299
},
313300
];
314301

315-
if (options?.logger) {
316-
providers.push(
317-
LoggerServiceFactory(appAlias, APP_LOGGER_SERVICE, appAlias),
318-
{
319-
provide: SYSTEM_LOGGER_SERVICE,
320-
useFactory: (logger) => logger,
321-
inject: [APP_LOGGER_SERVICE],
302+
providers.push(
303+
LoggerServiceFactory(APP_LOGGER_SERVICE, appAlias),
304+
{
305+
provide: AppConfigService,
306+
useFactory: (config: ConfigService) => {
307+
return new AppConfigService(config, appAlias);
322308
},
323-
);
324-
}
309+
inject: [ConfigService],
310+
},
311+
{
312+
provide: SYSTEM_LOGGER_SERVICE,
313+
useFactory: (logger) => logger,
314+
inject: [APP_LOGGER_SERVICE],
315+
},
316+
);
325317

326318
const exports: NonNullable<DynamicModule['exports']> = [
327319
APP_OPTION_TOKEN,
328320
APP_ALIAS_TOKEN,
329321
];
330322

331-
if (options?.logger) {
332-
exports.push(APP_LOGGER_SERVICE, SYSTEM_LOGGER_SERVICE);
333-
}
323+
exports.push(APP_LOGGER_SERVICE, SYSTEM_LOGGER_SERVICE);
334324

335-
/**
336-
* This is a workaround to make sure that the DefaultAppModule is always injected
337-
* with the last configurations, basically the ones defined from the app itself
338-
*
339-
* NOTE: The best way is to use the isApp flag to always identify when a module is an App
340-
*/
341-
const config = registerSingletonDynamicModule(
342-
true,
343-
YalcDefaultAppModule,
344-
{},
345-
);
346-
const newConfig = yalcBaseAppModuleMetadataFactory(
347-
YalcDefaultAppModule,
348-
`YalcDef-${appAlias}`,
349-
{
350-
global: true,
351-
isSingleton: false,
352-
logger: false,
353-
isStandalone: options?.isStandalone,
354-
skipDefaultApp: true,
355-
skipDuplicateAppCheck: true,
356-
imports,
357-
providers,
358-
exports,
359-
},
360-
) as IYalcBaseDynamicModule;
361-
362-
Object.assign(config, newConfig);
363-
364-
config.module = YalcDefaultAppModule;
365-
config.global = true;
366-
config.isSingleton = true;
367-
368-
return config;
325+
return {
326+
exports,
327+
providers,
328+
imports: _imports,
329+
module: YalcDefaultAppModule,
330+
global: true,
331+
};
369332
}
370333
}

app/src/base-app.interface.ts

+3-11
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,9 @@ export interface ISingletonOption {
1010
/**
1111
* This is used to make sure that the module is not instantiated multiple times
1212
* This is useful when you have a dynamic module that is imported in multiple other modules
13-
* If you use static modules, this is not needed
13+
* If you use static modules, this is not needed. Please use it with caution because it might
14+
* cause unexpected behavior
15+
*
1416
* NOTE: should be used in conjunction with the `global` property
1517
*/
1618
isSingleton?: boolean;
@@ -31,16 +33,6 @@ export interface IYalcBaseAppOptions extends Partial<ISingletonDynamicModule> {
3133
envDir?: string;
3234
migrations?: ClassType[];
3335
skipDuplicateAppCheck?: boolean;
34-
skipGlobalInterceptors?: boolean;
35-
/**
36-
* used internally, probably you don't wan to set thi
37-
*/
38-
skipDefaultApp?: boolean;
39-
isStandalone?: boolean;
40-
/**
41-
* used to define when a module is used as an application or loaded within an application
42-
*/
43-
isApp?: boolean;
4436
logger?: boolean;
4537
}
4638

0 commit comments

Comments
 (0)