166 lines
5.8 KiB
TypeScript
166 lines
5.8 KiB
TypeScript
|
/**
|
||
|
* @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
|
||
|
*/
|
||
|
|
||
|
import * as fs from 'fs';
|
||
|
import * as path from 'path';
|
||
|
import * as ts from 'typescript';
|
||
|
|
||
|
import * as ng from '../index';
|
||
|
import {FileChangeEvent, performWatchCompilation} from '../src/perform_watch';
|
||
|
|
||
|
import {TestSupport, expectNoDiagnostics, setup} from './test_support';
|
||
|
|
||
|
describe('perform watch', () => {
|
||
|
let testSupport: TestSupport;
|
||
|
let outDir: string;
|
||
|
|
||
|
beforeEach(() => {
|
||
|
testSupport = setup();
|
||
|
outDir = path.resolve(testSupport.basePath, 'outDir');
|
||
|
});
|
||
|
|
||
|
function createConfig(): ng.ParsedConfiguration {
|
||
|
const options = testSupport.createCompilerOptions({outDir});
|
||
|
return {
|
||
|
options,
|
||
|
rootNames: [path.resolve(testSupport.basePath, 'src/index.ts')],
|
||
|
project: path.resolve(testSupport.basePath, 'src/tsconfig.json'),
|
||
|
emitFlags: ng.EmitFlags.Default,
|
||
|
errors: []
|
||
|
};
|
||
|
}
|
||
|
|
||
|
it('should compile files during the initial run', () => {
|
||
|
const config = createConfig();
|
||
|
const host = new MockWatchHost(config);
|
||
|
|
||
|
testSupport.writeFiles({
|
||
|
'src/main.ts': createModuleAndCompSource('main'),
|
||
|
'src/index.ts': `export * from './main'; `,
|
||
|
});
|
||
|
|
||
|
const watchResult = performWatchCompilation(host);
|
||
|
expectNoDiagnostics(config.options, watchResult.firstCompileResult);
|
||
|
|
||
|
expect(fs.existsSync(path.resolve(outDir, 'src', 'main.ngfactory.js'))).toBe(true);
|
||
|
});
|
||
|
|
||
|
it('should cache files on subsequent runs', () => {
|
||
|
const config = createConfig();
|
||
|
const host = new MockWatchHost(config);
|
||
|
let fileExistsSpy: jasmine.Spy;
|
||
|
let getSourceFileSpy: jasmine.Spy;
|
||
|
host.createCompilerHost = (options: ng.CompilerOptions) => {
|
||
|
const ngHost = ng.createCompilerHost({options});
|
||
|
fileExistsSpy = spyOn(ngHost, 'fileExists').and.callThrough();
|
||
|
getSourceFileSpy = spyOn(ngHost, 'getSourceFile').and.callThrough();
|
||
|
return ngHost;
|
||
|
};
|
||
|
|
||
|
testSupport.writeFiles({
|
||
|
'src/main.ts': createModuleAndCompSource('main'),
|
||
|
'src/util.ts': `export const x = 1;`,
|
||
|
'src/index.ts': `
|
||
|
export * from './main';
|
||
|
export * from './util';
|
||
|
`,
|
||
|
});
|
||
|
|
||
|
const mainTsPath = path.resolve(testSupport.basePath, 'src', 'main.ts');
|
||
|
const utilTsPath = path.resolve(testSupport.basePath, 'src', 'util.ts');
|
||
|
const mainNgFactory = path.resolve(outDir, 'src', 'main.ngfactory.js');
|
||
|
performWatchCompilation(host);
|
||
|
expect(fs.existsSync(mainNgFactory)).toBe(true);
|
||
|
expect(fileExistsSpy !).toHaveBeenCalledWith(mainTsPath);
|
||
|
expect(fileExistsSpy !).toHaveBeenCalledWith(utilTsPath);
|
||
|
expect(getSourceFileSpy !).toHaveBeenCalledWith(mainTsPath, ts.ScriptTarget.ES5);
|
||
|
expect(getSourceFileSpy !).toHaveBeenCalledWith(utilTsPath, ts.ScriptTarget.ES5);
|
||
|
|
||
|
fileExistsSpy !.calls.reset();
|
||
|
getSourceFileSpy !.calls.reset();
|
||
|
|
||
|
// trigger a single file change
|
||
|
// -> all other files should be cached
|
||
|
fs.unlinkSync(mainNgFactory);
|
||
|
host.triggerFileChange(FileChangeEvent.Change, utilTsPath);
|
||
|
|
||
|
expect(fs.existsSync(mainNgFactory)).toBe(true);
|
||
|
expect(fileExistsSpy !).not.toHaveBeenCalledWith(mainTsPath);
|
||
|
expect(fileExistsSpy !).toHaveBeenCalledWith(utilTsPath);
|
||
|
expect(getSourceFileSpy !).not.toHaveBeenCalledWith(mainTsPath, ts.ScriptTarget.ES5);
|
||
|
expect(getSourceFileSpy !).toHaveBeenCalledWith(utilTsPath, ts.ScriptTarget.ES5);
|
||
|
|
||
|
// trigger a folder change
|
||
|
// -> nothing should be cached
|
||
|
fs.unlinkSync(mainNgFactory);
|
||
|
host.triggerFileChange(
|
||
|
FileChangeEvent.CreateDeleteDir, path.resolve(testSupport.basePath, 'src'));
|
||
|
|
||
|
expect(fs.existsSync(mainNgFactory)).toBe(true);
|
||
|
expect(fileExistsSpy !).toHaveBeenCalledWith(mainTsPath);
|
||
|
expect(fileExistsSpy !).toHaveBeenCalledWith(utilTsPath);
|
||
|
expect(getSourceFileSpy !).toHaveBeenCalledWith(mainTsPath, ts.ScriptTarget.ES5);
|
||
|
expect(getSourceFileSpy !).toHaveBeenCalledWith(utilTsPath, ts.ScriptTarget.ES5);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
function createModuleAndCompSource(prefix: string, template: string = prefix + 'template') {
|
||
|
const templateEntry =
|
||
|
template.endsWith('.html') ? `templateUrl: '${template}'` : `template: \`${template}\``;
|
||
|
return `
|
||
|
import {Component, NgModule} from '@angular/core';
|
||
|
|
||
|
@Component({selector: '${prefix}', ${templateEntry}})
|
||
|
export class ${prefix}Comp {}
|
||
|
|
||
|
@NgModule({declarations: [${prefix}Comp]})
|
||
|
export class ${prefix}Module {}
|
||
|
`;
|
||
|
}
|
||
|
|
||
|
class MockWatchHost {
|
||
|
timeoutListeners: Array<(() => void)|null> = [];
|
||
|
fileChangeListeners: Array<((event: FileChangeEvent, fileName: string) => void)|null> = [];
|
||
|
diagnostics: ng.Diagnostics = [];
|
||
|
constructor(public config: ng.ParsedConfiguration) {}
|
||
|
|
||
|
reportDiagnostics(diags: ng.Diagnostics) { this.diagnostics.push(...diags); }
|
||
|
readConfiguration() { return this.config; }
|
||
|
createCompilerHost(options: ng.CompilerOptions) { return ng.createCompilerHost({options}); };
|
||
|
createEmitCallback() { return undefined; }
|
||
|
onFileChange(
|
||
|
options: ng.CompilerOptions, listener: (event: FileChangeEvent, fileName: string) => void,
|
||
|
ready: () => void) {
|
||
|
const id = this.fileChangeListeners.length;
|
||
|
this.fileChangeListeners.push(listener);
|
||
|
ready();
|
||
|
return {
|
||
|
close: () => this.fileChangeListeners[id] = null,
|
||
|
};
|
||
|
}
|
||
|
setTimeout(callback: () => void, ms: number): any {
|
||
|
const id = this.timeoutListeners.length;
|
||
|
this.timeoutListeners.push(callback);
|
||
|
return id;
|
||
|
}
|
||
|
clearTimeout(timeoutId: any): void { this.timeoutListeners[timeoutId] = null; }
|
||
|
flushTimeouts() {
|
||
|
this.timeoutListeners.forEach(cb => {
|
||
|
if (cb) cb();
|
||
|
});
|
||
|
}
|
||
|
triggerFileChange(event: FileChangeEvent, fileName: string) {
|
||
|
this.fileChangeListeners.forEach(listener => {
|
||
|
if (listener) {
|
||
|
listener(event, fileName);
|
||
|
}
|
||
|
});
|
||
|
this.flushTimeouts();
|
||
|
}
|
||
|
}
|