-
Notifications
You must be signed in to change notification settings - Fork 12.6k
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
String literal types #5185
String literal types #5185
Changes from all commits
311a0cf
911e907
f04cc39
191be4f
84786d8
dc0e368
8891fba
82545ce
ed927d8
a3e7ccb
20c2c4e
87f2957
f721971
7b4e94d
d8d72aa
315b06d
4b736da
fd5dec4
f7a6ac7
a440f06
d2e2a55
6e3343c
74ac57d
61ece76
84b64c4
3788254
ebc47d5
725bda8
ec0d49a
1dbd8d1
307d73e
049d02f
6618fd2
bf4880a
5e23143
8dbfe1c
a1fcfaf
bb232f7
f939ff2
d234b8d
c011ed4
38090c6
d294524
ea4e21d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -81,7 +81,7 @@ namespace ts { | |
symbolToString, | ||
getAugmentedPropertiesOfType, | ||
getRootSymbols, | ||
getContextualType, | ||
getContextualType: getApparentTypeOfContextualType, | ||
getFullyQualifiedName, | ||
getResolvedSignature, | ||
getConstantValue, | ||
|
@@ -1653,7 +1653,7 @@ namespace ts { | |
writeAnonymousType(<ObjectType>type, flags); | ||
} | ||
else if (type.flags & TypeFlags.StringLiteral) { | ||
writer.writeStringLiteral((<StringLiteralType>type).text); | ||
writer.writeStringLiteral(`"${escapeString((<StringLiteralType>type).text)}"`); | ||
} | ||
else { | ||
// Should never get here | ||
|
@@ -4401,12 +4401,13 @@ namespace ts { | |
} | ||
|
||
function getStringLiteralType(node: StringLiteral): StringLiteralType { | ||
if (hasProperty(stringLiteralTypes, node.text)) { | ||
return stringLiteralTypes[node.text]; | ||
const text = node.text; | ||
if (hasProperty(stringLiteralTypes, text)) { | ||
return stringLiteralTypes[text]; | ||
} | ||
|
||
const type = stringLiteralTypes[node.text] = <StringLiteralType>createType(TypeFlags.StringLiteral); | ||
type.text = getTextOfNode(node); | ||
const type = stringLiteralTypes[text] = <StringLiteralType>createType(TypeFlags.StringLiteral); | ||
type.text = text; | ||
return type; | ||
} | ||
|
||
|
@@ -5737,6 +5738,10 @@ namespace ts { | |
return !!getPropertyOfType(type, "0"); | ||
} | ||
|
||
function isStringLiteralType(type: Type) { | ||
return type.flags & TypeFlags.StringLiteral; | ||
} | ||
|
||
/** | ||
* Check if a Type was written as a tuple type literal. | ||
* Prefer using isTupleLikeType() unless the use of `elementTypes` is required. | ||
|
@@ -6961,7 +6966,7 @@ namespace ts { | |
else if (operator === SyntaxKind.BarBarToken) { | ||
// When an || expression has a contextual type, the operands are contextually typed by that type. When an || | ||
// expression has no contextual type, the right operand is contextually typed by the type of the left operand. | ||
let type = getContextualType(binaryExpression); | ||
let type = getApparentTypeOfContextualType(binaryExpression); | ||
if (!type && node === binaryExpression.right) { | ||
type = checkExpression(binaryExpression.left); | ||
} | ||
|
@@ -7008,6 +7013,10 @@ namespace ts { | |
return applyToContextualType(type, t => getIndexTypeOfStructuredType(t, kind)); | ||
} | ||
|
||
function contextualTypeIsStringLiteralType(type: Type): boolean { | ||
return !!(type.flags & TypeFlags.Union ? forEach((<UnionType>type).types, isStringLiteralType) : isStringLiteralType(type)); | ||
} | ||
|
||
// Return true if the given contextual type is a tuple-like type | ||
function contextualTypeIsTupleLikeType(type: Type): boolean { | ||
return !!(type.flags & TypeFlags.Union ? forEach((<UnionType>type).types, isTupleLikeType) : isTupleLikeType(type)); | ||
|
@@ -7033,7 +7042,7 @@ namespace ts { | |
|
||
function getContextualTypeForObjectLiteralElement(element: ObjectLiteralElement) { | ||
const objectLiteral = <ObjectLiteralExpression>element.parent; | ||
const type = getContextualType(objectLiteral); | ||
const type = getApparentTypeOfContextualType(objectLiteral); | ||
if (type) { | ||
if (!hasDynamicName(element)) { | ||
// For a (non-symbol) computed property, there is no reason to look up the name | ||
|
@@ -7059,7 +7068,7 @@ namespace ts { | |
// type of T. | ||
function getContextualTypeForElementExpression(node: Expression): Type { | ||
const arrayLiteral = <ArrayLiteralExpression>node.parent; | ||
const type = getContextualType(arrayLiteral); | ||
const type = getApparentTypeOfContextualType(arrayLiteral); | ||
if (type) { | ||
const index = indexOf(arrayLiteral.elements, node); | ||
return getTypeOfPropertyOfContextualType(type, "" + index) | ||
|
@@ -7072,7 +7081,7 @@ namespace ts { | |
// In a contextually typed conditional expression, the true/false expressions are contextually typed by the same type. | ||
function getContextualTypeForConditionalOperand(node: Expression): Type { | ||
const conditional = <ConditionalExpression>node.parent; | ||
return node === conditional.whenTrue || node === conditional.whenFalse ? getContextualType(conditional) : undefined; | ||
return node === conditional.whenTrue || node === conditional.whenFalse ? getApparentTypeOfContextualType(conditional) : undefined; | ||
} | ||
|
||
function getContextualTypeForJsxExpression(expr: JsxExpression | JsxSpreadAttribute): Type { | ||
|
@@ -7097,12 +7106,22 @@ namespace ts { | |
|
||
// Return the contextual type for a given expression node. During overload resolution, a contextual type may temporarily | ||
// be "pushed" onto a node using the contextualType property. | ||
function getContextualType(node: Expression): Type { | ||
const type = getContextualTypeWorker(node); | ||
function getApparentTypeOfContextualType(node: Expression): Type { | ||
const type = getContextualType(node); | ||
return type && getApparentType(type); | ||
} | ||
|
||
function getContextualTypeWorker(node: Expression): Type { | ||
/** | ||
* Woah! Do you really want to use this function? | ||
* | ||
* Unless you're trying to get the *non-apparent* type for a value-literal type, | ||
* you probably meant to use 'getApparentTypeOfContextualType'. | ||
* Otherwise this is slightly less useful. | ||
* | ||
* @param node the expression whose contextual type will be returned. | ||
* @returns the contextual type of an expression. | ||
*/ | ||
function getContextualType(node: Expression): Type { | ||
if (isInsideWithStatementBody(node)) { | ||
// We cannot answer semantic questions within a with block, do not proceed any further | ||
return undefined; | ||
|
@@ -7141,7 +7160,7 @@ namespace ts { | |
Debug.assert(parent.parent.kind === SyntaxKind.TemplateExpression); | ||
return getContextualTypeForSubstitutionExpression(<TemplateExpression>parent.parent, node); | ||
case SyntaxKind.ParenthesizedExpression: | ||
return getContextualType(<ParenthesizedExpression>parent); | ||
return getApparentTypeOfContextualType(<ParenthesizedExpression>parent); | ||
case SyntaxKind.JsxExpression: | ||
case SyntaxKind.JsxSpreadAttribute: | ||
return getContextualTypeForJsxExpression(<JsxExpression>parent); | ||
|
@@ -7181,7 +7200,7 @@ namespace ts { | |
Debug.assert(node.kind !== SyntaxKind.MethodDeclaration || isObjectLiteralMethod(node)); | ||
const type = isObjectLiteralMethod(node) | ||
? getContextualTypeForObjectLiteralMethod(node) | ||
: getContextualType(node); | ||
: getApparentTypeOfContextualType(node); | ||
if (!type) { | ||
return undefined; | ||
} | ||
|
@@ -7311,7 +7330,7 @@ namespace ts { | |
type.pattern = node; | ||
return type; | ||
} | ||
const contextualType = getContextualType(node); | ||
const contextualType = getApparentTypeOfContextualType(node); | ||
if (contextualType && contextualTypeIsTupleLikeType(contextualType)) { | ||
const pattern = contextualType.pattern; | ||
// If array literal is contextually typed by a binding pattern or an assignment pattern, pad the resulting | ||
|
@@ -7403,7 +7422,7 @@ namespace ts { | |
|
||
const propertiesTable: SymbolTable = {}; | ||
const propertiesArray: Symbol[] = []; | ||
const contextualType = getContextualType(node); | ||
const contextualType = getApparentTypeOfContextualType(node); | ||
const contextualTypeHasPattern = contextualType && contextualType.pattern && | ||
(contextualType.pattern.kind === SyntaxKind.ObjectBindingPattern || contextualType.pattern.kind === SyntaxKind.ObjectLiteralExpression); | ||
let typeFlags: TypeFlags = 0; | ||
|
@@ -9404,7 +9423,12 @@ namespace ts { | |
const targetType = getTypeFromTypeNode(node.type); | ||
if (produceDiagnostics && targetType !== unknownType) { | ||
const widenedType = getWidenedType(exprType); | ||
if (!(isTypeAssignableTo(targetType, widenedType))) { | ||
|
||
// Permit 'number[] | "foo"' to be asserted to 'string'. | ||
const bothAreStringLike = | ||
someConstituentTypeHasKind(targetType, TypeFlags.StringLike) && | ||
someConstituentTypeHasKind(widenedType, TypeFlags.StringLike); | ||
if (!bothAreStringLike && !(isTypeAssignableTo(targetType, widenedType))) { | ||
checkTypeAssignableTo(exprType, targetType, node, Diagnostics.Neither_type_0_nor_type_1_is_assignable_to_the_other); | ||
} | ||
} | ||
|
@@ -10234,6 +10258,10 @@ namespace ts { | |
case SyntaxKind.ExclamationEqualsToken: | ||
case SyntaxKind.EqualsEqualsEqualsToken: | ||
case SyntaxKind.ExclamationEqualsEqualsToken: | ||
// Permit 'number[] | "foo"' to be asserted to 'string'. | ||
if (someConstituentTypeHasKind(leftType, TypeFlags.StringLike) && someConstituentTypeHasKind(rightType, TypeFlags.StringLike)) { | ||
return booleanType; | ||
} | ||
if (!isTypeAssignableTo(leftType, rightType) && !isTypeAssignableTo(rightType, leftType)) { | ||
reportOperatorError(); | ||
} | ||
|
@@ -10372,6 +10400,15 @@ namespace ts { | |
return getUnionType([type1, type2]); | ||
} | ||
|
||
function checkStringLiteralExpression(node: StringLiteral): Type { | ||
const contextualType = getContextualType(node); | ||
if (contextualType && contextualTypeIsStringLiteralType(contextualType)) { | ||
return getStringLiteralType(node); | ||
} | ||
|
||
return stringType; | ||
} | ||
|
||
function checkTemplateExpression(node: TemplateExpression): Type { | ||
// We just want to check each expressions, but we are unconcerned with | ||
// the type of each expression, as any value may be coerced into a string. | ||
|
@@ -10431,7 +10468,7 @@ namespace ts { | |
if (isInferentialContext(contextualMapper)) { | ||
const signature = getSingleCallSignature(type); | ||
if (signature && signature.typeParameters) { | ||
const contextualType = getContextualType(<Expression>node); | ||
const contextualType = getApparentTypeOfContextualType(<Expression>node); | ||
if (contextualType) { | ||
const contextualSignature = getSingleCallSignature(contextualType); | ||
if (contextualSignature && !contextualSignature.typeParameters) { | ||
|
@@ -10502,6 +10539,7 @@ namespace ts { | |
case SyntaxKind.TemplateExpression: | ||
return checkTemplateExpression(<TemplateExpression>node); | ||
case SyntaxKind.StringLiteral: | ||
return checkStringLiteralExpression(<StringLiteral>node); | ||
case SyntaxKind.NoSubstitutionTemplateLiteral: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Not that I want to sound negative but for every string literal, this gets called. It's a huge slowdown. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What kind of numbers are you seeing (and on what codebase)? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
I think the VM already optimize for this? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's useless to speculate about what's fast and what's not. Measure it. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Meh. Measure it in 1 version of Chrome, it changes in the next. You measure it, I've said what I had to say. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @jbondc 👍 for the benchmark test. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't think the original issue was so terrible. You only ran into this case when you were contextually typed by a type with a string literal constituent to begin with. In practice, this is not frequently encountered. However, @jbondc, I think you'll like the the current implementation a lot better. We now only create a string literal type if the constituent types have a string literal type, not just if their content is equal. Furthermore, the types are cached in a map of strings to string literal types. Whenever testing the assignability of two string literal types, reference equality kicks in (which is fast). Additionally, error messages are slightly better because you have a specific literal type to report (i.e. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @jbondc at worst that's probably much faster than trying to resolve the 100 different overloads. Though, I'm going to try to avoid speculating. I wasn't seeing a real perf hit in our benchmarks even with the initial change. |
||
return stringType; | ||
case SyntaxKind.RegularExpressionLiteral: | ||
|
@@ -12696,6 +12734,7 @@ namespace ts { | |
let hasDuplicateDefaultClause = false; | ||
|
||
const expressionType = checkExpression(node.expression); | ||
const expressionTypeIsStringLike = someConstituentTypeHasKind(expressionType, TypeFlags.StringLike); | ||
forEach(node.caseBlock.clauses, clause => { | ||
// Grammar check for duplicate default clauses, skip if we already report duplicate default clause | ||
if (clause.kind === SyntaxKind.DefaultClause && !hasDuplicateDefaultClause) { | ||
|
@@ -12716,6 +12755,12 @@ namespace ts { | |
// TypeScript 1.0 spec (April 2014):5.9 | ||
// In a 'switch' statement, each 'case' expression must be of a type that is assignable to or from the type of the 'switch' expression. | ||
const caseType = checkExpression(caseClause.expression); | ||
|
||
// Permit 'number[] | "foo"' to be asserted to 'string'. | ||
if (expressionTypeIsStringLike && someConstituentTypeHasKind(caseType, TypeFlags.StringLike)) { | ||
return; | ||
} | ||
|
||
if (!isTypeAssignableTo(expressionType, caseType)) { | ||
// check 'expressionType isAssignableTo caseType' failed, try the reversed check and report errors if it fails | ||
checkTypeAssignableTo(caseType, expressionType, caseClause.expression, /*headMessage*/ undefined); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1979,9 +1979,7 @@ namespace ts { | |
|
||
function parseParameterType(): TypeNode { | ||
if (parseOptional(SyntaxKind.ColonToken)) { | ||
return token === SyntaxKind.StringLiteral | ||
? <StringLiteral>parseLiteralNode(/*internName*/ true) | ||
: parseType(); | ||
return parseType(); | ||
} | ||
|
||
return undefined; | ||
|
@@ -2369,6 +2367,8 @@ namespace ts { | |
// If these are followed by a dot, then parse these out as a dotted type reference instead. | ||
const node = tryParse(parseKeywordAndNoDot); | ||
return node || parseTypeReferenceOrTypePredicate(); | ||
case SyntaxKind.StringLiteral: | ||
return <StringLiteral>parseLiteralNode(/*internName*/ true); | ||
case SyntaxKind.VoidKeyword: | ||
case SyntaxKind.ThisKeyword: | ||
return parseTokenNode<TypeNode>(); | ||
|
@@ -2399,6 +2399,7 @@ namespace ts { | |
case SyntaxKind.OpenBracketToken: | ||
case SyntaxKind.LessThanToken: | ||
case SyntaxKind.NewKeyword: | ||
case SyntaxKind.StringLiteral: | ||
return true; | ||
case SyntaxKind.OpenParenToken: | ||
// Only consider '(' the start of a type if followed by ')', '...', an identifier, a modifier, | ||
|
@@ -5616,6 +5617,7 @@ namespace ts { | |
return parseTokenNode<JSDocType>(); | ||
} | ||
|
||
// TODO (drosen): Parse string literal types in JSDoc as well. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Not sure we'd need to do this - strings types aren't directly valid jsdoc types. They'd have to be wrapped up inside an enum, 'a la: /**
* @enum
*/
var MyEnum = {
Member: "Member"
}
/**
* @param {MyEnum} param
*/
function foo(param) {
} So literal string types should never appear in JSDoc comments, since nothing really supports them. |
||
return parseJSDocTypeReference(); | ||
} | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,33 +1,33 @@ | ||
=== tests/cases/compiler/callSignatureFunctionOverload.ts === | ||
var foo: { | ||
>foo : { (name: string): string; (name: 'order'): string; (name: 'content'): string; (name: 'done'): string; } | ||
>foo : { (name: string): string; (name: "order"): string; (name: "content"): string; (name: "done"): string; } | ||
|
||
(name: string): string; | ||
>name : string | ||
|
||
(name: 'order'): string; | ||
>name : 'order' | ||
>name : "order" | ||
|
||
(name: 'content'): string; | ||
>name : 'content' | ||
>name : "content" | ||
|
||
(name: 'done'): string; | ||
>name : 'done' | ||
>name : "done" | ||
} | ||
|
||
var foo2: { | ||
>foo2 : { (name: string): string; (name: 'order'): string; (name: 'order'): string; (name: 'done'): string; } | ||
>foo2 : { (name: string): string; (name: "order"): string; (name: "order"): string; (name: "done"): string; } | ||
|
||
(name: string): string; | ||
>name : string | ||
|
||
(name: 'order'): string; | ||
>name : 'order' | ||
>name : "order" | ||
|
||
(name: 'order'): string; | ||
>name : 'order' | ||
>name : "order" | ||
|
||
(name: 'done'): string; | ||
>name : 'done' | ||
>name : "done" | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah... that was why I named it Worker...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, that makes sense, though it feels strange to get the actual contextual type when its name is worker. If you think it'd be better to rename it back to worker, I can do that.
I also suspect that we could shave off some time by using this function (instead of
getApparentTypeOfContextualType
) for array literals as well. I'll probably do it as part of another change.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The idea (however it is achieved) is to make the apparent type the default behavior when you don't specify anything. Getting the contextual type without its apparent type is quite an odd thing to do, so I prefer that to be the marked form. You can also call them getContextualType and getContextualTypeWithoutApparentType.