Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: nestjs/nest
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: v7.0.10
Choose a base ref
...
head repository: nestjs/nest
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: v7.0.11
Choose a head ref
  • 8 commits
  • 20 files changed
  • 2 contributors

Commits on May 1, 2020

  1. Copy the full SHA
    36ba595 View commit details

Commits on May 12, 2020

  1. Copy the full SHA
    3e0fb38 View commit details
  2. Copy the full SHA
    e418414 View commit details
  3. Copy the full SHA
    1bc9939 View commit details
  4. Merge pull request #4587 from BrunnerLivio/fix/improve-invalid-module…

    …-exception
    
    fix(core): improve error message when imported module is nil
    kamilmysliwiec authored May 12, 2020
    Copy the full SHA
    dcf3ca0 View commit details
  5. Copy the full SHA
    7775081 View commit details
  6. Copy the full SHA
    64df4a9 View commit details
  7. Copy the full SHA
    661664d View commit details
2 changes: 1 addition & 1 deletion lerna.json
Original file line number Diff line number Diff line change
@@ -3,5 +3,5 @@
"packages": [
"packages/*"
],
"version": "7.0.10"
"version": "7.0.11"
}
2 changes: 1 addition & 1 deletion packages/common/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@nestjs/common",
"version": "7.0.10",
"version": "7.0.11",
"description": "Nest - modern, fast, powerful node.js web framework (@common)",
"author": "Kamil Mysliwiec",
"homepage": "https://nestjs.com",
5 changes: 2 additions & 3 deletions packages/core/errors/exceptions/invalid-module.exception.ts
Original file line number Diff line number Diff line change
@@ -2,8 +2,7 @@ import { INVALID_MODULE_MESSAGE } from '../messages';
import { RuntimeException } from './runtime.exception';

