fix: disable injectable-pipe migration (#30180)

Disables the injectable pipe migration until we can decide whether this is the right solution for Ivy. Rolling it out properly will involve a more detailed plan and more changes like updating the styleguide, scaffolding schematics etc.

Context for the new `test-migrations.json`: since we use the `migrations.json` both for the real migrations and for tests, it doesn't allow us to disable a schematic, but continue running its tests. This change adds the test-specific file so that we can continue running the `injectable-pipe` tests, even though the schematic itself is disabled.

PR Close #30180
This commit is contained in:
crisbeto 2019-04-28 23:04:09 +02:00 committed by Andrew Kushnir
parent 6f433887e0
commit 572b54967c
9 changed files with 234 additions and 220 deletions

View File

@ -3,6 +3,7 @@ load("//tools:defaults.bzl", "npm_package")
exports_files([ exports_files([
"tsconfig.json", "tsconfig.json",
"migrations.json", "migrations.json",
"test-migrations.json",
]) ])
npm_package( npm_package(

View File

@ -14,11 +14,6 @@
"version": "8-beta", "version": "8-beta",
"description": "Warns developers if values are assigned to template variables", "description": "Warns developers if values are assigned to template variables",
"factory": "./migrations/template-var-assignment/index" "factory": "./migrations/template-var-assignment/index"
},
"migration-v8-injectable-pipe": {
"version": "8-beta",
"description": "Migrates all Pipe classes so that they have an Injectable annotation",
"factory": "./migrations/injectable-pipe/index"
} }
} }
} }

View File

@ -0,0 +1,20 @@
{
"schematics": {
"migration-move-document": {
"description": "Migrates DOCUMENT Injection token from platform-browser imports to common import",
"factory": "./migrations/move-document/index"
},
"migration-static-queries": {
"description": "Migrates ViewChild and ContentChild to explicit query timing",
"factory": "./migrations/static-queries/index"
},
"migration-template-local-variables": {
"description": "Warns developers if values are assigned to template variables",
"factory": "./migrations/template-var-assignment/index"
},
"migration-injectable-pipe": {
"description": "Migrates all Pipe classes so that they have an Injectable annotation",
"factory": "./migrations/injectable-pipe/index"
}
}
}

View File

