Skip to content

Commit f630f33

Browse files
alxhubzarend
authored andcommitted
fix(compiler-cli): show a more specific error for Ivy NgModules (#41534) (#41598)
When an Ivy NgModule is imported into a View Engine build, it doesn't have metadata.json files that describe it as an NgModule, so it appears to VE builds as a plain, undecorated class. The error message shown in this situation generic and confusing, since it recommends adding an @NgModule annotation to a class from a library. This commit adds special detection into the View Engine compiler to give a more specific error message when an Ivy NgModule is imported. PR Close #41534 PR Close #41598
1 parent ba8da74 commit f630f33

File tree

5 files changed

+124
-8
lines changed

5 files changed

+124
-8
lines changed

goldens/public-api/compiler-cli/error_code.d.ts

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ export declare enum ErrorCode {
2727
NGMODULE_MODULE_WITH_PROVIDERS_MISSING_GENERIC = 6005,
2828
NGMODULE_REEXPORT_NAME_COLLISION = 6006,
2929
NGMODULE_DECLARATION_NOT_UNIQUE = 6007,
30+
NGMODULE_VE_DEPENDENCY_ON_IVY_LIB = 6999,
3031
SCHEMA_INVALID_ELEMENT = 8001,
3132
SCHEMA_INVALID_ATTRIBUTE = 8002,
3233
MISSING_REFERENCE_TARGET = 8003,

packages/compiler-cli/src/ngtsc/diagnostics/src/error_code.ts

+6
Original file line numberDiff line numberDiff line change
@@ -99,6 +99,12 @@ export enum ErrorCode {
9999
*/
100100
NGMODULE_DECLARATION_NOT_UNIQUE = 6007,
101101

102+
/**
103+
* Not actually raised by the compiler, but reserved for documentation of a View Engine error when
104+
* a View Engine build depends on an Ivy-compiled NgModule.
105+
*/
106+
NGMODULE_VE_DEPENDENCY_ON_IVY_LIB = 6999,
107+
102108
/**
103109
* An element name failed validation against the DOM schema.
104110
*/

packages/compiler-cli/src/transformers/program.ts

+65-3
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
* found in the LICENSE file at https://angular.io/license
88
*/
99

10-
import {AotCompiler, AotCompilerOptions, core, createAotCompiler, FormattedMessageChain, GeneratedFile, getParseErrors, isFormattedError, isSyntaxError, MessageBundle, NgAnalyzedFileWithInjectables, NgAnalyzedModules, ParseSourceSpan, PartialModule, Serializer, Xliff, Xliff2, Xmb} from '@angular/compiler';
10+
import {AotCompiler, AotCompilerOptions, core, createAotCompiler, FormattedMessageChain, GeneratedFile, getMissingNgModuleMetadataErrorData, getParseErrors, isFormattedError, isSyntaxError, MessageBundle, NgAnalyzedFileWithInjectables, NgAnalyzedModules, ParseSourceSpan, PartialModule, Serializer, Xliff, Xliff2, Xmb} from '@angular/compiler';
1111
import * as fs from 'fs';
1212
import * as path from 'path';
1313
import * as ts from 'typescript';
@@ -676,7 +676,7 @@ class AngularCompilerProgram implements Program {
676676
private _addStructuralDiagnostics(error: Error) {
677677
const diagnostics = this._structuralDiagnostics || (this._structuralDiagnostics = []);
678678
if (isSyntaxError(error)) {
679-
diagnostics.push(...syntaxErrorToDiagnostics(error));
679+
diagnostics.push(...syntaxErrorToDiagnostics(error, this.tsProgram));
680680
} else {
681681
diagnostics.push({
682682
messageText: error.toString(),
@@ -1022,7 +1022,7 @@ function diagnosticChainFromFormattedDiagnosticChain(chain: FormattedMessageChai
10221022
};
10231023
}
10241024

1025-
function syntaxErrorToDiagnostics(error: Error): Diagnostic[] {
1025+
function syntaxErrorToDiagnostics(error: Error, program: ts.Program): Diagnostic[] {
10261026
const parserErrors = getParseErrors(error);
10271027
if (parserErrors && parserErrors.length) {
10281028
return parserErrors.map<Diagnostic>(e => ({
@@ -1044,6 +1044,33 @@ function syntaxErrorToDiagnostics(error: Error): Diagnostic[] {
10441044
position: error.position
10451045
}];
10461046
}
1047+
1048+
const ngModuleErrorData = getMissingNgModuleMetadataErrorData(error);
1049+
if (ngModuleErrorData !== null) {
1050+
// This error represents the import or export of an `NgModule` that didn't have valid metadata.
1051+
// This _might_ happen because the NgModule in question is an Ivy-compiled library, and we want
1052+
// to show a more useful error if that's the case.
1053+
const ngModuleClass =
1054+
getDtsClass(program, ngModuleErrorData.fileName, ngModuleErrorData.className);
1055+
if (ngModuleClass !== null && isIvyNgModule(ngModuleClass)) {
1056+
return [{
1057+
messageText: `The NgModule '${ngModuleErrorData.className}' in '${
1058+
ngModuleErrorData
1059+
.fileName}' is imported by this compilation, but appears to be part of a library compiled for Angular Ivy. This may occur because:
1060+
1061+
1) the library was processed with 'ngcc'. Removing and reinstalling node_modules may fix this problem.
1062+
1063+
2) the library was published for Angular Ivy and v12+ applications only. Check its peer dependencies carefully and ensure that you're using a compatible version of Angular.
1064+
1065+
See https://angular.io/errors/NG6999 for more information.
1066+
`,
1067+
category: ts.DiagnosticCategory.Error,
1068+
code: DEFAULT_ERROR_CODE,
1069+
source: SOURCE,
1070+
}];
1071+
}
1072+
}
1073+
10471074
// Produce a Diagnostic anyway since we know for sure `error` is a SyntaxError
10481075
return [{
10491076
messageText: error.message,
@@ -1052,3 +1079,38 @@ function syntaxErrorToDiagnostics(error: Error): Diagnostic[] {
10521079
source: SOURCE,
10531080
}];
10541081
}
1082+
1083+
function getDtsClass(program: ts.Program, fileName: string, className: string): ts.ClassDeclaration|
1084+
null {
1085+
const sf = program.getSourceFile(fileName);
1086+
if (sf === undefined || !sf.isDeclarationFile) {
1087+
return null;
1088+
}
1089+
for (const stmt of sf.statements) {
1090+
if (!ts.isClassDeclaration(stmt)) {
1091+
continue;
1092+
}
1093+
if (stmt.name === undefined || stmt.name.text !== className) {
1094+
continue;
1095+
}
1096+
1097+
return stmt;
1098+
}
1099+
1100+
// No classes found that matched the given name.
1101+
return null;
1102+
}
1103+
1104+
function isIvyNgModule(clazz: ts.ClassDeclaration): boolean {
1105+
for (const member of clazz.members) {
1106+
if (!ts.isPropertyDeclaration(member)) {
1107+
continue;
1108+
}
1109+
if (ts.isIdentifier(member.name) && member.name.text === 'ɵmod') {
1110+
return true;
1111+
}
1112+
}
1113+
1114+
// No Ivy 'ɵmod' property found.
1115+
return false;
1116+
}

packages/compiler-cli/test/ngc_spec.ts

+28
Original file line numberDiff line numberDiff line change
@@ -330,6 +330,34 @@ describe('ngc transformer command-line', () => {
330330
});
331331
});
332332

333+
it('should give a specific error when an Angular Ivy NgModule is imported', () => {
334+
writeConfig(`{
335+
"extends": "./tsconfig-base.json",
336+
"files": ["mymodule.ts"]
337+
}`);
338+
write('node_modules/test/index.d.ts', `
339+
export declare class FooModule {
340+
static ɵmod = null;
341+
}
342+
`);
343+
write('mymodule.ts', `
344+
import {NgModule} from '@angular/core';
345+
import {FooModule} from 'test';
346+
347+
@NgModule({
348+
imports: [FooModule],
349+
})
350+
export class TestModule {}
351+
`);
352+
353+
const exitCode = main(['-p', basePath], errorSpy);
354+
expect(errorSpy).toHaveBeenCalledTimes(1);
355+
const message = errorSpy.calls.mostRecent().args[0];
356+
357+
// The error message should mention Ivy specifically.
358+
expect(message).toContain('Angular Ivy');
359+
});
360+
333361
describe('compile ngfactory files', () => {
334362
it('should compile ngfactory files that are not referenced by root files', () => {
335363
writeConfig(`{

packages/compiler/src/metadata_resolver.ts

+24-5
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,18 @@ export type ErrorCollector = (error: any, type?: any) => void;
2929

3030
export const ERROR_COMPONENT_TYPE = 'ngComponentType';
3131

32+
const MISSING_NG_MODULE_METADATA_ERROR_DATA = 'ngMissingNgModuleMetadataErrorData';
33+
export interface MissingNgModuleMetadataErrorData {
34+
fileName: string;
35+
className: string;
36+
}
37+
38+
39+
export function getMissingNgModuleMetadataErrorData(error: any): MissingNgModuleMetadataErrorData|
40+
null {
41+
return error[MISSING_NG_MODULE_METADATA_ERROR_DATA] ?? null;
42+
}
43+
3244
// Design notes:
3345
// - don't lazily create metadata:
3446
// For some metadata, we need to do async work sometimes,
@@ -563,11 +575,18 @@ export class CompileMetadataResolver {
563575
this.getNgModuleSummary(importedModuleType, alreadyCollecting);
564576
alreadyCollecting.delete(importedModuleType);
565577
if (!importedModuleSummary) {
566-
this._reportError(
567-
syntaxError(`Unexpected ${this._getTypeDescriptor(importedType)} '${
568-
stringifyType(importedType)}' imported by the module '${
569-
stringifyType(moduleType)}'. Please add a @NgModule annotation.`),
570-
moduleType);
578+
const err = syntaxError(`Unexpected ${this._getTypeDescriptor(importedType)} '${
579+
stringifyType(importedType)}' imported by the module '${
580+
stringifyType(moduleType)}'. Please add a @NgModule annotation.`);
581+
// If possible, record additional context for this error to enable more useful
582+
// diagnostics on the compiler side.
583+
if (importedType instanceof StaticSymbol) {
584+
(err as any)[MISSING_NG_MODULE_METADATA_ERROR_DATA] = {
585+
fileName: importedType.filePath,
586+
className: importedType.name,
587+
} as MissingNgModuleMetadataErrorData;
588+
}
589+
this._reportError(err, moduleType);
571590
return;
572591
}
573592
importedModules.push(importedModuleSummary);

0 commit comments

Comments
 (0)