export class InvalidModuleException extends RuntimeException {
constructor(trace: any[]) {
const scope = (trace || []).map(module => module.name).join(' -> ');
super(INVALID_MODULE_MESSAGE`${scope}`);
constructor(parentModule: any, index: number, scope: any[]) {
super(INVALID_MODULE_MESSAGE(parentModule, index, scope));
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import { UNDEFINED_FORWARDREF_MESSAGE } from '../messages';
import { RuntimeException } from './runtime.exception';
import { Type } from '@nestjs/common';

export class UndefinedForwardRefException extends RuntimeException {
constructor(scope: Type<any>[]) {
super(UNDEFINED_FORWARDREF_MESSAGE(scope));
}
}
8 changes: 8 additions & 0 deletions packages/core/errors/exceptions/undefined-module.exception.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
import { RuntimeException } from './runtime.exception';
import { UNDEFINED_MODULE_MESSAGE } from '../messages';

export class UndefinedModuleException extends RuntimeException {
constructor(parentModule: any, index: number, scope: any[]) {
super(UNDEFINED_MODULE_MESSAGE(parentModule, index, scope));
}
}
45 changes: 38 additions & 7 deletions packages/core/errors/messages.ts
Original file line number Diff line number Diff line change
@@ -37,6 +37,9 @@ const getDependencyName = (dependency: InjectorDependency): string =>
const getModuleName = (module: Module) =>
(module && getInstanceName(module.metatype)) || 'current';

const stringifyScope = (scope: any[]): string =>
(scope || []).map(getInstanceName).join(' -> ');

export const UNKNOWN_DEPENDENCIES_MESSAGE = (
type: string | symbol,
unknownDependencyContext: InjectorDependencyContext,
@@ -84,15 +87,43 @@ export const INVALID_MIDDLEWARE_MESSAGE = (
name: string,
) => `The middleware doesn't provide the 'use' method (${name})`;

export const INVALID_MODULE_MESSAGE = (
text: TemplateStringsArray,
scope: string,
) =>
`Nest cannot create the module instance. Often, this is because of a circular dependency between modules. Use forwardRef() to avoid it.
export const UNDEFINED_FORWARDREF_MESSAGE = (
scope: Type<any>[],
) => `Nest cannot create the module instance. Often, this is because of a circular dependency between modules. Use forwardRef() to avoid it.
(Read more: https://docs.nestjs.com/fundamentals/circular-dependency)
Scope [${scope}]
`;
Scope [${stringifyScope(scope)}]
`;

export const INVALID_MODULE_MESSAGE = (
parentModule: any,
index: number,
scope: any[],
) => {
const parentModuleName = parentModule?.name || 'module';

return `Nest cannot create the ${parentModuleName} instance.
Received an unexpected value at index [${index}] of the ${parentModuleName} "imports" array.
Scope [${stringifyScope(scope)}]`;
};

export const UNDEFINED_MODULE_MESSAGE = (
parentModule: any,
index: number,
scope: any[],
) => {
const parentModuleName = parentModule?.name || 'module';

return `Nest cannot create the ${parentModuleName} instance.
The module at index [${index}] of the ${parentModuleName} "imports" array is undefined.
Potential causes:
- A circular dependency between modules. Use forwardRef() to avoid it. Read more: https://docs.nestjs.com/fundamentals/circular-dependency
- The module at index [${index}] is of type "undefined". Check your import statements and the type of the module.
Scope [${stringifyScope(scope)}]`;
};

export const UNKNOWN_EXPORT_MESSAGE = (token = 'item', module: string) => {
return `Nest cannot export a provider/module that is not a part of the currently processed module (${module}). Please verify whether the exported ${token} is available in this particular context.
6 changes: 4 additions & 2 deletions packages/core/injector/container.ts
Original file line number Diff line number Diff line change
@@ -4,7 +4,6 @@ import { Injectable } from '@nestjs/common/interfaces/injectable.interface';
import { Type } from '@nestjs/common/interfaces/type.interface';
import { ApplicationConfig } from '../application-config';
import { CircularDependencyException } from '../errors/exceptions/circular-dependency.exception';
import { InvalidModuleException } from '../errors/exceptions/invalid-module.exception';
import { UnknownModuleException } from '../errors/exceptions/unknown-module.exception';
import { ExternalContextCreator } from '../helpers/external-context-creator';
import { HttpAdapterHost } from '../helpers/http-adapter-host';
@@ -16,6 +15,7 @@ import { InternalProvidersStorage } from './internal-providers-storage';
import { Module } from './module';
import { ModuleTokenFactory } from './module-token-factory';
import { ModulesContainer } from './modules-container';
import { UndefinedForwardRefException } from '../errors/exceptions/undefined-forwardref.exception';

export class NestContainer {
private readonly globalModules = new Set<Module>();
@@ -55,8 +55,10 @@ export class NestContainer {
metatype: Type<any> | DynamicModule | Promise<DynamicModule>,
scope: Type<any>[],
): Promise<Module> {
// In DependenciesScanner#scanForModules we already check for undefined or invalid modules
// We sill need to catch the edge-case of `forwardRef(() => undefined)`
if (!metatype) {
throw new InvalidModuleException(scope);
throw new UndefinedForwardRefException(scope);
}
const { type, dynamicMetadata, token } = await this.moduleCompiler.compile(
metatype,
9 changes: 6 additions & 3 deletions packages/core/injector/instance-wrapper.ts
Original file line number Diff line number Diff line change
@@ -256,9 +256,11 @@ export class InstanceWrapper<T = any> {
): boolean {
const isDependencyTreeStatic = this.isDependencyTreeStatic();

return ((!isDependencyTreeStatic &&
return (
!isDependencyTreeStatic &&
contextId !== STATIC_CONTEXT &&
(!this.isTransient || (this.isTransient && inquirer))) as any) as boolean;
(!this.isTransient || (this.isTransient && !!inquirer))
);
}

public isLazyTransient(
@@ -299,7 +301,8 @@ export class InstanceWrapper<T = any> {
return (
this.isDependencyTreeStatic() &&
contextId === STATIC_CONTEXT &&
(!this.isTransient || (isStaticTransient && !!inquirer))
(!this.isTransient ||
(isStaticTransient && !!inquirer && !inquirer.isTransient))
);
}

3 changes: 2 additions & 1 deletion packages/core/injector/module-ref.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { Type } from '@nestjs/common';
import { UnknownElementException } from '../errors/exceptions/unknown-element.exception';
import { getClassScope } from '../helpers/get-class-scope';
import { NestContainer } from './container';
import { ContainerScanner } from './container-scanner';
import { Injector } from './injector';
@@ -38,8 +39,8 @@ export abstract class ModuleRef {
const wrapper = new InstanceWrapper({
name: type && type.name,
metatype: type,
instance: undefined,
isResolved: false,
scope: getClassScope(type),
host: moduleRef,
});
return new Promise<T>(async (resolve, reject) => {
4 changes: 2 additions & 2 deletions packages/core/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@nestjs/core",
"version": "7.0.10",
"version": "7.0.11",
"description": "Nest - modern, fast, powerful node.js web framework (@core)",
"author": "Kamil Mysliwiec",
"license": "MIT",
@@ -36,7 +36,7 @@
"uuid": "8.0.0"
},
"devDependencies": {
"@nestjs/common": "7.0.10"
"@nestjs/common": "7.0.11"
},
"peerDependencies": {
"@nestjs/common": "^7.0.0",
11 changes: 10 additions & 1 deletion packages/core/scanner.ts
Original file line number Diff line number Diff line change
@@ -43,6 +43,8 @@ import { NestContainer } from './injector/container';
import { InstanceWrapper } from './injector/instance-wrapper';
import { Module } from './injector/module';
import { MetadataScanner } from './metadata-scanner';
import { InvalidModuleException } from './errors/exceptions/invalid-module.exception';
import { UndefinedModuleException } from './errors/exceptions/undefined-module.exception';

interface ApplicationProviderWrapper {
moduleKey: string;
@@ -90,7 +92,14 @@ export class DependenciesScanner {
...((module as DynamicModule).imports || []),
];

for (const innerModule of modules) {
for (const [index, innerModule] of modules.entries()) {
// In case of a circular dependency (ES module system), JavaScript will resolve the type to `undefined`.
if (innerModule === undefined) {
throw new UndefinedModuleException(module, index, scope);
}
if (!innerModule) {
throw new InvalidModuleException(module, index, scope);
}
if (ctxRegistry.includes(innerModule)) {
continue;
}
Loading