2018-07-16 05:23:37 -04: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
|
|
|
|
*/
|
|
|
|
import * as fs from 'fs';
|
|
|
|
import * as ts from 'typescript';
|
|
|
|
|
|
|
|
import MagicString from 'magic-string';
|
|
|
|
import {fromObject, generateMapFileComment} from 'convert-source-map';
|
|
|
|
import {makeProgram} from '../helpers/utils';
|
2018-10-04 07:19:11 -04:00
|
|
|
import {AnalyzedClass, DecorationAnalyzer, DecorationAnalyses} from '../../src/analysis/decoration_analyzer';
|
|
|
|
import {SwitchMarkerAnalyzer} from '../../src/analysis/switch_marker_analyzer';
|
2018-08-22 15:16:54 -04:00
|
|
|
import {Fesm2015ReflectionHost} from '../../src/host/fesm2015_host';
|
2018-07-16 05:23:37 -04:00
|
|
|
import {Renderer} from '../../src/rendering/renderer';
|
|
|
|
|
|
|
|
class TestRenderer extends Renderer {
|
2018-10-04 07:19:11 -04:00
|
|
|
constructor(host: Fesm2015ReflectionHost) { super(host, false, null, '/src', '/dist'); }
|
2018-07-16 05:23:37 -04:00
|
|
|
addImports(output: MagicString, imports: {name: string, as: string}[]) {
|
|
|
|
output.prepend('\n// ADD IMPORTS\n');
|
|
|
|
}
|
2018-06-27 11:41:11 -04:00
|
|
|
addConstants(output: MagicString, constants: string, file: ts.SourceFile): void {
|
|
|
|
output.prepend('\n// ADD CONSTANTS\n');
|
|
|
|
}
|
2018-07-16 05:23:37 -04:00
|
|
|
addDefinitions(output: MagicString, analyzedClass: AnalyzedClass, definitions: string) {
|
|
|
|
output.prepend('\n// ADD DEFINITIONS\n');
|
|
|
|
}
|
|
|
|
removeDecorators(output: MagicString, decoratorsToRemove: Map<ts.Node, ts.Node[]>) {
|
|
|
|
output.prepend('\n// REMOVE DECORATORS\n');
|
|
|
|
}
|
2018-08-17 02:50:45 -04:00
|
|
|
rewriteSwitchableDeclarations(output: MagicString, sourceFile: ts.SourceFile): void {
|
|
|
|
output.prepend('\n// REWRITTEN DECLARATIONS\n');
|
|
|
|
}
|
2018-07-16 05:23:37 -04:00
|
|
|
}
|
|
|
|
|
2018-10-04 07:19:11 -04:00
|
|
|
function createTestRenderer(file: {name: string, contents: string}) {
|
|
|
|
const program = makeProgram(file);
|
|
|
|
const host = new Fesm2015ReflectionHost(false, program.getTypeChecker());
|
|
|
|
const decorationAnalyses =
|
|
|
|
new DecorationAnalyzer(program.getTypeChecker(), host, [''], false).analyzeProgram(program);
|
|
|
|
const switchMarkerAnalyses = new SwitchMarkerAnalyzer(host).analyzeProgram(program);
|
|
|
|
const renderer = new TestRenderer(host);
|
2018-07-16 05:23:37 -04:00
|
|
|
spyOn(renderer, 'addImports').and.callThrough();
|
|
|
|
spyOn(renderer, 'addDefinitions').and.callThrough();
|
|
|
|
spyOn(renderer, 'removeDecorators').and.callThrough();
|
2018-09-28 09:57:50 -04:00
|
|
|
|
2018-10-04 07:19:11 -04:00
|
|
|
return {renderer, program, decorationAnalyses, switchMarkerAnalyses};
|
2018-07-16 05:23:37 -04:00
|
|
|
}
|
|
|
|
|
2018-09-28 09:57:50 -04:00
|
|
|
|
2018-07-16 05:23:37 -04:00
|
|
|
describe('Renderer', () => {
|
|
|
|
const INPUT_PROGRAM = {
|
2018-10-04 07:19:11 -04:00
|
|
|
name: '/src/file.js',
|
2018-07-16 05:23:37 -04:00
|
|
|
contents:
|
|
|
|
`import { Directive } from '@angular/core';\nexport class A {\n foo(x) {\n return x;\n }\n}\nA.decorators = [\n { type: Directive, args: [{ selector: '[a]' }] }\n];\n`
|
|
|
|
};
|
2018-10-04 07:19:11 -04:00
|
|
|
|
2018-07-16 05:23:37 -04:00
|
|
|
const INPUT_PROGRAM_MAP = fromObject({
|
|
|
|
'version': 3,
|
2018-10-04 07:19:11 -04:00
|
|
|
'file': '/src/file.js',
|
2018-07-16 05:23:37 -04:00
|
|
|
'sourceRoot': '',
|
2018-10-04 07:19:11 -04:00
|
|
|
'sources': ['/src/file.ts'],
|
2018-07-16 05:23:37 -04:00
|
|
|
'names': [],
|
|
|
|
'mappings':
|
|
|
|
'AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1C,MAAM;IACF,GAAG,CAAC,CAAS;QACT,OAAO,CAAC,CAAC;IACb,CAAC;;AACM,YAAU,GAAG;IAChB,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC,EAAE;CACnD,CAAC',
|
2018-10-04 07:19:11 -04:00
|
|
|
'sourcesContent': [INPUT_PROGRAM.contents]
|
2018-07-16 05:23:37 -04:00
|
|
|
});
|
2018-10-04 07:19:11 -04:00
|
|
|
|
2018-07-16 05:23:37 -04:00
|
|
|
const RENDERED_CONTENTS =
|
2018-10-04 07:19:11 -04:00
|
|
|
`\n// REMOVE DECORATORS\n\n// ADD IMPORTS\n\n// ADD CONSTANTS\n\n// ADD DEFINITIONS\n` +
|
2018-06-27 11:41:11 -04:00
|
|
|
INPUT_PROGRAM.contents;
|
2018-10-04 07:19:11 -04:00
|
|
|
|
2018-07-16 05:23:37 -04:00
|
|
|
const OUTPUT_PROGRAM_MAP = fromObject({
|
|
|
|
'version': 3,
|
2018-10-04 07:19:11 -04:00
|
|
|
'file': '/dist/file.js',
|
|
|
|
'sources': ['/src/file.js'],
|
|
|
|
'sourcesContent': [INPUT_PROGRAM.contents],
|
2018-07-16 05:23:37 -04:00
|
|
|
'names': [],
|
2018-10-04 07:19:11 -04:00
|
|
|
'mappings': ';;;;;;;;AAAA;;;;;;;;;'
|
2018-07-16 05:23:37 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
const MERGED_OUTPUT_PROGRAM_MAP = fromObject({
|
|
|
|
'version': 3,
|
2018-10-04 07:19:11 -04:00
|
|
|
'sources': ['/src/file.ts'],
|
2018-07-16 05:23:37 -04:00
|
|
|
'names': [],
|
2018-10-04 07:19:11 -04:00
|
|
|
'mappings': ';;;;;;;;AAAA',
|
|
|
|
'file': '/dist/file.js',
|
|
|
|
'sourcesContent': [INPUT_PROGRAM.contents]
|
2018-07-16 05:23:37 -04:00
|
|
|
});
|
|
|
|
|
2018-10-04 07:19:11 -04:00
|
|
|
describe('renderProgram()', () => {
|
2018-07-16 05:23:37 -04:00
|
|
|
it('should render the modified contents; and a new map file, if the original provided no map file.',
|
|
|
|
() => {
|
2018-10-04 07:19:11 -04:00
|
|
|
const {renderer, program, decorationAnalyses, switchMarkerAnalyses} =
|
|
|
|
createTestRenderer(INPUT_PROGRAM);
|
|
|
|
const result = renderer.renderProgram(program, decorationAnalyses, switchMarkerAnalyses);
|
|
|
|
expect(result[0].path).toEqual('/dist/file.js');
|
|
|
|
expect(result[0].contents)
|
|
|
|
.toEqual(RENDERED_CONTENTS + '\n' + generateMapFileComment('/dist/file.js.map'));
|
|
|
|
expect(result[1].path).toEqual('/dist/file.js.map');
|
|
|
|
expect(result[1].contents).toEqual(OUTPUT_PROGRAM_MAP.toJSON());
|
2018-07-16 05:23:37 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should call addImports with the source code and info about the core Angular library.',
|
|
|
|
() => {
|
2018-10-04 07:19:11 -04:00
|
|
|
const {decorationAnalyses, program, renderer, switchMarkerAnalyses} =
|
|
|
|
createTestRenderer(INPUT_PROGRAM);
|
|
|
|
renderer.renderProgram(program, decorationAnalyses, switchMarkerAnalyses);
|
|
|
|
const addImportsSpy = renderer.addImports as jasmine.Spy;
|
|
|
|
expect(addImportsSpy.calls.first().args[0].toString()).toEqual(RENDERED_CONTENTS);
|
|
|
|
expect(addImportsSpy.calls.first().args[1]).toEqual([
|
2018-07-16 05:23:37 -04:00
|
|
|
{name: '@angular/core', as: 'ɵngcc0'}
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should call addDefinitions with the source code, the analyzed class and the renderered definitions.',
|
|
|
|
() => {
|
2018-10-04 07:19:11 -04:00
|
|
|
const {decorationAnalyses, program, renderer, switchMarkerAnalyses} =
|
|
|
|
createTestRenderer(INPUT_PROGRAM);
|
|
|
|
renderer.renderProgram(program, decorationAnalyses, switchMarkerAnalyses);
|
|
|
|
const addDefinitionsSpy = renderer.addDefinitions as jasmine.Spy;
|
|
|
|
expect(addDefinitionsSpy.calls.first().args[0].toString()).toEqual(RENDERED_CONTENTS);
|
|
|
|
expect(addDefinitionsSpy.calls.first().args[1]).toEqual(jasmine.objectContaining({
|
|
|
|
name: 'A',
|
|
|
|
decorators: [jasmine.objectContaining({name: 'Directive'})],
|
|
|
|
}));
|
|
|
|
expect(addDefinitionsSpy.calls.first().args[2])
|
2018-07-16 05:23:37 -04:00
|
|
|
.toEqual(
|
2018-10-18 03:23:18 -04:00
|
|
|
`A.ngDirectiveDef = ɵngcc0.ɵdefineDirective({ type: A, selectors: [["", "a", ""]], factory: function A_Factory(t) { return new (t || A)(); } });`);
|
2018-07-16 05:23:37 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should call removeDecorators with the source code, a map of class decorators that have been analyzed',
|
|
|
|
() => {
|
2018-10-04 07:19:11 -04:00
|
|
|
const {decorationAnalyses, program, renderer, switchMarkerAnalyses} =
|
|
|
|
createTestRenderer(INPUT_PROGRAM);
|
|
|
|
renderer.renderProgram(program, decorationAnalyses, switchMarkerAnalyses);
|
|
|
|
const removeDecoratorsSpy = renderer.removeDecorators as jasmine.Spy;
|
|
|
|
expect(removeDecoratorsSpy.calls.first().args[0].toString()).toEqual(RENDERED_CONTENTS);
|
2018-07-16 05:23:37 -04:00
|
|
|
|
|
|
|
// Each map key is the TS node of the decorator container
|
|
|
|
// Each map value is an array of TS nodes that are the decorators to remove
|
2018-10-04 07:19:11 -04:00
|
|
|
const map = removeDecoratorsSpy.calls.first().args[1] as Map<ts.Node, ts.Node[]>;
|
2018-07-16 05:23:37 -04:00
|
|
|
const keys = Array.from(map.keys());
|
|
|
|
expect(keys.length).toEqual(1);
|
|
|
|
expect(keys[0].getText())
|
|
|
|
.toEqual(`[\n { type: Directive, args: [{ selector: '[a]' }] }\n]`);
|
|
|
|
const values = Array.from(map.values());
|
|
|
|
expect(values.length).toEqual(1);
|
|
|
|
expect(values[0].length).toEqual(1);
|
|
|
|
expect(values[0][0].getText()).toEqual(`{ type: Directive, args: [{ selector: '[a]' }] }`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should merge any inline source map from the original file and write the output as an inline source map',
|
|
|
|
() => {
|
2018-10-04 07:19:11 -04:00
|
|
|
const {decorationAnalyses, program, renderer, switchMarkerAnalyses} = createTestRenderer({
|
2018-07-16 05:23:37 -04:00
|
|
|
...INPUT_PROGRAM,
|
|
|
|
contents: INPUT_PROGRAM.contents + '\n' + INPUT_PROGRAM_MAP.toComment()
|
|
|
|
});
|
2018-10-04 07:19:11 -04:00
|
|
|
const result = renderer.renderProgram(program, decorationAnalyses, switchMarkerAnalyses);
|
|
|
|
expect(result[0].path).toEqual('/dist/file.js');
|
|
|
|
expect(result[0].contents)
|
2018-07-16 05:23:37 -04:00
|
|
|
.toEqual(RENDERED_CONTENTS + '\n' + MERGED_OUTPUT_PROGRAM_MAP.toComment());
|
2018-10-04 07:19:11 -04:00
|
|
|
expect(result[1]).toBeUndefined();
|
2018-07-16 05:23:37 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should merge any external source map from the original file and write the output to an external source map',
|
|
|
|
() => {
|
|
|
|
// Mock out reading the map file from disk
|
2018-10-04 07:19:11 -04:00
|
|
|
spyOn(fs, 'readFileSync').and.returnValue(INPUT_PROGRAM_MAP.toJSON());
|
|
|
|
const {decorationAnalyses, program, renderer, switchMarkerAnalyses} = createTestRenderer({
|
2018-07-16 05:23:37 -04:00
|
|
|
...INPUT_PROGRAM,
|
|
|
|
contents: INPUT_PROGRAM.contents + '\n//# sourceMappingURL=file.js.map'
|
|
|
|
});
|
2018-10-04 07:19:11 -04:00
|
|
|
const result = renderer.renderProgram(program, decorationAnalyses, switchMarkerAnalyses);
|
|
|
|
expect(result[0].path).toEqual('/dist/file.js');
|
|
|
|
expect(result[0].contents)
|
|
|
|
.toEqual(RENDERED_CONTENTS + '\n' + generateMapFileComment('/dist/file.js.map'));
|
|
|
|
expect(result[1].path).toEqual('/dist/file.js.map');
|
|
|
|
expect(result[1].contents).toEqual(MERGED_OUTPUT_PROGRAM_MAP.toJSON());
|
2018-07-16 05:23:37 -04:00
|
|
|
});
|
|
|
|
});
|
2018-07-27 15:36:54 -04:00
|
|
|
});
|