128 lines
4.7 KiB
JavaScript
128 lines
4.7 KiB
JavaScript
#!/usr/bin/env node
|
|
/**
|
|
* @license
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
* found in the LICENSE file at https://angular.io/license
|
|
*/
|
|
/* tslint:disable:no-console */
|
|
|
|
// Must be imported first, because Angular decorators throw on load.
|
|
import 'reflect-metadata';
|
|
|
|
import * as path from 'path';
|
|
import * as ts from 'typescript';
|
|
import * as assert from 'assert';
|
|
import {tsc} from '@angular/tsc-wrapped/src/tsc';
|
|
import {CompilerOptions, CodeGenerator, CompilerHostContext, NodeCompilerHostContext} from '@angular/compiler-cli';
|
|
|
|
/**
|
|
* Main method.
|
|
* Standalone program that executes the real codegen and tests that
|
|
* ngsummary.json files are used for libraries.
|
|
*/
|
|
function main() {
|
|
console.log(`testing usage of ngsummary.json files in libraries...`);
|
|
const basePath = path.resolve(__dirname, '..');
|
|
const project = path.resolve(basePath, 'tsconfig-build.json');
|
|
const readFiles: string[] = [];
|
|
const writtenFiles: {fileName: string, content: string}[] = [];
|
|
|
|
class AssertingHostContext extends NodeCompilerHostContext {
|
|
readFile(fileName: string): string {
|
|
if (/.*\/node_modules\/.*/.test(fileName) && !/.*ngsummary\.json$/.test(fileName) &&
|
|
!/package\.json$/.test(fileName)) {
|
|
// Only allow to read summaries and package.json files from node_modules
|
|
// TODO (mhevery): Fix this. TypeScript.d.ts does not allow returning null.
|
|
return null !;
|
|
}
|
|
readFiles.push(path.relative(basePath, fileName));
|
|
return super.readFile(fileName);
|
|
}
|
|
readResource(s: string): Promise<string> {
|
|
readFiles.push(path.relative(basePath, s));
|
|
return super.readResource(s);
|
|
}
|
|
}
|
|
|
|
const config = tsc.readConfiguration(project, basePath);
|
|
config.ngOptions.basePath = basePath;
|
|
// This flag tells ngc do not recompile libraries.
|
|
config.ngOptions.generateCodeForLibraries = false;
|
|
|
|
console.log(`>>> running codegen for ${project}`);
|
|
codegen(
|
|
config,
|
|
(host) => {
|
|
host.writeFile = (fileName: string, content: string) => {
|
|
fileName = path.relative(basePath, fileName);
|
|
writtenFiles.push({fileName, content});
|
|
};
|
|
return new AssertingHostContext();
|
|
})
|
|
.then((exitCode: any) => {
|
|
console.log(`>>> codegen done, asserting read files`);
|
|
assertSomeFileMatch(readFiles, /^node_modules\/.*\.ngsummary\.json$/);
|
|
assertNoFileMatch(readFiles, /^node_modules\/.*\.metadata.json$/);
|
|
assertNoFileMatch(readFiles, /^node_modules\/.*\.html$/);
|
|
assertNoFileMatch(readFiles, /^node_modules\/.*\.css$/);
|
|
|
|
assertNoFileMatch(readFiles, /^src\/.*\.ngsummary\.json$/);
|
|
assertSomeFileMatch(readFiles, /^src\/.*\.html$/);
|
|
assertSomeFileMatch(readFiles, /^src\/.*\.css$/);
|
|
|
|
console.log(`>>> asserting written files`);
|
|
assertWrittenFile(writtenFiles, /^src\/module\.ngfactory\.ts$/, /class MainModuleInjector/);
|
|
|
|
console.log(`done, no errors.`);
|
|
process.exit(exitCode);
|
|
})
|
|
.catch((e: any) => {
|
|
console.error(e.stack);
|
|
console.error('Compilation failed');
|
|
process.exit(1);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Simple adaption of tsc-wrapped main to just run codegen with a CompilerHostContext
|
|
*/
|
|
function codegen(
|
|
config: {parsed: ts.ParsedCommandLine, ngOptions: CompilerOptions},
|
|
hostContextFactory: (host: ts.CompilerHost) => CompilerHostContext) {
|
|
const host = ts.createCompilerHost(config.parsed.options, true);
|
|
|
|
// HACK: patch the realpath to solve symlink issue here:
|
|
// https://github.com/Microsoft/TypeScript/issues/9552
|
|
// todo(misko): remove once facade symlinks are removed
|
|
host.realpath = (path) => path;
|
|
|
|
const program = ts.createProgram(config.parsed.fileNames, config.parsed.options, host);
|
|
|
|
return CodeGenerator.create(config.ngOptions, {
|
|
} as any, program, host, hostContextFactory(host)).codegen();
|
|
}
|
|
|
|
function assertSomeFileMatch(fileNames: string[], pattern: RegExp) {
|
|
assert(
|
|
fileNames.some(fileName => pattern.test(fileName)),
|
|
`Expected some read files match ${pattern}`);
|
|
}
|
|
|
|
function assertNoFileMatch(fileNames: string[], pattern: RegExp) {
|
|
const matches = fileNames.filter(fileName => pattern.test(fileName));
|
|
assert(
|
|
matches.length === 0,
|
|
`Expected no read files match ${pattern}, but found: \n${matches.join('\n')}`);
|
|
}
|
|
|
|
function assertWrittenFile(
|
|
files: {fileName: string, content: string}[], filePattern: RegExp, contentPattern: RegExp) {
|
|
assert(
|
|
files.some(file => filePattern.test(file.fileName) && contentPattern.test(file.content)),
|
|
`Expected some written files for ${filePattern} and content ${contentPattern}`);
|
|
}
|
|
|
|
main();
|