Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow dynamic names in types #15473

Merged
merged 65 commits into from
Nov 16, 2017
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
65 commits
Select commit Hold shift + click to select a range
d8cb3c6
Allow some dynamic names in types
rbuckton Apr 29, 2017
0c1eef7
Add support for declaration emit
rbuckton Apr 29, 2017
d572a54
Report errors from duplicate member names
rbuckton May 1, 2017
2714295
Add declaration file output to tests
rbuckton May 2, 2017
3181a8d
Merge branch 'master' into dynamicNames
rbuckton May 2, 2017
b5f1169
Accept baselines
rbuckton May 2, 2017
3b684d4
PR feedback
rbuckton May 2, 2017
64fd857
fix symbol display for computed properties
rbuckton May 3, 2017
d8ae9c0
Early support for unique symbol type
rbuckton May 4, 2017
83b5a75
Add freshness to unique symbol types
rbuckton May 4, 2017
57674dd
Emit dynamic names for object literal types
rbuckton May 5, 2017
fc61863
Merge branch 'dynamicNames' into symbolLiterals
rbuckton May 5, 2017
fe414a2
Ensure we get the correct symbol for nodes, clean up
rbuckton May 6, 2017
93ea56b
Improve union type reduction for symbol()
rbuckton May 6, 2017
2a73d08
Merge branch 'master' into dynamicNames
rbuckton May 10, 2017
625b37e
Merge branch 'master' into dynamicNames
rbuckton May 31, 2017
8c3b73f
Merge branch 'master' into dynamicNames
rbuckton May 31, 2017
c4e9ce5
Merge branch 'master' into dynamicNames
rbuckton Jun 7, 2017
5854e87
comment typo
rbuckton Jun 9, 2017
3f83b55
Added comments for fresh/regular unique symbol types
rbuckton Jun 9, 2017
38ee475
Fix escaping and follow symbols for element access
rbuckton Jun 9, 2017
022e81b
Merge branch 'master' into dynamicNames
rbuckton Jun 9, 2017
d7ef995
Merge branch 'master' into dynamicNames
rbuckton Sep 21, 2017
e81c83c
Merge branch 'master' into dynamicNames
rbuckton Sep 22, 2017
891e71d
Remove freshness, more comprehensive grammar checks and diagnostic me…
rbuckton Sep 22, 2017
7eedf2e
Update baselines
rbuckton Oct 1, 2017
fb3168d
Merge branch 'master' into dynamicNames
rbuckton Oct 2, 2017
1b45a05
Update baselines
rbuckton Oct 2, 2017
6f05e43
Merge branch 'master' into dynamicNames
rbuckton Oct 3, 2017
7ab451b
Merge branch 'master' into dynamicNames
rbuckton Oct 3, 2017
1745e17
Merge branch 'master' into dynamicNames
rbuckton Oct 3, 2017
4395f25
PR Feedback and API baselines
rbuckton Oct 3, 2017
43c151a
Merge branch 'master' into dynamicNames
rbuckton Oct 4, 2017
ee23f93
Switch to 'unique symbol'
rbuckton Oct 4, 2017
51ded0b
Additional tests
rbuckton Oct 5, 2017
fea6a87
General tidying up and comments.
rbuckton Oct 5, 2017
36f90b6
General tidying up and comments.
rbuckton Oct 5, 2017
906a79d
Support dynamic names on static members of class.
rbuckton Oct 5, 2017
180ca23
Additional documentation
rbuckton Oct 5, 2017
7fd38c8
Merge branch 'master' into dynamicNames
rbuckton Oct 5, 2017
55e63a8
Simplify getLateBoundSymbol
rbuckton Oct 20, 2017
3341e07
Refactor widening
rbuckton Oct 21, 2017
0b31860
Revert some minor changes to clean up PR
rbuckton Oct 21, 2017
ccd98af
Simplify property symbol logic in checkObjectLiteral
rbuckton Oct 21, 2017
b5a7b03
Address PR feedback
rbuckton Oct 21, 2017
51929ac
Merge branch 'master' into dynamicNames
rbuckton Oct 21, 2017
3febc80
More repetitive but less complex widening logic for literals/symbols
rbuckton Oct 26, 2017
170e6ec
Ensure correct errors when emitting declarations
rbuckton Oct 26, 2017
44117e1
Reduce noise in PR, minor cleanup
rbuckton Oct 27, 2017
ec90dbc
Unify logic for getMembers/Exports of symbol
rbuckton Oct 27, 2017
26ca98c
Merge branch 'master' into dynamicNames
rbuckton Oct 31, 2017
208dfa6
Merge branch 'master' into dynamicNames
rbuckton Nov 4, 2017
211b2f0
Shave off ~100ms by extracting ExpandingFlags
rbuckton Nov 4, 2017
33e09f9
Merge branch 'master' into dynamicNames
rbuckton Nov 6, 2017
8b717d3
PR Feedback
rbuckton Nov 7, 2017
ee36e6a
Merge branch 'master' into dynamicNames
rbuckton Nov 7, 2017
444e282
Update baselines after merge
rbuckton Nov 7, 2017
d0fb7e4
PR Feedback
rbuckton Nov 7, 2017
b9dbf5d
Simplify literal/unique symbol widening
rbuckton Nov 10, 2017
ae11ae5
Fix getReturnTypeFromBody widening
rbuckton Nov 13, 2017
804c7d3
Merge branch 'master' into dynamicNames
rbuckton Nov 13, 2017
a21a129
Merge branch 'master' into dynamicNames
rbuckton Nov 15, 2017
86b0759
PR feedback
rbuckton Nov 16, 2017
0b24f02
Use correct base primitive type
rbuckton Nov 16, 2017
ccba128
Use correct base primitive type
rbuckton Nov 16, 2017
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
153 changes: 145 additions & 8 deletions src/compiler/checker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -959,7 +959,7 @@ namespace ts {
case SyntaxKind.ClassDeclaration:
case SyntaxKind.ClassExpression:
case SyntaxKind.InterfaceDeclaration:
if (result = getSymbol(getSymbolOfNode(location).members, name, meaning & SymbolFlags.Type)) {
if (result = getSymbol(getMembersOfSymbol(getSymbolOfNode(location)), name, meaning & SymbolFlags.Type)) {
if (!isTypeParameterSymbolDeclaredInContainer(result, location)) {
// ignore type parameters not declared in this container
result = undefined;
Expand Down Expand Up @@ -2895,6 +2895,9 @@ namespace ts {
}

function getNameOfSymbol(symbol: Symbol): string {
if (symbol.flags & SymbolFlags.Dynamic) {
return unescapeIdentifier(symbol.name);
}
if (symbol.declarations && symbol.declarations.length) {
const declaration = symbol.declarations[0];
if (declaration.name) {
Expand Down Expand Up @@ -5133,7 +5136,7 @@ namespace ts {
function resolveDeclaredMembers(type: InterfaceType): InterfaceTypeWithDeclaredMembers {
if (!(<InterfaceTypeWithDeclaredMembers>type).declaredProperties) {
const symbol = type.symbol;
(<InterfaceTypeWithDeclaredMembers>type).declaredProperties = getNamedMembers(symbol.members);
(<InterfaceTypeWithDeclaredMembers>type).declaredProperties = getNamedMembers(getMembersOfSymbol(symbol));
(<InterfaceTypeWithDeclaredMembers>type).declaredCallSignatures = getSignaturesOfSymbol(symbol.members.get("__call"));
(<InterfaceTypeWithDeclaredMembers>type).declaredConstructSignatures = getSignaturesOfSymbol(symbol.members.get("__new"));
(<InterfaceTypeWithDeclaredMembers>type).declaredStringIndexInfo = getIndexInfoOfSymbol(symbol, IndexKind.String);
Expand All @@ -5142,6 +5145,137 @@ namespace ts {
return <InterfaceTypeWithDeclaredMembers>type;
}

function getMembersOfSymbol(symbol: Symbol) {
const links = getSymbolLinks(symbol);
if (!links.resolvedMembers) {
links.resolvedMembers = emptySymbols;
const dynamicMembers = getDynamicMembersOfSymbol(symbol);
if (!dynamicMembers || dynamicMembers.size === 0) {
return links.resolvedMembers = symbol.members || emptySymbols;
}
if (!symbol.members || symbol.members.size === 0) {
return links.resolvedMembers = dynamicMembers;
}
const resolvedMembers = createMap<Symbol>();
mergeSymbolTable(resolvedMembers, symbol.members);
mergeSymbolTable(resolvedMembers, dynamicMembers);
return links.resolvedMembers = resolvedMembers;
}
return links.resolvedMembers;
}

function getDynamicMembersOfSymbol(symbol: Symbol) {
if (symbol.flags & (SymbolFlags.Class | SymbolFlags.Interface | SymbolFlags.TypeLiteral | SymbolFlags.ObjectLiteral)) {
const links = getSymbolLinks(symbol);
if (!links.dynamicMembers) {
const members = createMap<Symbol>();
for (const decl of symbol.declarations) {
resolveDynamicMembersOfSymbol(decl, members);
}
links.dynamicMembers = members;
}
return links.dynamicMembers;
}
}

function resolveDynamicMembersOfSymbol(node: Declaration, symbolTable: SymbolTable) {
switch (node.kind) {
case SyntaxKind.InterfaceDeclaration:
case SyntaxKind.ClassDeclaration:
case SyntaxKind.ClassExpression:
case SyntaxKind.TypeLiteral:
resolveDynamicMembersOfClassOrInterfaceOrTypeLiteralNode(<ClassLikeDeclaration | InterfaceDeclaration | TypeLiteralNode>node, symbolTable);
break;
case SyntaxKind.ObjectLiteralExpression:
resolveDynamicMembersOfObjectLiteralExpression(<ObjectLiteralExpression>node, symbolTable);
break;
}
}

function resolveDynamicMembersOfClassOrInterfaceOrTypeLiteralNode(node: ClassLikeDeclaration | InterfaceDeclaration | TypeLiteralNode, symbolTable: SymbolTable) {
for (const member of node.members) {
if (member.name && isComputedPropertyName(member.name) && isEntityNameExpression(member.name.expression)) {
bindDynamicMember(symbolTable, node.symbol, member);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it would be simpler to have a third parameter symbol and only one resolveDynamicMembersOfNode(members: NodeArray<ClassElement | TypeElement | ObjectLiteralElementLike>, symbols: Symbol[], symbolTable: SymbolTable). Then resolveDynamicMembersOfSymbol could pull off the properties of node in each case.

}
}
}

function resolveDynamicMembersOfObjectLiteralExpression(node: ObjectLiteralExpression, symbolTable: SymbolTable) {
for (const member of node.properties) {
if (member.name && isComputedPropertyName(member.name) && isEntityNameExpression(member.name.expression)) {
bindDynamicMember(symbolTable, node.symbol, member);
}
}
}

function bindDynamicMember(symbolTable: SymbolTable, parent: Symbol, member: ClassElement | TypeElement | ObjectLiteralElement) {
const links = getNodeLinks(member);
if (!links.resolvedSymbol) {
switch (member.kind) {
case SyntaxKind.PropertyDeclaration:
case SyntaxKind.PropertySignature:
return resolveDynamicMember(symbolTable, parent, member,
SymbolFlags.Property | ((<PropertyDeclaration>member).questionToken ? SymbolFlags.Optional : SymbolFlags.None),
SymbolFlags.PropertyExcludes);
case SyntaxKind.PropertyAssignment:
return resolveDynamicMember(symbolTable, parent, member, SymbolFlags.Property, SymbolFlags.PropertyExcludes);
case SyntaxKind.MethodDeclaration:
case SyntaxKind.MethodSignature:
return resolveDynamicMember(symbolTable, parent, member,
SymbolFlags.Method | ((<MethodDeclaration>member).questionToken ? SymbolFlags.Optional : SymbolFlags.None),
isObjectLiteralMethod(member) ? SymbolFlags.PropertyExcludes : SymbolFlags.MethodExcludes);
case SyntaxKind.GetAccessor:
return resolveDynamicMember(symbolTable, parent, member, SymbolFlags.GetAccessor, SymbolFlags.GetAccessorExcludes);
case SyntaxKind.SetAccessor:
return resolveDynamicMember(symbolTable, parent, member, SymbolFlags.SetAccessor, SymbolFlags.SetAccessorExcludes);
}
}
return links.resolvedSymbol;
}

function resolveDynamicMember(symbolTable: SymbolTable, parent: Symbol, member: ClassElement | TypeElement | ObjectLiteralElement, includes: SymbolFlags, excludes: SymbolFlags) {
Debug.assert(isComputedPropertyName(member.name));
const nameType = checkComputedPropertyName(<ComputedPropertyName>member.name);
if (nameType.flags & TypeFlags.StringOrNumberLiteral) {
// TODO(rbuckton): ESSymbolLiteral
const memberName = escapeIdentifier((<LiteralType>nameType).text);
let symbol = symbolTable.get(memberName);
if (!symbol) {
symbolTable.set(memberName, symbol = createSymbol(SymbolFlags.Dynamic, memberName));
}
const staticMember = parent.members && parent.members.get(memberName);
if (symbol.flags & excludes || staticMember) {
const declarations = staticMember ? concatenate(staticMember.declarations, symbol.declarations) : symbol.declarations;
forEach(declarations, declaration => {
error(declaration.name || declaration, Diagnostics.Duplicate_identifier_0, memberName);
});
error(member.name || member, Diagnostics.Duplicate_identifier_0, memberName);
symbol = createSymbol(SymbolFlags.Dynamic, memberName);
}
addDeclarationToSymbol(symbol, member, includes);
symbol.parent = parent;
return symbol;
}
return getNodeLinks(member).resolvedSymbol = member.symbol || unknownSymbol;
}

function addDeclarationToSymbol(symbol: Symbol, member: ClassElement | TypeElement | ObjectLiteralElement, symbolFlags: SymbolFlags) {
symbol.flags |= symbolFlags;
getNodeLinks(member).resolvedSymbol = symbol;
if (!symbol.declarations) {
symbol.declarations = [member];
}
else {
symbol.declarations.push(member);
}
if (symbolFlags & SymbolFlags.Value) {
const valueDeclaration = symbol.valueDeclaration;
if (!valueDeclaration || valueDeclaration.kind !== member.kind) {
symbol.valueDeclaration = member;
}
}
}

function getTypeWithThisArgument(type: Type, thisArgument?: Type): Type {
if (getObjectFlags(type) & ObjectFlags.Reference) {
const target = (<TypeReference>type).target;
Expand All @@ -5165,7 +5299,7 @@ namespace ts {
let numberIndexInfo: IndexInfo;
if (rangeEquals(typeParameters, typeArguments, 0, typeParameters.length)) {
mapper = identityMapper;
members = source.symbol ? source.symbol.members : createSymbolTable(source.declaredProperties);
members = source.symbol ? getMembersOfSymbol(source.symbol) : createSymbolTable(source.declaredProperties);
callSignatures = source.declaredCallSignatures;
constructSignatures = source.declaredConstructSignatures;
stringIndexInfo = source.declaredStringIndexInfo;
Expand Down Expand Up @@ -5425,7 +5559,7 @@ namespace ts {
setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
}
else if (symbol.flags & SymbolFlags.TypeLiteral) {
const members = symbol.members;
const members = getMembersOfSymbol(symbol);
const callSignatures = getSignaturesOfSymbol(members.get("__call"));
const constructSignatures = getSignaturesOfSymbol(members.get("__new"));
const stringIndexInfo = getIndexInfoOfSymbol(symbol, IndexKind.String);
Expand Down Expand Up @@ -7396,7 +7530,7 @@ namespace ts {
if (!links.resolvedType) {
// Deferred resolution of members is handled by resolveObjectTypeMembers
const aliasSymbol = getAliasSymbolForTypeNode(node);
if (node.symbol.members.size === 0 && !aliasSymbol) {
if (getMembersOfSymbol(node.symbol).size === 0 && !aliasSymbol) {
links.resolvedType = emptyTypeLiteralType;
}
else {
Expand Down Expand Up @@ -15623,7 +15757,7 @@ namespace ts {
function getInferredClassType(symbol: Symbol) {
const links = getSymbolLinks(symbol);
if (!links.inferredClassType) {
links.inferredClassType = createAnonymousType(symbol, symbol.members, emptyArray, emptyArray, /*stringIndexType*/ undefined, /*numberIndexType*/ undefined);
links.inferredClassType = createAnonymousType(symbol, getMembersOfSymbol(symbol), emptyArray, emptyArray, /*stringIndexType*/ undefined, /*numberIndexType*/ undefined);
}
return links.inferredClassType;
}
Expand Down Expand Up @@ -21741,7 +21875,7 @@ namespace ts {
// (type parameters of classDeclaration/classExpression and interface are in member property of the symbol.
// Note: that the memberFlags come from previous iteration.
if (!(memberFlags & ModifierFlags.Static)) {
copySymbols(getSymbolOfNode(location).members, meaning & SymbolFlags.Type);
copySymbols(getMembersOfSymbol(getSymbolOfNode(location)), meaning & SymbolFlags.Type);
}
break;
case SyntaxKind.FunctionExpression:
Expand Down Expand Up @@ -23739,7 +23873,10 @@ namespace ts {

function checkGrammarForNonSymbolComputedProperty(node: DeclarationName, message: DiagnosticMessage) {
if (isDynamicName(node)) {
return grammarErrorOnNode(node, message);
if (!isEntityNameExpression((<ComputedPropertyName>node).expression) ||
(checkExpressionCached((<ComputedPropertyName>node).expression).flags & TypeFlags.StringOrNumberLiteral) === 0) {
return grammarErrorOnNode(node, message);
}
}
}

Expand Down
3 changes: 3 additions & 0 deletions src/compiler/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2775,6 +2775,7 @@ namespace ts {
ExportStar = 1 << 25, // Export * declaration
Optional = 1 << 26, // Optional property
Transient = 1 << 27, // Transient symbol (created during type check)
Dynamic = 1 << 28, // Dynamically resolved symbol from computed property

Enum = RegularEnum | ConstEnum,
Variable = FunctionScopedVariable | BlockScopedVariable,
Expand Down Expand Up @@ -2869,6 +2870,8 @@ namespace ts {
isDeclarationWithCollidingName?: boolean; // True if symbol is block scoped redeclaration
bindingElement?: BindingElement; // Binding element associated with property symbol
exportsSomeValue?: boolean; // True if module exports some value (not just types)
dynamicMembers?: SymbolTable; // Dynamic members with literal names resolved during check
resolvedMembers?: SymbolTable;
}

/* @internal */
Expand Down
3 changes: 3 additions & 0 deletions src/harness/unittests/transform.ts
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,9 @@ namespace ts {
transformers: {
before: [replaceUndefinedWithVoid0],
after: [replaceIdentifiersNamedOldNameWithNewName]
},
compilerOptions: {
newLine: ts.NewLineKind.CarriageReturnLineFeed
}
}).outputText;
});
Expand Down
Loading