2017-04-26 12:24:42 -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
|
|
|
|
*/
|
|
|
|
|
2017-06-09 17:00:03 -04:00
|
|
|
import {AotCompiler, AotCompilerHost, AotCompilerOptions, CompileSummaryKind, GeneratedFile, toTypeScript} from '@angular/compiler';
|
2017-04-26 12:24:42 -04:00
|
|
|
|
|
|
|
import {MockDirectory, compile, setup} from './test_util';
|
|
|
|
|
|
|
|
describe('aot summaries for jit', () => {
|
|
|
|
let angularFiles = setup();
|
|
|
|
|
|
|
|
function compileApp(rootDir: MockDirectory, options: {useSummaries?: boolean} = {}):
|
|
|
|
{genFiles: GeneratedFile[], outDir: MockDirectory} {
|
2017-06-09 17:00:03 -04:00
|
|
|
return compile([rootDir, angularFiles], {...options, enableSummariesForJit: true});
|
2017-04-26 12:24:42 -04:00
|
|
|
}
|
|
|
|
|
2017-05-17 18:39:08 -04:00
|
|
|
it('should create @Injectable summaries', () => {
|
|
|
|
const appDir = {
|
|
|
|
'app.module.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { Injectable } from '@angular/core';
|
|
|
|
|
|
|
|
export class Dep {}
|
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class MyService {
|
|
|
|
constructor(d: Dep) {}
|
|
|
|
}
|
|
|
|
`
|
2017-05-17 18:39:08 -04:00
|
|
|
};
|
|
|
|
const rootDir = {'app': appDir};
|
|
|
|
|
|
|
|
const genFile =
|
|
|
|
compileApp(rootDir).genFiles.find(f => f.genFileUrl === '/app/app.module.ngsummary.ts');
|
|
|
|
const genSource = toTypeScript(genFile);
|
|
|
|
|
|
|
|
expect(genSource).toContain(`import * as i0 from '/app/app.module'`);
|
|
|
|
expect(genSource).toContain('export function MyServiceNgSummary()');
|
|
|
|
// Note: CompileSummaryKind.Injectable = 3
|
|
|
|
expect(genSource).toMatch(/summaryKind:3,\s*type:\{\s*reference:i0.MyService/);
|
|
|
|
expect(genSource).toContain('token:{identifier:{reference:i0.Dep}}');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should create @Pipe summaries', () => {
|
|
|
|
const appDir = {
|
|
|
|
'app.module.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { Pipe, NgModule } from '@angular/core';
|
|
|
|
|
|
|
|
export class Dep {}
|
|
|
|
|
|
|
|
@Pipe({name: 'myPipe'})
|
|
|
|
export class MyPipe {
|
|
|
|
constructor(d: Dep) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({declarations: [MyPipe]})
|
|
|
|
export class MyModule {}
|
|
|
|
`
|
2017-05-17 18:39:08 -04:00
|
|
|
};
|
|
|
|
const rootDir = {'app': appDir};
|
|
|
|
|
|
|
|
const genFile =
|
|
|
|
compileApp(rootDir).genFiles.find(f => f.genFileUrl === '/app/app.module.ngsummary.ts');
|
|
|
|
const genSource = toTypeScript(genFile);
|
|
|
|
|
|
|
|
expect(genSource).toContain(`import * as i0 from '/app/app.module'`);
|
|
|
|
expect(genSource).toContain('export function MyPipeNgSummary()');
|
|
|
|
// Note: CompileSummaryKind.Pipe = 1
|
|
|
|
expect(genSource).toMatch(/summaryKind:0,\s*type:\{\s*reference:i0.MyPipe/);
|
|
|
|
expect(genSource).toContain('token:{identifier:{reference:i0.Dep}}');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should create @Directive summaries', () => {
|
|
|
|
const appDir = {
|
|
|
|
'app.module.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { Directive, NgModule } from '@angular/core';
|
|
|
|
|
|
|
|
export class Dep {}
|
|
|
|
|
|
|
|
@Directive({selector: '[myDir]'})
|
|
|
|
export class MyDirective {
|
|
|
|
constructor(a: Dep) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({declarations: [MyDirective]})
|
|
|
|
export class MyModule {}
|
|
|
|
`
|
2017-05-17 18:39:08 -04:00
|
|
|
};
|
|
|
|
const rootDir = {'app': appDir};
|
|
|
|
|
|
|
|
const genFile =
|
|
|
|
compileApp(rootDir).genFiles.find(f => f.genFileUrl === '/app/app.module.ngsummary.ts');
|
|
|
|
const genSource = toTypeScript(genFile);
|
|
|
|
|
|
|
|
expect(genSource).toContain(`import * as i0 from '/app/app.module'`);
|
|
|
|
expect(genSource).toContain('export function MyDirectiveNgSummary()');
|
|
|
|
// Note: CompileSummaryKind.Directive = 1
|
|
|
|
expect(genSource).toMatch(/summaryKind:1,\s*type:\{\s*reference:i0.MyDirective/);
|
|
|
|
expect(genSource).toContain('token:{identifier:{reference:i0.Dep}}');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should create @NgModule summaries', () => {
|
|
|
|
const appDir = {
|
|
|
|
'app.module.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { NgModule } from '@angular/core';
|
|
|
|
|
|
|
|
export class Dep {}
|
|
|
|
|
|
|
|
@NgModule()
|
|
|
|
export class MyModule {
|
|
|
|
constructor(d: Dep) {}
|
|
|
|
}
|
|
|
|
`
|
2017-05-17 18:39:08 -04:00
|
|
|
};
|
|
|
|
const rootDir = {'app': appDir};
|
|
|
|
|
|
|
|
const genFile =
|
|
|
|
compileApp(rootDir).genFiles.find(f => f.genFileUrl === '/app/app.module.ngsummary.ts');
|
|
|
|
const genSource = toTypeScript(genFile);
|
|
|
|
|
|
|
|
expect(genSource).toContain(`import * as i0 from '/app/app.module'`);
|
|
|
|
expect(genSource).toContain('export function MyModuleNgSummary()');
|
|
|
|
// Note: CompileSummaryKind.NgModule = 2
|
|
|
|
expect(genSource).toMatch(/summaryKind:2,\s*type:\{\s*reference:i0.MyModule/);
|
|
|
|
expect(genSource).toContain('token:{identifier:{reference:i0.Dep}}');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should embed useClass provider summaries in @Directive summaries', () => {
|
|
|
|
const appDir = {
|
|
|
|
'app.service.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { Injectable } from '@angular/core';
|
|
|
|
|
|
|
|
export class Dep {}
|
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class MyService {
|
|
|
|
constructor(d: Dep) {}
|
|
|
|
}
|
|
|
|
`,
|
2017-05-17 18:39:08 -04:00
|
|
|
'app.module.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { Directive, NgModule } from '@angular/core';
|
|
|
|
import { MyService } from './app.service';
|
|
|
|
|
|
|
|
@Directive({
|
|
|
|
selector: '[myDir]',
|
|
|
|
providers: [MyService]
|
|
|
|
})
|
|
|
|
export class MyDirective {}
|
|
|
|
|
|
|
|
@NgModule({declarations: [MyDirective]})
|
|
|
|
export class MyModule {}
|
|
|
|
`
|
2017-05-17 18:39:08 -04:00
|
|
|
};
|
|
|
|
const rootDir = {'app': appDir};
|
|
|
|
|
|
|
|
const genFile =
|
|
|
|
compileApp(rootDir).genFiles.find(f => f.genFileUrl === '/app/app.module.ngsummary.ts');
|
|
|
|
const genSource = toTypeScript(genFile);
|
|
|
|
|
|
|
|
expect(genSource).toMatch(/useClass:\{\s*reference:i1.MyService/);
|
|
|
|
// Note: CompileSummaryKind.Injectable = 3
|
|
|
|
expect(genSource).toMatch(/summaryKind:3,\s*type:\{\s*reference:i1.MyService/);
|
|
|
|
expect(genSource).toContain('token:{identifier:{reference:i1.Dep}}');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should embed useClass provider summaries into @NgModule summaries', () => {
|
|
|
|
const appDir = {
|
|
|
|
'app.service.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { Injectable } from '@angular/core';
|
|
|
|
|
|
|
|
export class Dep {}
|
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class MyService {
|
|
|
|
constructor(d: Dep) {}
|
|
|
|
}
|
|
|
|
`,
|
2017-05-17 18:39:08 -04:00
|
|
|
'app.module.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { NgModule } from '@angular/core';
|
|
|
|
import { MyService } from './app.service';
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
providers: [MyService]
|
|
|
|
})
|
|
|
|
export class MyModule {}
|
|
|
|
`
|
2017-05-17 18:39:08 -04:00
|
|
|
};
|
|
|
|
const rootDir = {'app': appDir};
|
|
|
|
|
|
|
|
const genFile =
|
|
|
|
compileApp(rootDir).genFiles.find(f => f.genFileUrl === '/app/app.module.ngsummary.ts');
|
|
|
|
const genSource = toTypeScript(genFile);
|
|
|
|
|
|
|
|
expect(genSource).toMatch(/useClass:\{\s*reference:i1.MyService/);
|
|
|
|
// Note: CompileSummaryKind.Injectable = 3
|
|
|
|
expect(genSource).toMatch(/summaryKind:3,\s*type:\{\s*reference:i1.MyService/);
|
|
|
|
expect(genSource).toContain('token:{identifier:{reference:i1.Dep}}');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should reference declared @Directive and @Pipe summaries in @NgModule summaries', () => {
|
|
|
|
const appDir = {
|
|
|
|
'app.module.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { Directive, Pipe, NgModule } from '@angular/core';
|
|
|
|
|
|
|
|
@Directive({selector: '[myDir]'})
|
|
|
|
export class MyDirective {}
|
|
|
|
|
|
|
|
@Pipe({name: 'myPipe'})
|
|
|
|
export class MyPipe {}
|
|
|
|
|
|
|
|
@NgModule({declarations: [MyDirective, MyPipe]})
|
|
|
|
export class MyModule {}
|
|
|
|
`
|
2017-05-17 18:39:08 -04:00
|
|
|
};
|
|
|
|
const rootDir = {'app': appDir};
|
2017-04-26 12:24:42 -04:00
|
|
|
|
2017-05-17 18:39:08 -04:00
|
|
|
const genFile =
|
|
|
|
compileApp(rootDir).genFiles.find(f => f.genFileUrl === '/app/app.module.ngsummary.ts');
|
|
|
|
const genSource = toTypeScript(genFile);
|
2017-04-26 12:24:42 -04:00
|
|
|
|
2017-05-17 18:39:08 -04:00
|
|
|
expect(genSource).toMatch(
|
|
|
|
/export function MyModuleNgSummary()[^;]*,\s*MyDirectiveNgSummary,\s*MyPipeNgSummary\s*\]\s*;/);
|
|
|
|
});
|
2017-04-26 12:24:42 -04:00
|
|
|
|
2017-05-17 18:39:08 -04:00
|
|
|
it('should reference imported @NgModule summaries in @NgModule summaries', () => {
|
|
|
|
const appDir = {
|
|
|
|
'app.module.ts': `
|
2017-04-26 12:24:42 -04:00
|
|
|
import { NgModule } from '@angular/core';
|
|
|
|
|
|
|
|
@NgModule()
|
|
|
|
export class MyImportedModule {}
|
|
|
|
|
|
|
|
@NgModule({imports: [MyImportedModule]})
|
|
|
|
export class MyModule {}
|
|
|
|
`
|
2017-05-17 18:39:08 -04:00
|
|
|
};
|
|
|
|
const rootDir = {'app': appDir};
|
2017-04-26 12:24:42 -04:00
|
|
|
|
2017-05-17 18:39:08 -04:00
|
|
|
const genFile =
|
|
|
|
compileApp(rootDir).genFiles.find(f => f.genFileUrl === '/app/app.module.ngsummary.ts');
|
|
|
|
const genSource = toTypeScript(genFile);
|
2017-04-26 12:24:42 -04:00
|
|
|
|
2017-05-17 18:39:08 -04:00
|
|
|
expect(genSource).toMatch(
|
|
|
|
/export function MyModuleNgSummary()[^;]*,\s*MyImportedModuleNgSummary\s*\]\s*;/);
|
|
|
|
});
|
2017-04-26 12:24:42 -04:00
|
|
|
|
|
|
|
it('should create and use reexports for imported NgModules ' +
|
|
|
|
'accross compilation units',
|
2017-05-17 18:39:08 -04:00
|
|
|
() => {
|
2017-04-26 12:24:42 -04:00
|
|
|
const lib1In = {
|
|
|
|
'lib1': {
|
|
|
|
'module.ts': `
|
|
|
|
import { NgModule } from '@angular/core';
|
|
|
|
|
|
|
|
@NgModule()
|
|
|
|
export class Lib1Module {}
|
|
|
|
`,
|
|
|
|
'reexport.ts': `
|
|
|
|
import { NgModule } from '@angular/core';
|
|
|
|
|
|
|
|
@NgModule()
|
|
|
|
export class ReexportModule {}
|
|
|
|
|
|
|
|
export const reexports: any[] = [ ReexportModule ];
|
|
|
|
`,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
const {outDir: lib2In, genFiles: lib1Gen} = compileApp(lib1In, {useSummaries: true});
|
|
|
|
|
|
|
|
lib2In['lib2'] = {
|
|
|
|
'module.ts': `
|
|
|
|
import { NgModule } from '@angular/core';
|
|
|
|
import { Lib1Module } from '../lib1/module';
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
imports: [Lib1Module]
|
|
|
|
})
|
|
|
|
export class Lib2Module {}
|
|
|
|
`,
|
|
|
|
'reexport.ts': `
|
|
|
|
import { reexports as reexports_lib1 } from '../lib1/reexport';
|
|
|
|
export const reexports: any[] = [ reexports_lib1 ];
|
|
|
|
`,
|
|
|
|
};
|
|
|
|
const {outDir: lib3In, genFiles: lib2Gen} = compileApp(lib2In, {useSummaries: true});
|
|
|
|
|
|
|
|
const lib2ModuleNgSummary = lib2Gen.find(f => f.genFileUrl === '/lib2/module.ngsummary.ts');
|
|
|
|
const lib2ReexportNgSummary =
|
|
|
|
lib2Gen.find(f => f.genFileUrl === '/lib2/reexport.ngsummary.ts');
|
|
|
|
|
|
|
|
// ngsummaries should add reexports for imported NgModules from a direct dependency
|
2017-05-17 14:21:08 -04:00
|
|
|
expect(toTypeScript(lib2ModuleNgSummary))
|
2017-04-26 12:24:42 -04:00
|
|
|
.toContain(
|
|
|
|
`export {Lib1ModuleNgSummary as Lib1Module_1NgSummary} from '/lib1/module.ngsummary'`);
|
|
|
|
// ngsummaries should add reexports for reexported values from a direct dependency
|
2017-05-17 14:21:08 -04:00
|
|
|
expect(toTypeScript(lib2ReexportNgSummary))
|
2017-04-26 12:24:42 -04:00
|
|
|
.toContain(
|
|
|
|
`export {ReexportModuleNgSummary as ReexportModule_2NgSummary} from '/lib1/reexport.ngsummary'`);
|
|
|
|
|
|
|
|
lib3In['lib3'] = {
|
|
|
|
'module.ts': `
|
|
|
|
import { NgModule } from '@angular/core';
|
|
|
|
import { Lib2Module } from '../lib2/module';
|
|
|
|
import { reexports } from '../lib2/reexport';
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
imports: [Lib2Module, reexports]
|
|
|
|
})
|
|
|
|
export class Lib3Module {}
|
|
|
|
`,
|
|
|
|
'reexport.ts': `
|
|
|
|
import { reexports as reexports_lib2 } from '../lib2/reexport';
|
|
|
|
export const reexports: any[] = [ reexports_lib2 ];
|
|
|
|
`,
|
|
|
|
};
|
|
|
|
|
|
|
|
const lib3Gen = compileApp(lib3In, {useSummaries: true}).genFiles;
|
|
|
|
const lib3ModuleNgSummary = lib3Gen.find(f => f.genFileUrl === '/lib3/module.ngsummary.ts');
|
|
|
|
const lib3ReexportNgSummary =
|
|
|
|
lib3Gen.find(f => f.genFileUrl === '/lib3/reexport.ngsummary.ts');
|
|
|
|
|
|
|
|
// ngsummary.ts files should use the reexported values from direct and deep deps
|
2017-05-17 14:21:08 -04:00
|
|
|
const lib3ModuleNgSummarySource = toTypeScript(lib3ModuleNgSummary);
|
|
|
|
expect(lib3ModuleNgSummarySource).toContain(`import * as i4 from '/lib2/module.ngsummary'`);
|
|
|
|
expect(lib3ModuleNgSummarySource)
|
2017-05-11 13:28:48 -04:00
|
|
|
.toContain(`import * as i5 from '/lib2/reexport.ngsummary'`);
|
2017-05-17 14:21:08 -04:00
|
|
|
expect(lib3ModuleNgSummarySource)
|
2017-04-26 12:24:42 -04:00
|
|
|
.toMatch(
|
2017-05-11 13:28:48 -04:00
|
|
|
/export function Lib3ModuleNgSummary()[^;]*,\s*i4.Lib1Module_1NgSummary,\s*i4.Lib2ModuleNgSummary,\s*i5.ReexportModule_2NgSummary\s*\]\s*;/);
|
2017-04-26 12:24:42 -04:00
|
|
|
|
|
|
|
// ngsummaries should add reexports for imported NgModules from a deep dependency
|
2017-05-17 14:21:08 -04:00
|
|
|
expect(lib3ModuleNgSummarySource)
|
2017-04-26 12:24:42 -04:00
|
|
|
.toContain(
|
|
|
|
`export {Lib1Module_1NgSummary as Lib1Module_1NgSummary,Lib2ModuleNgSummary as Lib2Module_2NgSummary} from '/lib2/module.ngsummary'`);
|
|
|
|
// ngsummaries should add reexports for reexported values from a deep dependency
|
2017-05-17 14:21:08 -04:00
|
|
|
expect(toTypeScript(lib3ReexportNgSummary))
|
2017-04-26 12:24:42 -04:00
|
|
|
.toContain(
|
|
|
|
`export {ReexportModule_2NgSummary as ReexportModule_3NgSummary} from '/lib2/reexport.ngsummary'`);
|
2017-05-17 18:39:08 -04:00
|
|
|
});
|
2017-04-26 12:24:42 -04:00
|
|
|
});
|