fix(compiler-cli): merge @fileoverview comments. (#20870)

Previously, this code would unconditionally add a @fileoverview
comment to generated files, and only if the contained any code at all.

However often existing fileoverview comments should be copied from the
file the generated file was originally based off of. This allows users
to e.g. include Closure Compiler directives in their original
`component.ts` file, which will then automaticallly also apply to code
generated from it.

This special cases `@license` comments, as Closure disregards directives
in comments containing `@license`.

PR Close #20870
This commit is contained in:
Martin Probst 2017-12-07 17:52:16 +01:00 committed by Jason Aden
parent add3589451
commit 8c52088346
5 changed files with 157 additions and 27 deletions

View File

@ -26,14 +26,7 @@ export class TypeScriptNodeEmitter {
...stmts.map(stmt => stmt.visitStatement(converter, null)).filter(stmt => stmt != null)); ...stmts.map(stmt => stmt.visitStatement(converter, null)).filter(stmt => stmt != null));
const preambleStmts: ts.Statement[] = []; const preambleStmts: ts.Statement[] = [];
if (preamble) { if (preamble) {
if (preamble.startsWith('/*') && preamble.endsWith('*/')) { const commentStmt = this.createCommentStatement(sourceFile, preamble);
preamble = preamble.substr(2, preamble.length - 4);
}
const commentStmt = ts.createNotEmittedStatement(sourceFile);
ts.setSyntheticLeadingComments(
commentStmt,
[{kind: ts.SyntaxKind.MultiLineCommentTrivia, text: preamble, pos: -1, end: -1}]);
ts.setEmitFlags(commentStmt, ts.EmitFlags.CustomPrologue);
preambleStmts.push(commentStmt); preambleStmts.push(commentStmt);
} }
const sourceStatments = const sourceStatments =
@ -42,6 +35,19 @@ export class TypeScriptNodeEmitter {
const newSourceFile = ts.updateSourceFileNode(sourceFile, sourceStatments); const newSourceFile = ts.updateSourceFileNode(sourceFile, sourceStatments);
return [newSourceFile, converter.getNodeMap()]; return [newSourceFile, converter.getNodeMap()];
} }
/** Creates a not emitted statement containing the given comment. */
createCommentStatement(sourceFile: ts.SourceFile, comment: string): ts.Statement {
if (comment.startsWith('/*') && comment.endsWith('*/')) {
comment = comment.substr(2, comment.length - 4);
}
const commentStmt = ts.createNotEmittedStatement(sourceFile);
ts.setSyntheticLeadingComments(
commentStmt,
[{kind: ts.SyntaxKind.MultiLineCommentTrivia, text: comment, pos: -1, end: -1}]);
ts.setEmitFlags(commentStmt, ts.EmitFlags.CustomPrologue);
return commentStmt;
}
} }
// A recorded node is a subtype of the node that is marked as being recoreded. This is used // A recorded node is a subtype of the node that is marked as being recoreded. This is used

View File

