| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  | import {CustomTransformers, Program} from '@angular/compiler-cli'; | 
					
						
							|  |  |  | import {setWrapHostForTest} from '@angular/compiler-cli/src/transformers/compiler_host'; | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  | import * as fs from 'fs'; | 
					
						
							|  |  |  | import * as path from 'path'; | 
					
						
							|  |  |  | import * as ts from 'typescript'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 17:54:43 +01:00
										 |  |  | import {createCompilerHost, createProgram} from '../../ngtools2'; | 
					
						
							|  |  |  | import {main, mainDiagnosticsForTest, readNgcCommandLineAndConfiguration} from '../../src/main'; | 
					
						
							|  |  |  | import {LazyRoute} from '../../src/ngtsc/routing'; | 
					
						
							| 
									
										
										
										
											2019-01-25 19:49:08 +01:00
										 |  |  | import {resolveNpmTreeArtifact} from '../runfile_helpers'; | 
					
						
							|  |  |  | import {TestSupport, setup} from '../test_support'; | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | function setupFakeCore(support: TestSupport): void { | 
					
						
							|  |  |  |   if (!process.env.TEST_SRCDIR) { | 
					
						
							|  |  |  |     throw new Error('`setupFakeCore` must be run within a Bazel test'); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-01-25 19:49:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const fakeNpmPackageDir = | 
					
						
							|  |  |  |       resolveNpmTreeArtifact('angular/packages/compiler-cli/test/ngtsc/fake_core/npm_package'); | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const nodeModulesPath = path.join(support.basePath, 'node_modules'); | 
					
						
							|  |  |  |   const angularCoreDirectory = path.join(nodeModulesPath, '@angular/core'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-26 16:14:05 +02:00
										 |  |  |   fs.symlinkSync(fakeNpmPackageDir, angularCoreDirectory, 'junction'); | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Manages a temporary testing directory structure and environment for testing ngtsc by feeding it | 
					
						
							|  |  |  |  * TypeScript code. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export class NgtscTestEnvironment { | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |   private multiCompileHostExt: MultiCompileHostExt|null = null; | 
					
						
							|  |  |  |   private oldProgram: Program|null = null; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |   private constructor(private support: TestSupport, readonly outDir: string) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   get basePath(): string { return this.support.basePath; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Set up a new testing environment. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static setup(): NgtscTestEnvironment { | 
					
						
							|  |  |  |     const support = setup(); | 
					
						
							| 
									
										
										
										
											2019-05-14 14:21:30 +02:00
										 |  |  |     const outDir = path.posix.join(support.basePath, 'built'); | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |     process.chdir(support.basePath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setupFakeCore(support); | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |     setWrapHostForTest(null); | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const env = new NgtscTestEnvironment(support, outDir); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     env.write('tsconfig-base.json', `{
 | 
					
						
							|  |  |  |       "compilerOptions": { | 
					
						
							| 
									
										
										
										
											2019-02-22 18:06:25 -08:00
										 |  |  |         "emitDecoratorMetadata": true, | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |         "experimentalDecorators": true, | 
					
						
							|  |  |  |         "skipLibCheck": true, | 
					
						
							|  |  |  |         "noImplicitAny": true, | 
					
						
							|  |  |  |         "strictNullChecks": true, | 
					
						
							|  |  |  |         "outDir": "built", | 
					
						
							|  |  |  |         "rootDir": ".", | 
					
						
							|  |  |  |         "baseUrl": ".", | 
					
						
							|  |  |  |         "declaration": true, | 
					
						
							|  |  |  |         "target": "es5", | 
					
						
							| 
									
										
										
										
											2019-01-25 19:49:08 +01:00
										 |  |  |         "newLine": "lf", | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |         "module": "es2015", | 
					
						
							|  |  |  |         "moduleResolution": "node", | 
					
						
							|  |  |  |         "lib": ["es6", "dom"], | 
					
						
							|  |  |  |         "typeRoots": ["node_modules/@types"] | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       "angularCompilerOptions": { | 
					
						
							| 
									
										
										
										
											2019-04-02 11:52:19 -07:00
										 |  |  |         "enableIvy": true, | 
					
						
							|  |  |  |         "ivyTemplateTypeCheck": false | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |       }, | 
					
						
							|  |  |  |       "exclude": [ | 
					
						
							|  |  |  |         "built" | 
					
						
							|  |  |  |       ] | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |     }`);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return env; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   assertExists(fileName: string) { | 
					
						
							|  |  |  |     if (!fs.existsSync(path.resolve(this.outDir, fileName))) { | 
					
						
							|  |  |  |       throw new Error(`Expected ${fileName} to be emitted (outDir: ${this.outDir})`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   assertDoesNotExist(fileName: string) { | 
					
						
							|  |  |  |     if (fs.existsSync(path.resolve(this.outDir, fileName))) { | 
					
						
							|  |  |  |       throw new Error(`Did not expect ${fileName} to be emitted (outDir: ${this.outDir})`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getContents(fileName: string): string { | 
					
						
							|  |  |  |     this.assertExists(fileName); | 
					
						
							|  |  |  |     const modulePath = path.resolve(this.outDir, fileName); | 
					
						
							|  |  |  |     return fs.readFileSync(modulePath, 'utf8'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |   enableMultipleCompilations(): void { | 
					
						
							|  |  |  |     this.multiCompileHostExt = new MultiCompileHostExt(); | 
					
						
							|  |  |  |     setWrapHostForTest(makeWrapHost(this.multiCompileHostExt)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   flushWrittenFileTracking(): void { | 
					
						
							|  |  |  |     if (this.multiCompileHostExt === null) { | 
					
						
							|  |  |  |       throw new Error(`Not tracking written files - call enableMultipleCompilations()`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     this.multiCompileHostExt.flushWrittenFileTracking(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getFilesWrittenSinceLastFlush(): Set<string> { | 
					
						
							|  |  |  |     if (this.multiCompileHostExt === null) { | 
					
						
							|  |  |  |       throw new Error(`Not tracking written files - call enableMultipleCompilations()`); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-26 16:14:05 +02:00
										 |  |  |     const outDir = path.posix.join(this.support.basePath, 'built'); | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |     const writtenFiles = new Set<string>(); | 
					
						
							|  |  |  |     this.multiCompileHostExt.getFilesWrittenSinceLastFlush().forEach(rawFile => { | 
					
						
							|  |  |  |       if (rawFile.startsWith(outDir)) { | 
					
						
							|  |  |  |         writtenFiles.add(rawFile.substr(outDir.length)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     return writtenFiles; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   write(fileName: string, content: string) { | 
					
						
							|  |  |  |     if (this.multiCompileHostExt !== null) { | 
					
						
							| 
									
										
										
										
											2019-05-14 14:21:30 +02:00
										 |  |  |       const absFilePath = path.posix.resolve(this.support.basePath, fileName); | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |       this.multiCompileHostExt.invalidate(absFilePath); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     this.support.write(fileName, content); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   invalidateCachedFile(fileName: string): void { | 
					
						
							|  |  |  |     if (this.multiCompileHostExt === null) { | 
					
						
							|  |  |  |       throw new Error(`Not caching files - call enableMultipleCompilations()`); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-14 14:21:30 +02:00
										 |  |  |     const fullFile = path.posix.join(this.support.basePath, fileName); | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |     this.multiCompileHostExt.invalidate(fullFile); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-30 10:37:06 -08:00
										 |  |  |   tsconfig(extraOpts: {[key: string]: string | boolean} = {}, extraRootDirs?: string[]): void { | 
					
						
							|  |  |  |     const tsconfig: {[key: string]: any} = { | 
					
						
							|  |  |  |       extends: './tsconfig-base.json', | 
					
						
							| 
									
										
										
										
											2019-02-08 11:37:21 +00:00
										 |  |  |       angularCompilerOptions: {...extraOpts, enableIvy: true}, | 
					
						
							| 
									
										
										
										
											2018-11-30 10:37:06 -08:00
										 |  |  |     }; | 
					
						
							|  |  |  |     if (extraRootDirs !== undefined) { | 
					
						
							|  |  |  |       tsconfig.compilerOptions = { | 
					
						
							|  |  |  |         rootDirs: ['.', ...extraRootDirs], | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     this.write('tsconfig.json', JSON.stringify(tsconfig, null, 2)); | 
					
						
							| 
									
										
										
										
											2019-02-19 17:36:26 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (extraOpts['_useHostForImportGeneration'] === true) { | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |       setWrapHostForTest(makeWrapHost(new FileNameToModuleNameHost())); | 
					
						
							| 
									
										
										
										
											2019-02-19 17:36:26 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Run the compiler to completion, and assert that no errors occurred. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2019-01-03 12:23:00 +02:00
										 |  |  |   driveMain(customTransformers?: CustomTransformers): void { | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |     const errorSpy = jasmine.createSpy('consoleError').and.callFake(console.error); | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |     let reuseProgram: {program: Program | undefined}|undefined = undefined; | 
					
						
							|  |  |  |     if (this.multiCompileHostExt !== null) { | 
					
						
							|  |  |  |       reuseProgram = { | 
					
						
							|  |  |  |         program: this.oldProgram || undefined, | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const exitCode = | 
					
						
							|  |  |  |         main(['-p', this.basePath], errorSpy, undefined, customTransformers, reuseProgram); | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |     expect(errorSpy).not.toHaveBeenCalled(); | 
					
						
							|  |  |  |     expect(exitCode).toBe(0); | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  |     if (this.multiCompileHostExt !== null) { | 
					
						
							|  |  |  |       this.oldProgram = reuseProgram !.program !; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Run the compiler to completion, and return any `ts.Diagnostic` errors that may have occurred. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   driveDiagnostics(): ReadonlyArray<ts.Diagnostic> { | 
					
						
							|  |  |  |     // Cast is safe as ngtsc mode only produces ts.Diagnostics.
 | 
					
						
							|  |  |  |     return mainDiagnosticsForTest(['-p', this.basePath]) as ReadonlyArray<ts.Diagnostic>; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 17:54:43 +01:00
										 |  |  |   driveRoutes(entryPoint?: string): LazyRoute[] { | 
					
						
							|  |  |  |     const {rootNames, options} = readNgcCommandLineAndConfiguration(['-p', this.basePath]); | 
					
						
							|  |  |  |     const host = createCompilerHost({options}); | 
					
						
							|  |  |  |     const program = createProgram({rootNames, host, options}); | 
					
						
							|  |  |  |     return program.listLazyRoutes(entryPoint); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-09-25 15:35:03 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-03-18 11:21:29 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | class AugmentedCompilerHost { | 
					
						
							|  |  |  |   delegate !: ts.CompilerHost; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FileNameToModuleNameHost extends AugmentedCompilerHost { | 
					
						
							|  |  |  |   // CWD must be initialized lazily as `this.delegate` is not set until later.
 | 
					
						
							|  |  |  |   private cwd: string|null = null; | 
					
						
							|  |  |  |   fileNameToModuleName(importedFilePath: string): string { | 
					
						
							|  |  |  |     if (this.cwd === null) { | 
					
						
							|  |  |  |       this.cwd = this.delegate.getCurrentDirectory(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 'root' + importedFilePath.substr(this.cwd.length).replace(/(\.d)?.ts$/, ''); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MultiCompileHostExt extends AugmentedCompilerHost implements Partial<ts.CompilerHost> { | 
					
						
							|  |  |  |   private cache = new Map<string, ts.SourceFile>(); | 
					
						
							|  |  |  |   private writtenFiles = new Set<string>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getSourceFile( | 
					
						
							|  |  |  |       fileName: string, languageVersion: ts.ScriptTarget, onError?: (message: string) => void, | 
					
						
							|  |  |  |       shouldCreateNewSourceFile?: boolean): ts.SourceFile|undefined { | 
					
						
							|  |  |  |     if (this.cache.has(fileName)) { | 
					
						
							|  |  |  |       return this.cache.get(fileName) !; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const sf = | 
					
						
							|  |  |  |         this.delegate.getSourceFile(fileName, languageVersion, onError, shouldCreateNewSourceFile); | 
					
						
							|  |  |  |     if (sf !== undefined) { | 
					
						
							|  |  |  |       this.cache.set(sf.fileName, sf); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return sf; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   flushWrittenFileTracking(): void { this.writtenFiles.clear(); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   writeFile( | 
					
						
							|  |  |  |       fileName: string, data: string, writeByteOrderMark: boolean, | 
					
						
							|  |  |  |       onError: ((message: string) => void)|undefined, | 
					
						
							|  |  |  |       sourceFiles?: ReadonlyArray<ts.SourceFile>): void { | 
					
						
							|  |  |  |     this.delegate.writeFile(fileName, data, writeByteOrderMark, onError, sourceFiles); | 
					
						
							|  |  |  |     this.writtenFiles.add(fileName); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getFilesWrittenSinceLastFlush(): Set<string> { return this.writtenFiles; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   invalidate(fileName: string): void { this.cache.delete(fileName); } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function makeWrapHost(wrapped: AugmentedCompilerHost): (host: ts.CompilerHost) => ts.CompilerHost { | 
					
						
							|  |  |  |   return (delegate) => { | 
					
						
							|  |  |  |     wrapped.delegate = delegate; | 
					
						
							|  |  |  |     return new Proxy(delegate, { | 
					
						
							|  |  |  |       get: (target: ts.CompilerHost, name: string): any => { | 
					
						
							|  |  |  |         if ((wrapped as any)[name] !== undefined) { | 
					
						
							|  |  |  |           return (wrapped as any)[name] !.bind(wrapped); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return (target as any)[name]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } |