fix(compiler): allow tree-shakeable injectables to depend on string tokens (#22376)
Previously the injectable compiler assumed all tree-shakeable injectables would have dependencies that were injectables or InjectionTokens. However old code still uses string tokens (e.g. NgUpgrade and '$injector'). Using such tokens would cause the injectable compiler to crash. Now, the injectable compiler can properly generate a dependency on such a string token. PR Close #22376
This commit is contained in:
parent
8bb2f5c71d
commit
dd534471ec
@ -0,0 +1,38 @@
|
|||||||
|
/**
|
||||||
|
* @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 {Component, Inject, Injectable, NgModule} from '@angular/core';
|
||||||
|
import {BrowserModule} from '@angular/platform-browser';
|
||||||
|
import {ServerModule} from '@angular/platform-server';
|
||||||
|
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'string-app',
|
||||||
|
template: '{{data}}',
|
||||||
|
})
|
||||||
|
export class AppComponent {
|
||||||
|
data: string;
|
||||||
|
constructor(service: Service) { this.data = service.data; }
|
||||||
|
}
|
||||||
|
|
||||||
|
@NgModule({
|
||||||
|
imports: [
|
||||||
|
BrowserModule.withServerTransition({appId: 'id-app'}),
|
||||||
|
ServerModule,
|
||||||
|
],
|
||||||
|
declarations: [AppComponent],
|
||||||
|
bootstrap: [AppComponent],
|
||||||
|
providers: [{provide: 'someStringToken', useValue: 'works'}],
|
||||||
|
})
|
||||||
|
export class StringAppModule {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Injectable({scope: StringAppModule})
|
||||||
|
export class Service {
|
||||||
|
constructor(@Inject('someStringToken') readonly data: string) {}
|
||||||
|
}
|
@ -13,6 +13,7 @@ import {DepAppModuleNgFactory} from 'app_built/src/dep.ngfactory';
|
|||||||
import {HierarchyAppModuleNgFactory} from 'app_built/src/hierarchy.ngfactory';
|
import {HierarchyAppModuleNgFactory} from 'app_built/src/hierarchy.ngfactory';
|
||||||
import {RootAppModuleNgFactory} from 'app_built/src/root.ngfactory';
|
import {RootAppModuleNgFactory} from 'app_built/src/root.ngfactory';
|
||||||
import {SelfAppModuleNgFactory} from 'app_built/src/self.ngfactory';
|
import {SelfAppModuleNgFactory} from 'app_built/src/self.ngfactory';
|
||||||
|
import {StringAppModuleNgFactory} from 'app_built/src/string.ngfactory';
|
||||||
import {TokenAppModuleNgFactory} from 'app_built/src/token.ngfactory';
|
import {TokenAppModuleNgFactory} from 'app_built/src/token.ngfactory';
|
||||||
|
|
||||||
enableProdMode();
|
enableProdMode();
|
||||||
@ -77,4 +78,14 @@ describe('ngInjectableDef Bazel Integration', () => {
|
|||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('string tokens work', done => {
|
||||||
|
renderModuleFactory(StringAppModuleNgFactory, {
|
||||||
|
document: '<string-app></string-app>',
|
||||||
|
url: '/',
|
||||||
|
}).then(html => {
|
||||||
|
expect(html).toMatch(/>works<\//);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -55,10 +55,11 @@ export class InjectableCompiler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
const tokenExpr = typeof token === 'string' ? o.literal(token) : ctx.importExpr(token);
|
||||||
if (flags !== InjectFlags.Default || defaultValue !== undefined) {
|
if (flags !== InjectFlags.Default || defaultValue !== undefined) {
|
||||||
args = [ctx.importExpr(token), o.literal(defaultValue), o.literal(flags)];
|
args = [tokenExpr, o.literal(defaultValue), o.literal(flags)];
|
||||||
} else {
|
} else {
|
||||||
args = [ctx.importExpr(token)];
|
args = [tokenExpr];
|
||||||
}
|
}
|
||||||
return o.importExpr(Identifiers.inject).callFn(args);
|
return o.importExpr(Identifiers.inject).callFn(args);
|
||||||
});
|
});
|
||||||
|
Loading…
x
Reference in New Issue
Block a user