@ -12,26 +12,68 @@ import * as ts from 'typescript';
import {TypeScriptNodeEmitter} from './node_emitter'; import {TypeScriptNodeEmitter} from './node_emitter';
import {GENERATED_FILES} from './util'; import {GENERATED_FILES} from './util';
const PREAMBLE = `/** function getPreamble(original: string) {
* @fileoverview This file was generated by the Angular template compiler. return `/**
* Do not edit. * @fileoverview This file was generated by the Angular template compiler. Do not edit.
* @suppress {suspiciousCode,uselessCode,missingProperties,missingOverride,checkTypes} * ${original}
* tslint:disable * @suppress {suspiciousCode,uselessCode,missingProperties,missingOverride,checkTypes}
*/`; * tslint:disable
*/`;
}
export function getAngularEmitterTransformFactory(generatedFiles: Map<string, GeneratedFile>): () => /**
* Returns a transformer that does two things for generated files (ngfactory etc):
* - adds a fileoverview JSDoc comment containing Closure Compiler specific "suppress"ions in JSDoc.
* The new comment will contain any fileoverview comment text from the original source file this
* file was generated from.
* - updates generated files that are not in the given map of generatedFiles to have an empty
* list of statements as their body.
*/
export function getAngularEmitterTransformFactory(
generatedFiles: Map<string, GeneratedFile>, program: ts.Program): () =>
(sourceFile: ts.SourceFile) => ts.SourceFile { (sourceFile: ts.SourceFile) => ts.SourceFile {
return function() { return function() {
const emitter = new TypeScriptNodeEmitter(); const emitter = new TypeScriptNodeEmitter();
return function(sourceFile: ts.SourceFile): ts.SourceFile { return function(sourceFile: ts.SourceFile): ts.SourceFile {
const g = generatedFiles.get(sourceFile.fileName); const g = generatedFiles.get(sourceFile.fileName);
const orig = g && program.getSourceFile(g.srcFileUrl);
let originalComment = '';
if (orig) originalComment = getFileoverviewComment(orig);
const preamble = getPreamble(originalComment);
if (g && g.stmts) { if (g && g.stmts) {
const [newSourceFile] = emitter.updateSourceFile(sourceFile, g.stmts, PREAMBLE); const orig = program.getSourceFile(g.srcFileUrl);
let originalComment = '';
if (orig) originalComment = getFileoverviewComment(orig);
const [newSourceFile] = emitter.updateSourceFile(sourceFile, g.stmts, preamble);
return newSourceFile; return newSourceFile;
} else if (GENERATED_FILES.test(sourceFile.fileName)) { } else if (GENERATED_FILES.test(sourceFile.fileName)) {
return ts.updateSourceFileNode(sourceFile, []); // The file should be empty, but emitter.updateSourceFile would still add imports
// and various minutiae.
// Clear out the source file entirely, only including the preamble comment, so that
// ngc produces an empty .js file.
return ts.updateSourceFileNode(
sourceFile, [emitter.createCommentStatement(sourceFile, preamble)]);
} }
return sourceFile; return sourceFile;
}; };
}; };
} }
/**
* Parses and returns the comment text (without start and end markers) of a \@fileoverview comment
* in the given source file. Returns the empty string if no such comment can be found.
*/
function getFileoverviewComment(sourceFile: ts.SourceFile): string {
const trivia = sourceFile.getFullText().substring(0, sourceFile.getStart());
const leadingComments = ts.getLeadingCommentRanges(trivia, 0);
if (!leadingComments || leadingComments.length === 0) return '';
const comment = leadingComments[0];
if (comment.kind !== ts.SyntaxKind.MultiLineCommentTrivia) return '';
// Only comments separated with a \n\n from the file contents are considered file-level comments
// in TypeScript.
if (sourceFile.getFullText().substring(comment.end, comment.end + 2) !== '\n\n') return '';
const commentText = sourceFile.getFullText().substring(comment.pos, comment.end);
// Closure Compiler ignores @suppress and similar if the comment contains @license.
if (commentText.indexOf('@license') !== -1) return '';
return commentText.replace(/^\/\*\*/, '').replace(/ ?\*\/$/, '');
}

View File