@ -5,7 +5,7 @@ ts_library(
testonly = True, testonly = True,
srcs = glob(["**/*.ts"]), srcs = glob(["**/*.ts"]),
data = [ data = [
"//packages/core/schematics:migrations.json", "//packages/core/schematics:test-migrations.json",
], ],
deps = [ deps = [
"//packages/core/schematics/migrations/injectable-pipe", "//packages/core/schematics/migrations/injectable-pipe",

View File

@ -20,7 +20,7 @@ describe('injectable pipe migration', () => {
let previousWorkingDir: string; let previousWorkingDir: string;
beforeEach(() => { beforeEach(() => {
runner = new SchematicTestRunner('test', require.resolve('../migrations.json')); runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
host = new TempScopedNodeJsSyncHost(); host = new TempScopedNodeJsSyncHost();
tree = new UnitTestTree(new HostTree(host)); tree = new UnitTestTree(new HostTree(host));
@ -123,5 +123,5 @@ describe('injectable pipe migration', () => {
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents)); host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
} }
function runMigration() { runner.runSchematic('migration-v8-injectable-pipe', {}, tree); } function runMigration() { runner.runSchematic('migration-injectable-pipe', {}, tree); }
}); });

View File

@ -20,7 +20,7 @@ describe('move-document migration', () => {
let previousWorkingDir: string; let previousWorkingDir: string;
beforeEach(() => { beforeEach(() => {
runner = new SchematicTestRunner('test', require.resolve('../migrations.json')); runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
host = new TempScopedNodeJsSyncHost(); host = new TempScopedNodeJsSyncHost();
tree = new UnitTestTree(new HostTree(host)); tree = new UnitTestTree(new HostTree(host));
@ -151,5 +151,5 @@ describe('move-document migration', () => {
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents)); host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
} }
function runMigration() { runner.runSchematic('migration-v8-move-document', {}, tree); } function runMigration() { runner.runSchematic('migration-move-document', {}, tree); }
}); });

View File

@ -21,7 +21,7 @@ describe('static-queries migration with template strategy', () => {
let warnOutput: string[]; let warnOutput: string[];
beforeEach(() => { beforeEach(() => {
runner = new SchematicTestRunner('test', require.resolve('../migrations.json')); runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
host = new TempScopedNodeJsSyncHost(); host = new TempScopedNodeJsSyncHost();
tree = new UnitTestTree(new HostTree(host)); tree = new UnitTestTree(new HostTree(host));
@ -97,7 +97,7 @@ describe('static-queries migration with template strategy', () => {
} }
async function runMigration() { async function runMigration() {
await runner.runSchematicAsync('migration-v8-static-queries', {}, tree).toPromise(); await runner.runSchematicAsync('migration-static-queries', {}, tree).toPromise();
} }
describe('ViewChild', () => { describe('ViewChild', () => {
@ -105,7 +105,7 @@ describe('static-queries migration with template strategy', () => {
it('should detect queries selecting elements through template reference', async() => { it('should detect queries selecting elements through template reference', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, NgModule, ViewChild} from '@angular/core'; import {Component, NgModule, ViewChild} from '@angular/core';
@Component({template: \` @Component({template: \`
<ng-template> <ng-template>
<button #myButton>My Button</button> <button #myButton>My Button</button>
@ -118,7 +118,7 @@ describe('static-queries migration with template strategy', () => {
private @ViewChild('myButton') query: any; private @ViewChild('myButton') query: any;
private @ViewChild('myStaticButton') query2: any; private @ViewChild('myStaticButton') query2: any;
} }
@NgModule({declarations: [MyComp]}) @NgModule({declarations: [MyComp]})
export class MyModule {} export class MyModule {}
`); `);
@ -134,7 +134,7 @@ describe('static-queries migration with template strategy', () => {
it('should detect queries selecting ng-template as static', async() => { it('should detect queries selecting ng-template as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, NgModule, ViewChild} from '@angular/core'; import {Component, NgModule, ViewChild} from '@angular/core';
@Component({template: \` @Component({template: \`
<ng-template #myTmpl> <ng-template #myTmpl>
My template My template
@ -143,7 +143,7 @@ describe('static-queries migration with template strategy', () => {
export class MyComp { export class MyComp {
private @ViewChild('myTmpl') query: any; private @ViewChild('myTmpl') query: any;
} }
@NgModule({declarations: [MyComp]}) @NgModule({declarations: [MyComp]})
export class MyModule {} export class MyModule {}
`); `);
@ -157,7 +157,7 @@ describe('static-queries migration with template strategy', () => {
it('should detect queries selecting component view providers through string token', async() => { it('should detect queries selecting component view providers through string token', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, Directive, NgModule, ViewChild} from '@angular/core'; import {Component, Directive, NgModule, ViewChild} from '@angular/core';
@Directive({ @Directive({
selector: '[myDirective]', selector: '[myDirective]',
providers: [ providers: [
@ -165,7 +165,7 @@ describe('static-queries migration with template strategy', () => {
] ]
}) })
export class MyDirective {} export class MyDirective {}
@Directive({ @Directive({
selector: '[myDirective2]', selector: '[myDirective2]',
providers: [ providers: [
@ -173,13 +173,13 @@ describe('static-queries migration with template strategy', () => {
] ]
}) })
export class MyDirective2 {} export class MyDirective2 {}
@Component({templateUrl: './my-tmpl.html'}) @Component({templateUrl: './my-tmpl.html'})
export class MyComp { export class MyComp {
private @ViewChild('my-token') query: any; private @ViewChild('my-token') query: any;
private @ViewChild('my-token-2') query2: any; private @ViewChild('my-token-2') query2: any;
} }
@NgModule({declarations: [MyComp, MyDirective, MyDirective2]}) @NgModule({declarations: [MyComp, MyDirective, MyDirective2]})
export class MyModule {} export class MyModule {}
`); `);
@ -202,28 +202,28 @@ describe('static-queries migration with template strategy', () => {
it('should detect queries selecting component view providers using class token', async() => { it('should detect queries selecting component view providers using class token', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, Directive, NgModule, ViewChild} from '@angular/core'; import {Component, Directive, NgModule, ViewChild} from '@angular/core';
export class MyService {} export class MyService {}
export class MyService2 {} export class MyService2 {}
@Directive({ @Directive({
selector: '[myDirective]', selector: '[myDirective]',
providers: [MyService] providers: [MyService]
}) })
export class MyDirective {} export class MyDirective {}
@Directive({ @Directive({
selector: '[myDirective2]', selector: '[myDirective2]',
providers: [MyService2] providers: [MyService2]
}) })
export class MyDirective2 {} export class MyDirective2 {}
@Component({templateUrl: './my-tmpl.html'}) @Component({templateUrl: './my-tmpl.html'})
export class MyComp { export class MyComp {
private @ViewChild(MyService) query: any; private @ViewChild(MyService) query: any;
private @ViewChild(MyService2) query2: any; private @ViewChild(MyService2) query2: any;
} }
@NgModule({declarations: [MyComp, MyDirective, MyDirective2]}) @NgModule({declarations: [MyComp, MyDirective, MyDirective2]})
export class MyModule {} export class MyModule {}
`); `);
@ -247,7 +247,7 @@ describe('static-queries migration with template strategy', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, NgModule, ViewChild} from '@angular/core'; import {Component, NgModule, ViewChild} from '@angular/core';
import {HomeComponent, HomeComponent2} from './home-comp'; import {HomeComponent, HomeComponent2} from './home-comp';
@Component({ @Component({
template: \` template: \`
<home-comp></home-comp> <home-comp></home-comp>
@ -260,20 +260,20 @@ describe('static-queries migration with template strategy', () => {
private @ViewChild(HomeComponent) query: any; private @ViewChild(HomeComponent) query: any;
private @ViewChild(HomeComponent2) query2: any; private @ViewChild(HomeComponent2) query2: any;
} }
@NgModule({declarations: [MyComp, HomeComponent, HomeComponent2]}) @NgModule({declarations: [MyComp, HomeComponent, HomeComponent2]})
export class MyModule {} export class MyModule {}
`); `);
writeFile(`/home-comp.ts`, ` writeFile(`/home-comp.ts`, `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
selector: 'home-comp', selector: 'home-comp',
template: '<span>Home</span>' template: '<span>Home</span>'
}) })
export class HomeComponent {} export class HomeComponent {}
@Component({ @Component({
selector: 'home-comp2', selector: 'home-comp2',
template: '<span>Home 2</span>' template: '<span>Home 2</span>'
@ -294,12 +294,12 @@ describe('static-queries migration with template strategy', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, NgModule, ViewChild} from '@angular/core'; import {Component, NgModule, ViewChild} from '@angular/core';
import {MyLibComponent} from 'my-lib'; import {MyLibComponent} from 'my-lib';
@Component({templateUrl: './my-tmpl.html'}) @Component({templateUrl: './my-tmpl.html'})
export class MyComp { export class MyComp {
private @ViewChild(MyLibComponent) query: any; private @ViewChild(MyLibComponent) query: any;
} }
@NgModule({declarations: [MyComp, MyLibComponent]}) @NgModule({declarations: [MyComp, MyLibComponent]})
export class MyModule {} export class MyModule {}
`); `);
@ -319,12 +319,12 @@ describe('static-queries migration with template strategy', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, NgModule, ViewChild} from '@angular/core'; import {Component, NgModule, ViewChild} from '@angular/core';
import {MyLibComponent} from 'my-lib'; import {MyLibComponent} from 'my-lib';
@Component({templateUrl: './my-tmpl.html'}) @Component({templateUrl: './my-tmpl.html'})
export class MyComp { export class MyComp {
private @ViewChild(MyLibComponent) query: any; private @ViewChild(MyLibComponent) query: any;
} }
@NgModule({declarations: [MyComp, MyLibComponent]}) @NgModule({declarations: [MyComp, MyLibComponent]})
export class MyModule {} export class MyModule {}
`); `);
@ -345,16 +345,16 @@ describe('static-queries migration with template strategy', () => {
it('should detect queries within structural directive', async() => { it('should detect queries within structural directive', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, Directive, NgModule, ViewChild} from '@angular/core'; import {Component, Directive, NgModule, ViewChild} from '@angular/core';
@Directive({selector: '[ngIf]'}) @Directive({selector: '[ngIf]'})
export class FakeNgIf {} export class FakeNgIf {}
@Component({templateUrl: 'my-tmpl.html'}) @Component({templateUrl: 'my-tmpl.html'})
export class MyComp { export class MyComp {
private @ViewChild('myRef') query: any; private @ViewChild('myRef') query: any;
private @ViewChild('myRef2') query2: any; private @ViewChild('myRef2') query2: any;
} }
@NgModule({declarations: [MyComp, FakeNgIf]}) @NgModule({declarations: [MyComp, FakeNgIf]})
export class MyModule {} export class MyModule {}
`); `);
@ -375,14 +375,14 @@ describe('static-queries migration with template strategy', () => {
it('should detect inherited queries', async() => { it('should detect inherited queries', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, NgModule, ViewChild} from '@angular/core'; import {Component, NgModule, ViewChild} from '@angular/core';
export class BaseClass { export class BaseClass {
@ViewChild('myRef') query: any; @ViewChild('myRef') query: any;
} }
@Component({templateUrl: 'my-tmpl.html'}) @Component({templateUrl: 'my-tmpl.html'})
export class MyComp extends BaseClass {} export class MyComp extends BaseClass {}
@NgModule({declarations: [MyComp]}) @NgModule({declarations: [MyComp]})
export class MyModule {} export class MyModule {}
`); `);
@ -400,7 +400,7 @@ describe('static-queries migration with template strategy', () => {
it('should add a todo if a query is not declared in any component', async() => { it('should add a todo if a query is not declared in any component', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, NgModule, ViewChild, SomeToken} from '@angular/core'; import {Component, NgModule, ViewChild, SomeToken} from '@angular/core';
export class NotAComponent { export class NotAComponent {
@ViewChild('myRef', {read: SomeToken}) query: any; @ViewChild('myRef', {read: SomeToken}) query: any;
} }
@ -420,17 +420,17 @@ describe('static-queries migration with template strategy', () => {
it('should add a todo if a query is used multiple times with different timing', async() => { it('should add a todo if a query is used multiple times with different timing', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, NgModule, ViewChild} from '@angular/core'; import {Component, NgModule, ViewChild} from '@angular/core';
export class BaseClass { export class BaseClass {
@ViewChild('myRef') query: any; @ViewChild('myRef') query: any;
} }
@Component({template: '<ng-template><p #myRef></p></ng-template>'}) @Component({template: '<ng-template><p #myRef></p></ng-template>'})
export class FirstComp extends BaseClass {} export class FirstComp extends BaseClass {}
@Component({template: '<span #myRef></span>'}) @Component({template: '<span #myRef></span>'})
export class SecondComp extends BaseClass {} export class SecondComp extends BaseClass {}
@NgModule({declarations: [FirstComp, SecondComp]}) @NgModule({declarations: [FirstComp, SecondComp]})
export class MyModule {} export class MyModule {}
`); `);
@ -448,12 +448,12 @@ describe('static-queries migration with template strategy', () => {
it('should gracefully exit migration if queries could not be analyzed', async() => { it('should gracefully exit migration if queries could not be analyzed', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ViewChild} from '@angular/core'; import {Component, ViewChild} from '@angular/core';
@Component({template: '<ng-template><p #myRef></p></ng-template>'}) @Component({template: '<ng-template><p #myRef></p></ng-template>'})
export class MyComp { export class MyComp {
@ViewChild('myRef') query: any; @ViewChild('myRef') query: any;
} }
// **NOTE**: Analysis will fail as there is no "NgModule" that declares the component. // **NOTE**: Analysis will fail as there is no "NgModule" that declares the component.
`); `);
@ -533,7 +533,7 @@ describe('static-queries migration with template strategy', () => {
writeFile('/src/test.ts', ` writeFile('/src/test.ts', `
import {ViewChild} from '@angular/core'; import {ViewChild} from '@angular/core';
import {AppComponent} from './app.component'; import {AppComponent} from './app.component';
@Component({template: '<span #test>Test</span>'}) @Component({template: '<span #test>Test</span>'})
class MyTestComponent { class MyTestComponent {
@ViewChild('test') query: any; @ViewChild('test') query: any;
@ -542,7 +542,7 @@ describe('static-queries migration with template strategy', () => {
writeFile('/src/app.component.ts', ` writeFile('/src/app.component.ts', `
import {Component, ViewChild} from '@angular/core'; import {Component, ViewChild} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class AppComponent { export class AppComponent {
@ViewChild('test') query: any; @ViewChild('test') query: any;
@ -552,7 +552,7 @@ describe('static-queries migration with template strategy', () => {
writeFile('/src/app.module.ts', ` writeFile('/src/app.module.ts', `
import {NgModule} from '@angular/core'; import {NgModule} from '@angular/core';
import {AppComponent} from './app.component'; import {AppComponent} from './app.component';
@NgModule({declarations: [AppComponent]}) @NgModule({declarations: [AppComponent]})
export class MyModule {} export class MyModule {}
`); `);

View File

@ -26,7 +26,7 @@ describe('static-queries migration with usage strategy', () => {
afterAll(() => process.env['NG_STATIC_QUERY_USAGE_STRATEGY'] = ''); afterAll(() => process.env['NG_STATIC_QUERY_USAGE_STRATEGY'] = '');
beforeEach(() => { beforeEach(() => {
runner = new SchematicTestRunner('test', require.resolve('../migrations.json')); runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
host = new TempScopedNodeJsSyncHost(); host = new TempScopedNodeJsSyncHost();
tree = new UnitTestTree(new HostTree(host)); tree = new UnitTestTree(new HostTree(host));
@ -58,11 +58,11 @@ describe('static-queries migration with usage strategy', () => {
it('should mark view queries used in "ngAfterContentInit" as static', async() => { it('should mark view queries used in "ngAfterContentInit" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ViewChild} from '@angular/core'; import {Component, ViewChild} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@ViewChild('test') query: any; @ViewChild('test') query: any;
ngAfterContentInit() { ngAfterContentInit() {
this.query.classList.add('test'); this.query.classList.add('test');
} }
@ -78,11 +78,11 @@ describe('static-queries migration with usage strategy', () => {
it('should mark view queries used in "ngAfterContentChecked" as static', async() => { it('should mark view queries used in "ngAfterContentChecked" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ViewChild} from '@angular/core'; import {Component, ViewChild} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@ViewChild('test') query: any; @ViewChild('test') query: any;
ngAfterContentChecked() { ngAfterContentChecked() {
this.query.classList.add('test'); this.query.classList.add('test');
} }
@ -102,11 +102,11 @@ describe('static-queries migration with usage strategy', () => {
it('should not mark content queries used in "ngAfterContentInit" as static', async() => { it('should not mark content queries used in "ngAfterContentInit" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ContentChild} from '@angular/core'; import {Component, ContentChild} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@ContentChild('test') query: any; @ContentChild('test') query: any;
ngAfterContentInit() { ngAfterContentInit() {
this.query.classList.add('test'); this.query.classList.add('test');
} }
@ -122,11 +122,11 @@ describe('static-queries migration with usage strategy', () => {
it('should not mark content queries used in "ngAfterContentChecked" as static', async() => { it('should not mark content queries used in "ngAfterContentChecked" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ContentChild} from '@angular/core'; import {Component, ContentChild} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@ContentChild('test') query: any; @ContentChild('test') query: any;
ngAfterContentChecked() { ngAfterContentChecked() {
this.query.classList.add('test'); this.query.classList.add('test');
} }
@ -145,19 +145,19 @@ describe('static-queries migration with usage strategy', () => {
} }
async function runMigration() { async function runMigration() {
await runner.runSchematicAsync('migration-v8-static-queries', {}, tree).toPromise(); await runner.runSchematicAsync('migration-static-queries', {}, tree).toPromise();
} }
function createQueryTests(queryType: 'ViewChild' | 'ContentChild') { function createQueryTests(queryType: 'ViewChild' | 'ContentChild') {
it('should mark queries as dynamic', async() => { it('should mark queries as dynamic', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') unused: any; @${queryType}('test') unused: any;
@${queryType}('dynamic') dynamic: any; @${queryType}('dynamic') dynamic: any;
onClick() { onClick() {
this.dynamicQuery.classList.add('test'); this.dynamicQuery.classList.add('test');
} }
@ -175,13 +175,13 @@ describe('static-queries migration with usage strategy', () => {
it('should mark queries used in "ngOnChanges" as static', async() => { it('should mark queries used in "ngOnChanges" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
ngOnChanges() { ngOnChanges() {
this.query.classList.add('test'); this.query.classList.add('test');
} }
} }
`); `);
@ -195,13 +195,13 @@ describe('static-queries migration with usage strategy', () => {
it('should mark queries used in "ngOnInit" as static', async() => { it('should mark queries used in "ngOnInit" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
ngOnInit() { ngOnInit() {
this.query.classList.add('test'); this.query.classList.add('test');
} }
} }
`); `);
@ -215,13 +215,13 @@ describe('static-queries migration with usage strategy', () => {
it('should mark queries used in "ngDoCheck" as static', async() => { it('should mark queries used in "ngDoCheck" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
ngDoCheck() { ngDoCheck() {
this.query.classList.add('test'); this.query.classList.add('test');
} }
} }
`); `);
@ -235,11 +235,11 @@ describe('static-queries migration with usage strategy', () => {
it('should keep existing query options when updating timing', async() => { it('should keep existing query options when updating timing', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test', { /* test */ read: null }) query: any; @${queryType}('test', { /* test */ read: null }) query: any;
ngOnInit() { ngOnInit() {
this.query.classList.add('test'); this.query.classList.add('test');
} }
@ -255,7 +255,7 @@ describe('static-queries migration with usage strategy', () => {
it('should not overwrite existing explicit query timing', async() => { it('should not overwrite existing explicit query timing', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test', {static: /* untouched */ someVal}) query: any; @${queryType}('test', {static: /* untouched */ someVal}) query: any;
@ -271,26 +271,26 @@ describe('static-queries migration with usage strategy', () => {
it('should detect queries used in deep method chain', async() => { it('should detect queries used in deep method chain', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
// We intentionally add this comma for the second parameter in order // We intentionally add this comma for the second parameter in order
// to ensure that the migration does not incorrectly create an invalid // to ensure that the migration does not incorrectly create an invalid
// decorator call with three parameters. e.g. "ViewQuery('test', {...}, )" // decorator call with three parameters. e.g. "ViewQuery('test', {...}, )"
@${queryType}('test', ) query: any; @${queryType}('test', ) query: any;
ngOnInit() { ngOnInit() {
this.a(); this.a();
} }
a() { a() {
this.b(); this.b();
} }
b() { b() {
this.c(); this.c();
} }
c() { c() {
console.log(this.query); console.log(this.query);
} }
@ -306,16 +306,16 @@ describe('static-queries migration with usage strategy', () => {
it('should properly exit if recursive function is analyzed', async() => { it('should properly exit if recursive function is analyzed', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
ngOnInit() { ngOnInit() {
this.recursive(); this.recursive();
} }
recursive() { recursive() {
this.recursive(); this.recursive();
} }
} }
@ -330,27 +330,27 @@ describe('static-queries migration with usage strategy', () => {
it('should detect queries used in newly instantiated classes', async() => { it('should detect queries used in newly instantiated classes', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
@${queryType}('test') query2: any; @${queryType}('test') query2: any;
ngOnInit() { ngOnInit() {
new A(this); new A(this);
new class Inline { new class Inline {
constructor(private ctx: MyComp) { constructor(private ctx: MyComp) {
this.a(); this.a();
} }
a() { a() {
this.ctx.query2.useStatically(); this.ctx.query2.useStatically();
} }
}(this); }(this);
} }
} }
export class A { export class A {
constructor(ctx: MyComp) { constructor(ctx: MyComp) {
ctx.query.test(); ctx.query.test();
@ -369,16 +369,16 @@ describe('static-queries migration with usage strategy', () => {
it('should detect queries used in parenthesized new expressions', async() => { it('should detect queries used in parenthesized new expressions', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
ngOnInit() { ngOnInit() {
new ((A))(this); new ((A))(this);
} }
} }
export class A { export class A {
constructor(ctx: MyComp) { constructor(ctx: MyComp) {
ctx.query.test(); ctx.query.test();
@ -395,11 +395,11 @@ describe('static-queries migration with usage strategy', () => {
it('should detect queries in lifecycle hook with string literal name', async() => { it('should detect queries in lifecycle hook with string literal name', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
'ngOnInit'() { 'ngOnInit'() {
this.query.test(); this.query.test();
} }
@ -415,19 +415,19 @@ describe('static-queries migration with usage strategy', () => {
it('should detect static queries within nested inheritance', async() => { it('should detect static queries within nested inheritance', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
} }
export class A extends MyComp {} export class A extends MyComp {}
export class B extends A { export class B extends A {
ngOnInit() { ngOnInit() {
this.query.testFn(); this.query.testFn();
} }
} }
`); `);
@ -440,11 +440,11 @@ describe('static-queries migration with usage strategy', () => {
it('should detect static queries used within input setters', async() => { it('should detect static queries used within input setters', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, Input, ${queryType}} from '@angular/core'; import {Component, Input, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
@Input() @Input()
get myVal() { return null; } get myVal() { return null; }
set myVal(newVal: any) { set myVal(newVal: any) {
@ -462,14 +462,14 @@ describe('static-queries migration with usage strategy', () => {
it('should detect inputs defined in metadata', async() => { it('should detect inputs defined in metadata', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({ @Component({
template: '<span #test></span>', template: '<span #test></span>',
inputs: ["myVal"], inputs: ["myVal"],
}) })
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
// We don't use the input decorator here as we want to verify // We don't use the input decorator here as we want to verify
// that it properly detects the input through the component metadata. // that it properly detects the input through the component metadata.
get myVal() { return null; } get myVal() { return null; }
@ -488,14 +488,14 @@ describe('static-queries migration with usage strategy', () => {
it('should detect aliased inputs declared in metadata', async() => { it('should detect aliased inputs declared in metadata', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({ @Component({
template: '<span #test></span>', template: '<span #test></span>',
inputs: ['prop: publicName'], inputs: ['prop: publicName'],
}) })
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
set prop(val: any) { set prop(val: any) {
this.query.test(); this.query.test();
} }
@ -511,11 +511,11 @@ describe('static-queries migration with usage strategy', () => {
it('should not mark query as static if query is used in non-input setter', async() => { it('should not mark query as static if query is used in non-input setter', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
set myProperty(val: any) { set myProperty(val: any) {
this.query.test(); this.query.test();
} }
@ -531,13 +531,13 @@ describe('static-queries migration with usage strategy', () => {
it('should detect input decorator on setter', async() => { it('should detect input decorator on setter', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Input, Component, ${queryType}} from '@angular/core'; import {Input, Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
get myProperty() { return null; } get myProperty() { return null; }
// Usually the decorator is set on the get accessor, but it's also possible // Usually the decorator is set on the get accessor, but it's also possible
// to declare the input on the setter. This ensures that it is handled properly. // to declare the input on the setter. This ensures that it is handled properly.
@Input() @Input()
@ -556,7 +556,7 @@ describe('static-queries migration with usage strategy', () => {
it('should detect setter inputs in derived classes', async() => { it('should detect setter inputs in derived classes', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({ @Component({
template: '<span #test></span>', template: '<span #test></span>',
inputs: ['childSetter'], inputs: ['childSetter'],
@ -564,7 +564,7 @@ describe('static-queries migration with usage strategy', () => {
export class MyComp { export class MyComp {
protected @${queryType}('test') query: any; protected @${queryType}('test') query: any;
} }
export class B extends MyComp { export class B extends MyComp {
set childSetter(newVal: any) { set childSetter(newVal: any) {
this.query.test(); this.query.test();
@ -581,7 +581,7 @@ describe('static-queries migration with usage strategy', () => {
it('should properly detect static query in external derived class', async() => { it('should properly detect static query in external derived class', async() => {
writeFile('/src/index.ts', ` writeFile('/src/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
@ -590,7 +590,7 @@ describe('static-queries migration with usage strategy', () => {
writeFile('/src/external.ts', ` writeFile('/src/external.ts', `
import {MyComp} from './index'; import {MyComp} from './index';
export class ExternalComp extends MyComp { export class ExternalComp extends MyComp {
ngOnInit() { ngOnInit() {
this.query.test(); this.query.test();
@ -614,30 +614,30 @@ describe('static-queries migration with usage strategy', () => {
it('should not mark queries used in promises as static', async() => { it('should not mark queries used in promises as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
private @${queryType}('test') query2: any; private @${queryType}('test') query2: any;
ngOnInit() { ngOnInit() {
const a = Promise.resolve(); const a = Promise.resolve();
Promise.resolve().then(() => { Promise.resolve().then(() => {
this.query.doSomething(); this.query.doSomething();
}); });
Promise.reject().catch(() => { Promise.reject().catch(() => {
this.query.doSomething(); this.query.doSomething();
}); });
a.then(() => {}).then(() => { a.then(() => {}).then(() => {
this.query.doSomething(); this.query.doSomething();
}); });
Promise.resolve().then(this.createPromiseCb()); Promise.resolve().then(this.createPromiseCb());
} }
createPromiseCb() { createPromiseCb() {
this.query2.doSomething(); this.query2.doSomething();
return () => { /* empty callback */} return () => { /* empty callback */}
@ -656,19 +656,19 @@ describe('static-queries migration with usage strategy', () => {
it('should handle function callbacks which statically access queries', async() => { it('should handle function callbacks which statically access queries', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
ngOnInit() { ngOnInit() {
this.callSync(() => this.query.doSomething()); this.callSync(() => this.query.doSomething());
} }
callSync(cb: Function) { callSync(cb: Function) {
this.callSync2(cb); this.callSync2(cb);
} }
callSync2(cb: Function) { callSync2(cb: Function) {
cb(); cb();
} }
@ -686,12 +686,12 @@ describe('static-queries migration with usage strategy', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
import {External} from './external'; import {External} from './external';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
ngOnInit() { ngOnInit() {
new External(() => this.query.doSomething()); new External(() => this.query.doSomething());
} }
} }
@ -700,7 +700,7 @@ describe('static-queries migration with usage strategy', () => {
writeFile('/external.ts', ` writeFile('/external.ts', `
export class External { export class External {
constructor(cb: () => void) { constructor(cb: () => void) {
// Add extra parentheses to ensure that expression is unwrapped. // Add extra parentheses to ensure that expression is unwrapped.
((cb))(); ((cb))();
} }
} }
@ -715,21 +715,21 @@ describe('static-queries migration with usage strategy', () => {
it('should handle nested functions with arguments from parent closure', async() => { it('should handle nested functions with arguments from parent closure', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
ngOnInit() { ngOnInit() {
this.callSync(() => this.query.doSomething()); this.callSync(() => this.query.doSomething());
} }
callSync(cb: Function) { callSync(cb: Function) {
function callSyncNested() { function callSyncNested() {
// The "cb" identifier comes from the "callSync" function. // The "cb" identifier comes from the "callSync" function.
cb(); cb();
} }
callSyncNested(); callSyncNested();
} }
} }
@ -744,22 +744,22 @@ describe('static-queries migration with usage strategy', () => {
it('should not mark queries used in setTimeout as static', async() => { it('should not mark queries used in setTimeout as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
private @${queryType}('test') query2: any; private @${queryType}('test') query2: any;
private @${queryType}('test') query3: any; private @${queryType}('test') query3: any;
ngOnInit() { ngOnInit() {
setTimeout(function() { setTimeout(function() {
this.query.doSomething(); this.query.doSomething();
}); });
setTimeout(createCallback(this)); setTimeout(createCallback(this));
} }
} }
function createCallback(instance: MyComp) { function createCallback(instance: MyComp) {
instance.query2.doSomething(); instance.query2.doSomething();
return () => instance.query3.doSomething(); return () => instance.query3.doSomething();
@ -779,13 +779,13 @@ describe('static-queries migration with usage strategy', () => {
it('should not mark queries used in "addEventListener" as static', async() => { it('should not mark queries used in "addEventListener" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ElementRef, ${queryType}} from '@angular/core'; import {Component, ElementRef, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
constructor(private elementRef: ElementRef) {} constructor(private elementRef: ElementRef) {}
ngOnInit() { ngOnInit() {
this.elementRef.addEventListener(() => { this.elementRef.addEventListener(() => {
this.query.classList.add('test'); this.query.classList.add('test');
@ -803,13 +803,13 @@ describe('static-queries migration with usage strategy', () => {
it('should not mark queries used in "requestAnimationFrame" as static', async() => { it('should not mark queries used in "requestAnimationFrame" as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ElementRef, ${queryType}} from '@angular/core'; import {Component, ElementRef, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
constructor(private elementRef: ElementRef) {} constructor(private elementRef: ElementRef) {}
ngOnInit() { ngOnInit() {
requestAnimationFrame(() => { requestAnimationFrame(() => {
this.query.classList.add('test'); this.query.classList.add('test');
@ -827,17 +827,17 @@ describe('static-queries migration with usage strategy', () => {
it('should mark queries used in immediately-invoked function expression as static', async() => { it('should mark queries used in immediately-invoked function expression as static', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
private @${queryType}('test') query2: any; private @${queryType}('test') query2: any;
ngOnInit() { ngOnInit() {
(() => { (() => {
this.query.usedStatically(); this.query.usedStatically();
})(); })();
(function(ctx) { (function(ctx) {
ctx.query2.useStatically(); ctx.query2.useStatically();
})(this); })(this);
@ -857,11 +857,11 @@ describe('static-queries migration with usage strategy', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
import {externalFn} from './external'; import {externalFn} from './external';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
ngOnInit() { ngOnInit() {
externalFn(this); externalFn(this);
} }
@ -870,7 +870,7 @@ describe('static-queries migration with usage strategy', () => {
writeFile('/external.ts', ` writeFile('/external.ts', `
import {MyComp} from './index'; import {MyComp} from './index';
export function externalFn(ctx: MyComp) { export function externalFn(ctx: MyComp) {
ctx.query.usedStatically(); ctx.query.usedStatically();
} }
@ -885,15 +885,15 @@ describe('static-queries migration with usage strategy', () => {
it('should detect static queries used through getter property access', async() => { it('should detect static queries used through getter property access', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
get myProp() { get myProp() {
return this.query.myValue; return this.query.myValue;
} }
ngOnInit() { ngOnInit() {
this.myProp.test(); this.myProp.test();
} }
@ -910,17 +910,17 @@ describe('static-queries migration with usage strategy', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
import {External} from './external'; import {External} from './external';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
@${queryType}('test') query: any; @${queryType}('test') query: any;
private external = new External(this); private external = new External(this);
get myProp() { get myProp() {
return this.query.myValue; return this.query.myValue;
} }
ngOnInit() { ngOnInit() {
console.log(this.external.query); console.log(this.external.query);
} }
@ -929,10 +929,10 @@ describe('static-queries migration with usage strategy', () => {
writeFile('/external.ts', ` writeFile('/external.ts', `
import {MyComp} from './index'; import {MyComp} from './index';
export class External { export class External {
constructor(private comp: MyComp) {} constructor(private comp: MyComp) {}
set query() { /** noop */ } set query() { /** noop */ }
get query() { return this.comp.query; } get query() { return this.comp.query; }
} }
@ -947,16 +947,16 @@ describe('static-queries migration with usage strategy', () => {
it('should not mark queries as static if a value is assigned to accessor property', async() => { it('should not mark queries as static if a value is assigned to accessor property', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
set myProp(value: any) { /* noop */} set myProp(value: any) { /* noop */}
get myProp() { get myProp() {
return this.query.myValue; return this.query.myValue;
} }
ngOnInit() { ngOnInit() {
this.myProp = true; this.myProp = true;
} }
@ -972,16 +972,16 @@ describe('static-queries migration with usage strategy', () => {
it('should mark queries as static if non-input setter uses query', async() => { it('should mark queries as static if non-input setter uses query', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
get myProp() { return null; } get myProp() { return null; }
set myProp(value: any) { set myProp(value: any) {
this.query.doSomething(); this.query.doSomething();
} }
ngOnInit() { ngOnInit() {
this.myProp = 'newValue'; this.myProp = 'newValue';
} }
@ -997,17 +997,17 @@ describe('static-queries migration with usage strategy', () => {
it('should check setter and getter when using compound assignment', async() => { it('should check setter and getter when using compound assignment', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
private @${queryType}('test') query2: any; private @${queryType}('test') query2: any;
get myProp() { return this.query2 } get myProp() { return this.query2 }
set myProp(value: any) { set myProp(value: any) {
this.query.doSomething(); this.query.doSomething();
} }
ngOnInit() { ngOnInit() {
this.myProp *= 5; this.myProp *= 5;
} }
@ -1025,14 +1025,14 @@ describe('static-queries migration with usage strategy', () => {
it('should check getters when using comparison operator in binary expression', async() => { it('should check getters when using comparison operator in binary expression', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;
get myProp() { return this.query } get myProp() { return this.query }
set myProp(value: any) { /* noop */ } set myProp(value: any) { /* noop */ }
ngOnInit() { ngOnInit() {
if (this.myProp === 3) { if (this.myProp === 3) {
// noop // noop
@ -1050,29 +1050,29 @@ describe('static-queries migration with usage strategy', () => {
it('should check derived abstract class methods', async() => { it('should check derived abstract class methods', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
export abstract class RootBaseClass { export abstract class RootBaseClass {
abstract getQuery(): any; abstract getQuery(): any;
ngOnInit() { ngOnInit() {
this.getQuery().doSomething(); this.getQuery().doSomething();
} }
} }
export abstract class BaseClass extends RootBaseClass { export abstract class BaseClass extends RootBaseClass {
abstract getQuery2(): any; abstract getQuery2(): any;
getQuery() { getQuery() {
this.getQuery2(); this.getQuery2();
} }
} }
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class Subclass extends BaseClass { export class Subclass extends BaseClass {
@${queryType}('test') query: any; @${queryType}('test') query: any;
getQuery2(): any { getQuery2(): any {
return this.query; return this.query;
} }
} }
`); `);
@ -1086,25 +1086,25 @@ describe('static-queries migration with usage strategy', () => {
it('should detect queries accessed through deep abstract class method', async() => { it('should detect queries accessed through deep abstract class method', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
export abstract class RootBaseClass { export abstract class RootBaseClass {
abstract getQuery(): any; abstract getQuery(): any;
ngOnInit() { ngOnInit() {
this.getQuery().doSomething(); this.getQuery().doSomething();
} }
} }
export abstract class BaseClass extends RootBaseClass { export abstract class BaseClass extends RootBaseClass {
/* additional layer of indirection */ /* additional layer of indirection */
} }
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class Subclass extends BaseClass { export class Subclass extends BaseClass {
@${queryType}('test') query: any; @${queryType}('test') query: any;
getQuery(): any { getQuery(): any {
return this.query; return this.query;
} }
} }
`); `);
@ -1118,19 +1118,19 @@ describe('static-queries migration with usage strategy', () => {
it('should detect queries accessed through abstract property getter', async() => { it('should detect queries accessed through abstract property getter', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
export abstract class BaseClass { export abstract class BaseClass {
abstract myQuery: any; abstract myQuery: any;
ngOnInit() { ngOnInit() {
this.myQuery.doSomething(); this.myQuery.doSomething();
} }
} }
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class Subclass extends BaseClass { export class Subclass extends BaseClass {
@${queryType}('test') query: any; @${queryType}('test') query: any;
get myQuery() { return this.query; } get myQuery() { return this.query; }
} }
`); `);
@ -1144,19 +1144,19 @@ describe('static-queries migration with usage strategy', () => {
it('should detect queries accessed through abstract property setter', async() => { it('should detect queries accessed through abstract property setter', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
export abstract class BaseClass { export abstract class BaseClass {
abstract myQuery: any; abstract myQuery: any;
ngOnInit() { ngOnInit() {
this.myQuery = "trigger"; this.myQuery = "trigger";
} }
} }
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class Subclass extends BaseClass { export class Subclass extends BaseClass {
@${queryType}('test') query: any; @${queryType}('test') query: any;
set myQuery(val: any) { this.query.doSomething() } set myQuery(val: any) { this.query.doSomething() }
get myQuery() { /* noop */ } get myQuery() { /* noop */ }
} }
@ -1171,27 +1171,27 @@ describe('static-queries migration with usage strategy', () => {
it('should detect query usage in abstract class methods accessing inherited query', async() => { it('should detect query usage in abstract class methods accessing inherited query', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
export abstract class RootBaseClass { export abstract class RootBaseClass {
abstract getQuery(): any; abstract getQuery(): any;
ngOnInit() { ngOnInit() {
this.getQuery().doSomething(); this.getQuery().doSomething();
} }
} }
export abstract class BaseClass extends RootBaseClass { export abstract class BaseClass extends RootBaseClass {
@${queryType}('test') query: any; @${queryType}('test') query: any;
abstract getQuery2(): any; abstract getQuery2(): any;
getQuery() { getQuery() {
this.getQuery2(); this.getQuery2();
} }
} }
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class Subclass extends BaseClass { export class Subclass extends BaseClass {
getQuery2(): any { getQuery2(): any {
return this.query; return this.query;
} }
@ -1207,7 +1207,7 @@ describe('static-queries migration with usage strategy', () => {
it('should detect query usage within component template', async() => { it('should detect query usage within component template', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({templateUrl: 'my-template.html'}) @Component({templateUrl: 'my-template.html'})
export class MyComponent { export class MyComponent {
@${queryType}('test') query: any; @${queryType}('test') query: any;
@ -1228,7 +1228,7 @@ describe('static-queries migration with usage strategy', () => {
it('should detect query usage with nested property read within component template', async() => { it('should detect query usage with nested property read within component template', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({templateUrl: 'my-template.html'}) @Component({templateUrl: 'my-template.html'})
export class MyComponent { export class MyComponent {
@${queryType}('test') query: any; @${queryType}('test') query: any;
@ -1250,7 +1250,7 @@ describe('static-queries migration with usage strategy', () => {
async() => { async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({templateUrl: 'my-template.html'}) @Component({templateUrl: 'my-template.html'})
export class MyComponent { export class MyComponent {
@${queryType}('test') query: any; @${queryType}('test') query: any;
@ -1274,7 +1274,7 @@ describe('static-queries migration with usage strategy', () => {
async() => { async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({templateUrl: 'my-template.html'}) @Component({templateUrl: 'my-template.html'})
export class MyComponent { export class MyComponent {
myObject: {someProp: any}; myObject: {someProp: any};
@ -1299,7 +1299,7 @@ describe('static-queries migration with usage strategy', () => {
it('should ignore queries accessed within <ng-template> element', async() => { it('should ignore queries accessed within <ng-template> element', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({templateUrl: 'my-template.html'}) @Component({templateUrl: 'my-template.html'})
export class MyComponent { export class MyComponent {
@${queryType}('test') query: any; @${queryType}('test') query: any;
@ -1308,7 +1308,7 @@ describe('static-queries migration with usage strategy', () => {
writeFile(`/my-template.html`, ` writeFile(`/my-template.html`, `
<foo #test></foo> <foo #test></foo>
<ng-template> <ng-template>
<my-comp [myInput]="query"></my-comp> <my-comp [myInput]="query"></my-comp>
</ng-template> </ng-template>
@ -1323,11 +1323,11 @@ describe('static-queries migration with usage strategy', () => {
it('should detect inherited queries used in templates', async() => { it('should detect inherited queries used in templates', async() => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
export class ParentClass { export class ParentClass {
@${queryType}('test') query: any; @${queryType}('test') query: any;
} }
@Component({templateUrl: 'my-template.html'}) @Component({templateUrl: 'my-template.html'})
export class MyComponent extends ParentClass {} export class MyComponent extends ParentClass {}
`); `);
@ -1346,7 +1346,7 @@ describe('static-queries migration with usage strategy', () => {
it('should properly handle multiple tsconfig files', async() => { it('should properly handle multiple tsconfig files', async() => {
writeFile('/src/index.ts', ` writeFile('/src/index.ts', `
import {Component, ${queryType}} from '@angular/core'; import {Component, ${queryType}} from '@angular/core';
@Component({template: '<span #test></span>'}) @Component({template: '<span #test></span>'})
export class MyComp { export class MyComp {
private @${queryType}('test') query: any; private @${queryType}('test') query: any;

View File

@ -21,7 +21,7 @@ describe('template variable assignment migration', () => {
let warnOutput: string[]; let warnOutput: string[];
beforeEach(() => { beforeEach(() => {
runner = new SchematicTestRunner('test', require.resolve('../migrations.json')); runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
host = new TempScopedNodeJsSyncHost(); host = new TempScopedNodeJsSyncHost();
tree = new UnitTestTree(new HostTree(host)); tree = new UnitTestTree(new HostTree(host));
@ -58,14 +58,12 @@ describe('template variable assignment migration', () => {
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents)); host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
} }
function runMigration() { function runMigration() { runner.runSchematic('migration-template-local-variables', {}, tree); }
runner.runSchematic('migration-v8-template-local-variables', {}, tree);
}
it('should warn for two-way data binding variable assignment', () => { it('should warn for two-way data binding variable assignment', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
template: '<cmp *ngFor="let optionName of options" [(opt)]="optionName"></cmp>', template: '<cmp *ngFor="let optionName of options" [(opt)]="optionName"></cmp>',
}) })
@ -81,7 +79,7 @@ describe('template variable assignment migration', () => {
it('should warn for two-way data binding assigning to "as" variable', () => { it('should warn for two-way data binding assigning to "as" variable', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
templateUrl: './tmpl.html', templateUrl: './tmpl.html',
}) })
@ -103,7 +101,7 @@ describe('template variable assignment migration', () => {
it('should warn for bound event assignments to "as" variable', () => { it('should warn for bound event assignments to "as" variable', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
templateUrl: './sub_dir/tmpl.html', templateUrl: './sub_dir/tmpl.html',
}) })
@ -127,7 +125,7 @@ describe('template variable assignment migration', () => {
it('should warn for bound event assignments to template "let" variables', () => { it('should warn for bound event assignments to template "let" variables', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
templateUrl: './sub_dir/tmpl.html', templateUrl: './sub_dir/tmpl.html',
}) })
@ -151,7 +149,7 @@ describe('template variable assignment migration', () => {
it('should not warn for bound event assignments to component property', () => { it('should not warn for bound event assignments to component property', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
templateUrl: './sub_dir/tmpl.html', templateUrl: './sub_dir/tmpl.html',
}) })
@ -168,7 +166,7 @@ describe('template variable assignment migration', () => {
it('should not warn for bound event assignments to template variable object property', () => { it('should not warn for bound event assignments to template variable object property', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
templateUrl: './sub_dir/tmpl.html', templateUrl: './sub_dir/tmpl.html',
}) })
@ -188,7 +186,7 @@ describe('template variable assignment migration', () => {
() => { () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
templateUrl: './sub_dir/tmpl.html', templateUrl: './sub_dir/tmpl.html',
}) })
@ -213,7 +211,7 @@ describe('template variable assignment migration', () => {
it('should not warn for property writes with template variable name but different scope', () => { it('should not warn for property writes with template variable name but different scope', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
templateUrl: './sub_dir/tmpl.html', templateUrl: './sub_dir/tmpl.html',
}) })
@ -236,7 +234,7 @@ describe('template variable assignment migration', () => {
it('should not throw an error if a detected template fails parsing', () => { it('should not throw an error if a detected template fails parsing', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
templateUrl: './sub_dir/tmpl.html', templateUrl: './sub_dir/tmpl.html',
}) })
@ -253,12 +251,12 @@ describe('template variable assignment migration', () => {
it('should be able to report multiple templates within the same source file', () => { it('should be able to report multiple templates within the same source file', () => {
writeFile('/index.ts', ` writeFile('/index.ts', `
import {Component} from '@angular/core'; import {Component} from '@angular/core';
@Component({ @Component({
template: '<ng-template let-one><a (sayHello)="one=true"></a></ng-template>', template: '<ng-template let-one><a (sayHello)="one=true"></a></ng-template>',
}) })
export class MyComp {} export class MyComp {}
@Component({ @Component({
template: '<ng-template let-two><b (greet)="two=true"></b></ng-template>', template: '<ng-template let-two><b (greet)="two=true"></b></ng-template>',
}) })