- `src/runtime_compiler.ts` -> `src/jit/compiler.ts` - `src/compiler.ts` -> `src/jit/compiler_factory.ts` - `src/offline_compiler` -> `src/aot/compiler.ts` Part of #12867
667 lines
27 KiB
TypeScript
667 lines
27 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
* found in the LICENSE file at https://angular.io/license
|
|
*/
|
|
|
|
import {Attribute, Component, ContentChild, ContentChildren, Directive, Host, HostBinding, HostListener, Inject, Injectable, Input, NgModule, Optional, Output, Pipe, Self, SkipSelf, ViewChild, ViewChildren, animate, group, keyframes, sequence, state, style, transition, trigger} from '@angular/core';
|
|
|
|
import {ReflectorReader} from '../private_import_core';
|
|
|
|
import {StaticSymbol} from './static_symbol';
|
|
|
|
const SUPPORTED_SCHEMA_VERSION = 1;
|
|
|
|
/**
|
|
* The host of the static resolver is expected to be able to provide module metadata in the form of
|
|
* ModuleMetadata. Angular 2 CLI will produce this metadata for a module whenever a .d.ts files is
|
|
* produced and the module has exported variables or classes with decorators. Module metadata can
|
|
* also be produced directly from TypeScript sources by using MetadataCollector in tools/metadata.
|
|
*/
|
|
export interface StaticReflectorHost {
|
|
/**
|
|
* Return a ModuleMetadata for the given module.
|
|
*
|
|
* @param modulePath is a string identifier for a module as an absolute path.
|
|
* @returns the metadata for the given module.
|
|
*/
|
|
getMetadataFor(modulePath: string): {[key: string]: any}|{[key: string]: any}[];
|
|
|
|
/**
|
|
* Resolve a symbol from an import statement form, to the file where it is declared.
|
|
* @param module the location imported from
|
|
* @param containingFile for relative imports, the path of the file containing the import
|
|
*/
|
|
findDeclaration(modulePath: string, symbolName: string, containingFile?: string): StaticSymbol;
|
|
|
|
getStaticSymbol(declarationFile: string, name: string, members?: string[]): StaticSymbol;
|
|
|
|
angularImportLocations(): {
|
|
coreDecorators: string,
|
|
diDecorators: string,
|
|
diMetadata: string,
|
|
diOpaqueToken: string,
|
|
animationMetadata: string,
|
|
provider: string
|
|
};
|
|
|
|
getCanonicalFileName(fileName: string): string;
|
|
}
|
|
|
|
/**
|
|
* A static reflector implements enough of the Reflector API that is necessary to compile
|
|
* templates statically.
|
|
*/
|
|
export class StaticReflector implements ReflectorReader {
|
|
private annotationCache = new Map<StaticSymbol, any[]>();
|
|
private propertyCache = new Map<StaticSymbol, {[key: string]: any}>();
|
|
private parameterCache = new Map<StaticSymbol, any[]>();
|
|
private metadataCache = new Map<string, {[key: string]: any}>();
|
|
private conversionMap = new Map<StaticSymbol, (context: StaticSymbol, args: any[]) => any>();
|
|
private opaqueToken: StaticSymbol;
|
|
|
|
constructor(private host: StaticReflectorHost) { this.initializeConversionMap(); }
|
|
|
|
importUri(typeOrFunc: StaticSymbol): string {
|
|
const staticSymbol = this.host.findDeclaration(typeOrFunc.filePath, typeOrFunc.name, '');
|
|
return staticSymbol ? staticSymbol.filePath : null;
|
|
}
|
|
|
|
resolveIdentifier(name: string, moduleUrl: string, runtime: any): any {
|
|
return this.host.findDeclaration(moduleUrl, name, '');
|
|
}
|
|
|
|
resolveEnum(enumIdentifier: any, name: string): any {
|
|
const staticSymbol: StaticSymbol = enumIdentifier;
|
|
return this.host.getStaticSymbol(staticSymbol.filePath, staticSymbol.name, [name]);
|
|
}
|
|
|
|
public annotations(type: StaticSymbol): any[] {
|
|
let annotations = this.annotationCache.get(type);
|
|
if (!annotations) {
|
|
const classMetadata = this.getTypeMetadata(type);
|
|
if (classMetadata['decorators']) {
|
|
annotations = this.simplify(type, classMetadata['decorators']);
|
|
} else {
|
|
annotations = [];
|
|
}
|
|
this.annotationCache.set(type, annotations.filter(ann => !!ann));
|
|
}
|
|
return annotations;
|
|
}
|
|
|
|
public propMetadata(type: StaticSymbol): {[key: string]: any} {
|
|
let propMetadata = this.propertyCache.get(type);
|
|
if (!propMetadata) {
|
|
const classMetadata = this.getTypeMetadata(type);
|
|
const members = classMetadata ? classMetadata['members'] : {};
|
|
propMetadata = mapStringMap(members, (propData, propName) => {
|
|
const prop = (<any[]>propData)
|
|
.find(a => a['__symbolic'] == 'property' || a['__symbolic'] == 'method');
|
|
if (prop && prop['decorators']) {
|
|
return this.simplify(type, prop['decorators']);
|
|
} else {
|
|
return [];
|
|
}
|
|
});
|
|
this.propertyCache.set(type, propMetadata);
|
|
}
|
|
return propMetadata;
|
|
}
|
|
|
|
public parameters(type: StaticSymbol): any[] {
|
|
if (!(type instanceof StaticSymbol)) {
|
|
throw new Error(`parameters received ${JSON.stringify(type)} which is not a StaticSymbol`);
|
|
}
|
|
try {
|
|
let parameters = this.parameterCache.get(type);
|
|
if (!parameters) {
|
|
const classMetadata = this.getTypeMetadata(type);
|
|
const members = classMetadata ? classMetadata['members'] : null;
|
|
const ctorData = members ? members['__ctor__'] : null;
|
|
if (ctorData) {
|
|
const ctor = (<any[]>ctorData).find(a => a['__symbolic'] == 'constructor');
|
|
const parameterTypes = <any[]>this.simplify(type, ctor['parameters'] || []);
|
|
const parameterDecorators = <any[]>this.simplify(type, ctor['parameterDecorators'] || []);
|
|
parameters = [];
|
|
parameterTypes.forEach((paramType, index) => {
|
|
const nestedResult: any[] = [];
|
|
if (paramType) {
|
|
nestedResult.push(paramType);
|
|
}
|
|
const decorators = parameterDecorators ? parameterDecorators[index] : null;
|
|
if (decorators) {
|
|
nestedResult.push(...decorators);
|
|
}
|
|
parameters.push(nestedResult);
|
|
});
|
|
}
|
|
if (!parameters) {
|
|
parameters = [];
|
|
}
|
|
this.parameterCache.set(type, parameters);
|
|
}
|
|
return parameters;
|
|
} catch (e) {
|
|
console.log(`Failed on type ${JSON.stringify(type)} with error ${e}`);
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
hasLifecycleHook(type: any, lcProperty: string): boolean {
|
|
if (!(type instanceof StaticSymbol)) {
|
|
throw new Error(
|
|
`hasLifecycleHook received ${JSON.stringify(type)} which is not a StaticSymbol`);
|
|
}
|
|
const classMetadata = this.getTypeMetadata(type);
|
|
const members = classMetadata ? classMetadata['members'] : null;
|
|
const member: any[] =
|
|
members && members.hasOwnProperty(lcProperty) ? members[lcProperty] : null;
|
|
return member ? member.some(a => a['__symbolic'] == 'method') : false;
|
|
}
|
|
|
|
private registerDecoratorOrConstructor(type: StaticSymbol, ctor: any): void {
|
|
this.conversionMap.set(type, (context: StaticSymbol, args: any[]) => new ctor(...args));
|
|
}
|
|
|
|
private registerFunction(type: StaticSymbol, fn: any): void {
|
|
this.conversionMap.set(type, (context: StaticSymbol, args: any[]) => fn.apply(undefined, args));
|
|
}
|
|
|
|
private initializeConversionMap(): void {
|
|
const {coreDecorators, diDecorators, diMetadata, diOpaqueToken, animationMetadata, provider} =
|
|
this.host.angularImportLocations();
|
|
this.opaqueToken = this.host.findDeclaration(diOpaqueToken, 'OpaqueToken');
|
|
|
|
this.registerDecoratorOrConstructor(this.host.findDeclaration(diDecorators, 'Host'), Host);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(diDecorators, 'Injectable'), Injectable);
|
|
this.registerDecoratorOrConstructor(this.host.findDeclaration(diDecorators, 'Self'), Self);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(diDecorators, 'SkipSelf'), SkipSelf);
|
|
this.registerDecoratorOrConstructor(this.host.findDeclaration(diDecorators, 'Inject'), Inject);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(diDecorators, 'Optional'), Optional);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'Attribute'), Attribute);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'ContentChild'), ContentChild);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'ContentChildren'), ContentChildren);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'ViewChild'), ViewChild);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'ViewChildren'), ViewChildren);
|
|
this.registerDecoratorOrConstructor(this.host.findDeclaration(coreDecorators, 'Input'), Input);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'Output'), Output);
|
|
this.registerDecoratorOrConstructor(this.host.findDeclaration(coreDecorators, 'Pipe'), Pipe);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'HostBinding'), HostBinding);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'HostListener'), HostListener);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'Directive'), Directive);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'Component'), Component);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(coreDecorators, 'NgModule'), NgModule);
|
|
|
|
// Note: Some metadata classes can be used directly with Provider.deps.
|
|
this.registerDecoratorOrConstructor(this.host.findDeclaration(diMetadata, 'Host'), Host);
|
|
this.registerDecoratorOrConstructor(this.host.findDeclaration(diMetadata, 'Self'), Self);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(diMetadata, 'SkipSelf'), SkipSelf);
|
|
this.registerDecoratorOrConstructor(
|
|
this.host.findDeclaration(diMetadata, 'Optional'), Optional);
|
|
|
|
this.registerFunction(this.host.findDeclaration(animationMetadata, 'trigger'), trigger);
|
|
this.registerFunction(this.host.findDeclaration(animationMetadata, 'state'), state);
|
|
this.registerFunction(this.host.findDeclaration(animationMetadata, 'transition'), transition);
|
|
this.registerFunction(this.host.findDeclaration(animationMetadata, 'style'), style);
|
|
this.registerFunction(this.host.findDeclaration(animationMetadata, 'animate'), animate);
|
|
this.registerFunction(this.host.findDeclaration(animationMetadata, 'keyframes'), keyframes);
|
|
this.registerFunction(this.host.findDeclaration(animationMetadata, 'sequence'), sequence);
|
|
this.registerFunction(this.host.findDeclaration(animationMetadata, 'group'), group);
|
|
}
|
|
|
|
/** @internal */
|
|
public simplify(context: StaticSymbol, value: any): any {
|
|
const _this = this;
|
|
let scope = BindingScope.empty;
|
|
const calling = new Map<StaticSymbol, boolean>();
|
|
|
|
function simplifyInContext(context: StaticSymbol, value: any, depth: number): any {
|
|
function resolveReference(context: StaticSymbol, expression: any): StaticSymbol {
|
|
let staticSymbol: StaticSymbol;
|
|
if (expression['module']) {
|
|
staticSymbol = _this.host.findDeclaration(
|
|
expression['module'], expression['name'], context.filePath);
|
|
} else {
|
|
staticSymbol = _this.host.getStaticSymbol(context.filePath, expression['name']);
|
|
}
|
|
return staticSymbol;
|
|
}
|
|
|
|
function resolveReferenceValue(staticSymbol: StaticSymbol): any {
|
|
const moduleMetadata = _this.getModuleMetadata(staticSymbol.filePath);
|
|
const declarationValue =
|
|
moduleMetadata ? moduleMetadata['metadata'][staticSymbol.name] : null;
|
|
return declarationValue;
|
|
}
|
|
|
|
function isOpaqueToken(context: StaticSymbol, value: any): boolean {
|
|
if (value && value.__symbolic === 'new' && value.expression) {
|
|
const target = value.expression;
|
|
if (target.__symbolic == 'reference') {
|
|
return sameSymbol(resolveReference(context, target), _this.opaqueToken);
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function simplifyCall(expression: any) {
|
|
let callContext: {[name: string]: string}|undefined = undefined;
|
|
if (expression['__symbolic'] == 'call') {
|
|
const target = expression['expression'];
|
|
let functionSymbol: StaticSymbol;
|
|
let targetFunction: any;
|
|
if (target) {
|
|
switch (target.__symbolic) {
|
|
case 'reference':
|
|
// Find the function to call.
|
|
callContext = {name: target.name};
|
|
functionSymbol = resolveReference(context, target);
|
|
targetFunction = resolveReferenceValue(functionSymbol);
|
|
break;
|
|
case 'select':
|
|
// Find the static method to call
|
|
if (target.expression.__symbolic == 'reference') {
|
|
functionSymbol = resolveReference(context, target.expression);
|
|
const classData = resolveReferenceValue(functionSymbol);
|
|
if (classData && classData.statics) {
|
|
targetFunction = classData.statics[target.member];
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if (targetFunction && targetFunction['__symbolic'] == 'function') {
|
|
if (calling.get(functionSymbol)) {
|
|
throw new Error('Recursion not supported');
|
|
}
|
|
calling.set(functionSymbol, true);
|
|
try {
|
|
const value = targetFunction['value'];
|
|
if (value && (depth != 0 || value.__symbolic != 'error')) {
|
|
// Determine the arguments
|
|
const args: any[] =
|
|
(expression['arguments'] || []).map((arg: any) => simplify(arg));
|
|
const parameters: string[] = targetFunction['parameters'];
|
|
const defaults: any[] = targetFunction.defaults;
|
|
if (defaults && defaults.length > args.length) {
|
|
args.push(...defaults.slice(args.length).map((value: any) => simplify(value)));
|
|
}
|
|
const functionScope = BindingScope.build();
|
|
for (let i = 0; i < parameters.length; i++) {
|
|
functionScope.define(parameters[i], args[i]);
|
|
}
|
|
const oldScope = scope;
|
|
let result: any;
|
|
try {
|
|
scope = functionScope.done();
|
|
result = simplifyInContext(functionSymbol, value, depth + 1);
|
|
} finally {
|
|
scope = oldScope;
|
|
}
|
|
return result;
|
|
}
|
|
} finally {
|
|
calling.delete(functionSymbol);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (depth === 0) {
|
|
// If depth is 0 we are evaluating the top level expression that is describing element
|
|
// decorator. In this case, it is a decorator we don't understand, such as a custom
|
|
// non-angular decorator, and we should just ignore it.
|
|
return {__symbolic: 'ignore'};
|
|
}
|
|
return simplify(
|
|
{__symbolic: 'error', message: 'Function call not supported', context: callContext});
|
|
}
|
|
|
|
function simplify(expression: any): any {
|
|
if (isPrimitive(expression)) {
|
|
return expression;
|
|
}
|
|
if (expression instanceof Array) {
|
|
const result: any[] = [];
|
|
for (const item of (<any>expression)) {
|
|
// Check for a spread expression
|
|
if (item && item.__symbolic === 'spread') {
|
|
const spreadArray = simplify(item.expression);
|
|
if (Array.isArray(spreadArray)) {
|
|
for (const spreadItem of spreadArray) {
|
|
result.push(spreadItem);
|
|
}
|
|
continue;
|
|
}
|
|
}
|
|
const value = simplify(item);
|
|
if (shouldIgnore(value)) {
|
|
continue;
|
|
}
|
|
result.push(value);
|
|
}
|
|
return result;
|
|
}
|
|
if (expression instanceof StaticSymbol) {
|
|
return expression;
|
|
}
|
|
if (expression) {
|
|
if (expression['__symbolic']) {
|
|
let staticSymbol: StaticSymbol;
|
|
switch (expression['__symbolic']) {
|
|
case 'binop':
|
|
let left = simplify(expression['left']);
|
|
if (shouldIgnore(left)) return left;
|
|
let right = simplify(expression['right']);
|
|
if (shouldIgnore(right)) return right;
|
|
switch (expression['operator']) {
|
|
case '&&':
|
|
return left && right;
|
|
case '||':
|
|
return left || right;
|
|
case '|':
|
|
return left | right;
|
|
case '^':
|
|
return left ^ right;
|
|
case '&':
|
|
return left & right;
|
|
case '==':
|
|
return left == right;
|
|
case '!=':
|
|
return left != right;
|
|
case '===':
|
|
return left === right;
|
|
case '!==':
|
|
return left !== right;
|
|
case '<':
|
|
return left < right;
|
|
case '>':
|
|
return left > right;
|
|
case '<=':
|
|
return left <= right;
|
|
case '>=':
|
|
return left >= right;
|
|
case '<<':
|
|
return left << right;
|
|
case '>>':
|
|
return left >> right;
|
|
case '+':
|
|
return left + right;
|
|
case '-':
|
|
return left - right;
|
|
case '*':
|
|
return left * right;
|
|
case '/':
|
|
return left / right;
|
|
case '%':
|
|
return left % right;
|
|
}
|
|
return null;
|
|
case 'if':
|
|
let condition = simplify(expression['condition']);
|
|
return condition ? simplify(expression['thenExpression']) :
|
|
simplify(expression['elseExpression']);
|
|
case 'pre':
|
|
let operand = simplify(expression['operand']);
|
|
if (shouldIgnore(operand)) return operand;
|
|
switch (expression['operator']) {
|
|
case '+':
|
|
return operand;
|
|
case '-':
|
|
return -operand;
|
|
case '!':
|
|
return !operand;
|
|
case '~':
|
|
return ~operand;
|
|
}
|
|
return null;
|
|
case 'index':
|
|
let indexTarget = simplify(expression['expression']);
|
|
let index = simplify(expression['index']);
|
|
if (indexTarget && isPrimitive(index)) return indexTarget[index];
|
|
return null;
|
|
case 'select':
|
|
let selectTarget = simplify(expression['expression']);
|
|
if (selectTarget instanceof StaticSymbol) {
|
|
// Access to a static instance variable
|
|
const declarationValue = resolveReferenceValue(selectTarget);
|
|
if (declarationValue && declarationValue.statics) {
|
|
selectTarget = declarationValue.statics;
|
|
} else {
|
|
const member: string = expression['member'];
|
|
const members = selectTarget.members ?
|
|
(selectTarget.members as string[]).concat(member) :
|
|
[member];
|
|
return _this.host.getStaticSymbol(
|
|
selectTarget.filePath, selectTarget.name, members);
|
|
}
|
|
}
|
|
const member = simplify(expression['member']);
|
|
if (selectTarget && isPrimitive(member)) return simplify(selectTarget[member]);
|
|
return null;
|
|
case 'reference':
|
|
if (!expression.module) {
|
|
const name: string = expression['name'];
|
|
const localValue = scope.resolve(name);
|
|
if (localValue != BindingScope.missing) {
|
|
return localValue;
|
|
}
|
|
}
|
|
staticSymbol = resolveReference(context, expression);
|
|
let result: any = staticSymbol;
|
|
let declarationValue = resolveReferenceValue(result);
|
|
if (declarationValue) {
|
|
if (isOpaqueToken(staticSymbol, declarationValue)) {
|
|
// If the referenced symbol is initalized by a new OpaqueToken we can keep the
|
|
// reference to the symbol.
|
|
return staticSymbol;
|
|
}
|
|
result = simplifyInContext(staticSymbol, declarationValue, depth + 1);
|
|
}
|
|
return result;
|
|
case 'class':
|
|
return context;
|
|
case 'function':
|
|
return context;
|
|
case 'new':
|
|
case 'call':
|
|
// Determine if the function is a built-in conversion
|
|
let target = expression['expression'];
|
|
if (target['module']) {
|
|
staticSymbol = _this.host.findDeclaration(
|
|
target['module'], target['name'], context.filePath);
|
|
} else {
|
|
staticSymbol = _this.host.getStaticSymbol(context.filePath, target['name']);
|
|
}
|
|
let converter = _this.conversionMap.get(staticSymbol);
|
|
if (converter) {
|
|
let args: any[] = expression['arguments'];
|
|
if (!args) {
|
|
args = [];
|
|
}
|
|
return converter(
|
|
context, args.map(arg => simplifyInContext(context, arg, depth + 1)));
|
|
}
|
|
|
|
// Determine if the function is one we can simplify.
|
|
return simplifyCall(expression);
|
|
|
|
case 'error':
|
|
let message = produceErrorMessage(expression);
|
|
if (expression['line']) {
|
|
message =
|
|
`${message} (position ${expression['line']+1}:${expression['character']+1} in the original .ts file)`;
|
|
throw positionalError(
|
|
message, context.filePath, expression['line'], expression['character']);
|
|
}
|
|
throw new Error(message);
|
|
}
|
|
return null;
|
|
}
|
|
return mapStringMap(expression, (value, name) => simplify(value));
|
|
}
|
|
return null;
|
|
}
|
|
|
|
try {
|
|
return simplify(value);
|
|
} catch (e) {
|
|
const message = `${e.message}, resolving symbol ${context.name} in ${context.filePath}`;
|
|
if (e.fileName) {
|
|
throw positionalError(message, e.fileName, e.line, e.column);
|
|
}
|
|
throw new Error(message);
|
|
}
|
|
}
|
|
|
|
const result = simplifyInContext(context, value, 0);
|
|
if (shouldIgnore(result)) {
|
|
return undefined;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* @param module an absolute path to a module file.
|
|
*/
|
|
public getModuleMetadata(module: string): {[key: string]: any} {
|
|
let moduleMetadata = this.metadataCache.get(module);
|
|
if (!moduleMetadata) {
|
|
moduleMetadata = this.host.getMetadataFor(module);
|
|
if (Array.isArray(moduleMetadata)) {
|
|
moduleMetadata = moduleMetadata.find(md => md['version'] === SUPPORTED_SCHEMA_VERSION) ||
|
|
moduleMetadata[0];
|
|
}
|
|
if (!moduleMetadata) {
|
|
moduleMetadata =
|
|
{__symbolic: 'module', version: SUPPORTED_SCHEMA_VERSION, module: module, metadata: {}};
|
|
}
|
|
if (moduleMetadata['version'] != SUPPORTED_SCHEMA_VERSION) {
|
|
throw new Error(
|
|
`Metadata version mismatch for module ${module}, found version ${moduleMetadata['version']}, expected ${SUPPORTED_SCHEMA_VERSION}`);
|
|
}
|
|
this.metadataCache.set(module, moduleMetadata);
|
|
}
|
|
return moduleMetadata;
|
|
}
|
|
|
|
private getTypeMetadata(type: StaticSymbol): {[key: string]: any} {
|
|
const moduleMetadata = this.getModuleMetadata(type.filePath);
|
|
return moduleMetadata['metadata'][type.name] || {__symbolic: 'class'};
|
|
}
|
|
}
|
|
|
|
function expandedMessage(error: any): string {
|
|
switch (error.message) {
|
|
case 'Reference to non-exported class':
|
|
if (error.context && error.context.className) {
|
|
return `Reference to a non-exported class ${error.context.className}. Consider exporting the class`;
|
|
}
|
|
break;
|
|
case 'Variable not initialized':
|
|
return 'Only initialized variables and constants can be referenced because the value of this variable is needed by the template compiler';
|
|
case 'Destructuring not supported':
|
|
return 'Referencing an exported destructured variable or constant is not supported by the template compiler. Consider simplifying this to avoid destructuring';
|
|
case 'Could not resolve type':
|
|
if (error.context && error.context.typeName) {
|
|
return `Could not resolve type ${error.context.typeName}`;
|
|
}
|
|
break;
|
|
case 'Function call not supported':
|
|
let prefix =
|
|
error.context && error.context.name ? `Calling function '${error.context.name}', f` : 'F';
|
|
return prefix +
|
|
'unction calls are not supported. Consider replacing the function or lambda with a reference to an exported function';
|
|
case 'Reference to a local symbol':
|
|
if (error.context && error.context.name) {
|
|
return `Reference to a local (non-exported) symbol '${error.context.name}'. Consider exporting the symbol`;
|
|
}
|
|
}
|
|
return error.message;
|
|
}
|
|
|
|
function produceErrorMessage(error: any): string {
|
|
return `Error encountered resolving symbol values statically. ${expandedMessage(error)}`;
|
|
}
|
|
|
|
function mapStringMap(input: {[key: string]: any}, transform: (value: any, key: string) => any):
|
|
{[key: string]: any} {
|
|
if (!input) return {};
|
|
const result: {[key: string]: any} = {};
|
|
Object.keys(input).forEach((key) => {
|
|
const value = transform(input[key], key);
|
|
if (!shouldIgnore(value)) {
|
|
result[key] = value;
|
|
}
|
|
});
|
|
return result;
|
|
}
|
|
|
|
function isPrimitive(o: any): boolean {
|
|
return o === null || (typeof o !== 'function' && typeof o !== 'object');
|
|
}
|
|
|
|
interface BindingScopeBuilder {
|
|
define(name: string, value: any): BindingScopeBuilder;
|
|
done(): BindingScope;
|
|
}
|
|
|
|
abstract class BindingScope {
|
|
abstract resolve(name: string): any;
|
|
public static missing = {};
|
|
public static empty: BindingScope = {resolve: name => BindingScope.missing};
|
|
|
|
public static build(): BindingScopeBuilder {
|
|
const current = new Map<string, any>();
|
|
return {
|
|
define: function(name, value) {
|
|
current.set(name, value);
|
|
return this;
|
|
},
|
|
done: function() {
|
|
return current.size > 0 ? new PopulatedScope(current) : BindingScope.empty;
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
class PopulatedScope extends BindingScope {
|
|
constructor(private bindings: Map<string, any>) { super(); }
|
|
|
|
resolve(name: string): any {
|
|
return this.bindings.has(name) ? this.bindings.get(name) : BindingScope.missing;
|
|
}
|
|
}
|
|
|
|
function sameSymbol(a: StaticSymbol, b: StaticSymbol): boolean {
|
|
return a === b || (a.name == b.name && a.filePath == b.filePath);
|
|
}
|
|
|
|
function shouldIgnore(value: any): boolean {
|
|
return value && value.__symbolic == 'ignore';
|
|
}
|
|
|
|
function positionalError(message: string, fileName: string, line: number, column: number): Error {
|
|
const result = new Error(message);
|
|
(result as any).fileName = fileName;
|
|
(result as any).line = line;
|
|
(result as any).column = column;
|
|
return result;
|
|
} |