@ -399,7 +399,7 @@ class AngularCompilerProgram implements Program {
if (!this.options.disableExpressionLowering) { if (!this.options.disableExpressionLowering) {
beforeTs.push(getExpressionLoweringTransformFactory(this.metadataCache, this.tsProgram)); beforeTs.push(getExpressionLoweringTransformFactory(this.metadataCache, this.tsProgram));
} }
beforeTs.push(getAngularEmitterTransformFactory(genFiles)); beforeTs.push(getAngularEmitterTransformFactory(genFiles, this.getTsProgram()));
if (customTransformers && customTransformers.beforeTs) { if (customTransformers && customTransformers.beforeTs) {
beforeTs.push(...customTransformers.beforeTs); beforeTs.push(...customTransformers.beforeTs);
} }
@ -628,15 +628,12 @@ class AngularCompilerProgram implements Program {
// Filter out generated files for which we didn't generate code. // Filter out generated files for which we didn't generate code.
// This can happen as the stub caclulation is not completely exact. // This can happen as the stub caclulation is not completely exact.
// Note: sourceFile refers to the .ngfactory.ts / .ngsummary.ts file // Note: sourceFile refers to the .ngfactory.ts / .ngsummary.ts file
// node_emitter_transform already set the file contents to be empty,
// so this code only needs to skip the file if !allowEmptyCodegenFiles.
const isGenerated = GENERATED_FILES.test(outFileName); const isGenerated = GENERATED_FILES.test(outFileName);
if (isGenerated) { if (isGenerated && !this.options.allowEmptyCodegenFiles &&
if (!genFile || !genFile.stmts || genFile.stmts.length === 0) { (!genFile || !genFile.stmts || genFile.stmts.length === 0)) {
if (this.options.allowEmptyCodegenFiles) { return;
outData = '';
} else {
return;
}
}
} }
if (baseFile) { if (baseFile) {
sourceFiles = sourceFiles ? [...sourceFiles, baseFile] : [baseFile]; sourceFiles = sourceFiles ? [...sourceFiles, baseFile] : [baseFile];

View File

@ -249,6 +249,90 @@ describe('ngc transformer command-line', () => {
.toBe(true); .toBe(true);
}); });
describe('comments', () => {
function compileAndRead(contents: string) {
writeConfig(`{
"extends": "./tsconfig-base.json",
"files": ["mymodule.ts"],
"angularCompilerOptions": {"allowEmptyCodegenFiles": true}
}`);
write('mymodule.ts', contents);
const exitCode = main(['-p', basePath], errorSpy);
expect(exitCode).toEqual(0);
const modPath = path.resolve(outDir, 'mymodule.ngfactory.js');
expect(fs.existsSync(modPath)).toBe(true);
return fs.readFileSync(modPath, {encoding: 'UTF-8'});
}
it('should be added', () => {
const contents = compileAndRead(`
import {CommonModule} from '@angular/common';
import {NgModule} from '@angular/core';
@NgModule({
imports: [CommonModule]
})
export class MyModule {}
`);
expect(contents).toContain('@fileoverview');
expect(contents).toContain('generated by the Angular template compiler');
expect(contents).toContain('@suppress {suspiciousCode');
});
it('should be merged with existing fileoverview comments', () => {
const contents = compileAndRead(`/** Hello world. */
import {CommonModule} from '@angular/common';
import {NgModule} from '@angular/core';
@NgModule({
imports: [CommonModule]
})
export class MyModule {}
`);
expect(contents).toContain('Hello world.');
});
it('should only pick file comments', () => {
const contents = compileAndRead(`
/** Comment on class. */
class MyClass {
}
`);
expect(contents).toContain('@fileoverview');
expect(contents).not.toContain('Comment on class.');
});
it('should not be merged with @license comments', () => {
const contents = compileAndRead(`/** @license Some license. */
import {CommonModule} from '@angular/common';
import {NgModule} from '@angular/core';
@NgModule({
imports: [CommonModule]
})
export class MyModule {}
`);
expect(contents).toContain('@fileoverview');
expect(contents).not.toContain('@license');
});
it('should be included in empty files', () => {
const contents = compileAndRead(`/** My comment. */
import {Inject, Injectable, Optional} from '@angular/core';
@Injectable()
export class NotAnAngularComponent {}
`);
expect(contents).toContain('My comment');
});
});
it('should compile with an explicit tsconfig reference', () => { it('should compile with an explicit tsconfig reference', () => {
writeConfig(`{ writeConfig(`{
"extends": "./tsconfig-base.json", "extends": "./tsconfig-base.json",

View File

@ -437,7 +437,8 @@ describe('ng program', () => {
sf => sf.fileName === path.join(testSupport.basePath, checks.originalFileName))) sf => sf.fileName === path.join(testSupport.basePath, checks.originalFileName)))
.toBe(true); .toBe(true);
if (checks.shouldBeEmpty) { if (checks.shouldBeEmpty) {
expect(writeData !.data).toBe(''); // The file should only contain comments (the preamble comment added by ngc).
expect(writeData !.data).toMatch(/^(\s*\/\*([^*]|\*[^/])*\*\/\s*)?$/);
} else { } else {
expect(writeData !.data).not.toBe(''); expect(writeData !.data).not.toBe('');
} }