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

Walk all outer expression kinds before comitting to the return type of a symbol call #61071

Open
wants to merge 5 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
2 changes: 1 addition & 1 deletion src/compiler/checker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -37102,7 +37102,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
// Treat any call to the global 'Symbol' function that is part of a const variable or readonly property
// as a fresh unique symbol literal type.
if (returnType.flags & TypeFlags.ESSymbolLike && isSymbolOrSymbolForCall(node)) {
return getESSymbolLikeTypeForNode(walkUpParenthesizedExpressions(node.parent));
return getESSymbolLikeTypeForNode(walkUpOuterExpressions(node, OuterExpressionKinds.Parentheses | OuterExpressionKinds.Satisfies));
}
if (
node.kind === SyntaxKind.CallExpression && !node.questionDotToken && node.parent.kind === SyntaxKind.ExpressionStatement &&
Expand Down
279 changes: 279 additions & 0 deletions tests/baselines/reference/uniqueSymbols.errors.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,279 @@
uniqueSymbols.ts(272,27): error TS2352: Conversion of type 'symbol' to type 'string' may be a mistake because neither type sufficiently overlaps with the other. If this was intentional, convert the expression to 'unknown' first.


==== uniqueSymbols.ts (1 errors) ====
// declarations with call initializer
const constCall = Symbol();
let letCall = Symbol();
var varCall = Symbol();

// ambient declaration with type
declare const constType: unique symbol;

// declaration with type and call initializer
const constTypeAndCall: unique symbol = Symbol();

// declaration from initializer
const constInitToConstCall = constCall;
const constInitToLetCall = letCall;
const constInitToVarCall = varCall;
const constInitToConstDeclAmbient = constType;
let letInitToConstCall = constCall;
let letInitToLetCall = letCall;
let letInitToVarCall = varCall;
let letInitToConstDeclAmbient = constType;
var varInitToConstCall = constCall;
var varInitToLetCall = letCall;
var varInitToVarCall = varCall;
var varInitToConstDeclAmbient = constType;

// declaration from initializer with type query
const constInitToConstCallWithTypeQuery: typeof constCall = constCall;
const constInitToConstDeclAmbientWithTypeQuery: typeof constType = constType;

// assignment from any
// https://github.com/Microsoft/TypeScript/issues/29108
const fromAny: unique symbol = {} as any;

// function return inference
function funcReturnConstCall() { return constCall; }
function funcReturnLetCall() { return letCall; }
function funcReturnVarCall() { return varCall; }

// function return value with type query
function funcReturnConstCallWithTypeQuery(): typeof constCall { return constCall; }

// generator function yield inference
function* genFuncYieldConstCall() { yield constCall; }
function* genFuncYieldLetCall() { yield letCall; }
function* genFuncYieldVarCall() { yield varCall; }

// generator function yield with return type query
function* genFuncYieldConstCallWithTypeQuery(): IterableIterator<typeof constCall> { yield constCall; }

// async function return inference
async function asyncFuncReturnConstCall() { return constCall; }
async function asyncFuncReturnLetCall() { return letCall; }
async function asyncFuncReturnVarCall() { return varCall; }

// async generator function yield inference
async function* asyncGenFuncYieldConstCall() { yield constCall; }
async function* asyncGenFuncYieldLetCall() { yield letCall; }
async function* asyncGenFuncYieldVarCall() { yield varCall; }

// classes
class C {
static readonly readonlyStaticCall = Symbol();
static readonly readonlyStaticType: unique symbol;
static readonly readonlyStaticTypeAndCall: unique symbol = Symbol();
static readwriteStaticCall = Symbol();

readonly readonlyCall = Symbol();
readwriteCall = Symbol();
}
declare const c: C;

const constInitToCReadonlyStaticCall = C.readonlyStaticCall;
const constInitToCReadonlyStaticType = C.readonlyStaticType;
const constInitToCReadonlyStaticTypeAndCall = C.readonlyStaticTypeAndCall;
const constInitToCReadwriteStaticCall = C.readwriteStaticCall;

const constInitToCReadonlyStaticCallWithTypeQuery: typeof C.readonlyStaticCall = C.readonlyStaticCall;
const constInitToCReadonlyStaticTypeWithTypeQuery: typeof C.readonlyStaticType = C.readonlyStaticType;
const constInitToCReadonlyStaticTypeAndCallWithTypeQuery: typeof C.readonlyStaticTypeAndCall = C.readonlyStaticTypeAndCall;
const constInitToCReadwriteStaticCallWithTypeQuery: typeof C.readwriteStaticCall = C.readwriteStaticCall;

const constInitToCReadonlyCall = c.readonlyCall;
const constInitToCReadwriteCall = c.readwriteCall;
const constInitToCReadonlyCallWithTypeQuery: typeof c.readonlyCall = c.readonlyCall;
const constInitToCReadwriteCallWithTypeQuery: typeof c.readwriteCall = c.readwriteCall;
const constInitToCReadonlyCallWithIndexedAccess: C["readonlyCall"] = c.readonlyCall;
const constInitToCReadwriteCallWithIndexedAccess: C["readwriteCall"] = c.readwriteCall;

// interfaces
interface I {
readonly readonlyType: unique symbol;
}
declare const i: I;

const constInitToIReadonlyType = i.readonlyType;
const constInitToIReadonlyTypeWithTypeQuery: typeof i.readonlyType = i.readonlyType;
const constInitToIReadonlyTypeWithIndexedAccess: I["readonlyType"] = i.readonlyType;

// type literals
type L = {
readonly readonlyType: unique symbol;
nested: {
readonly readonlyNestedType: unique symbol;
}
};
declare const l: L;

const constInitToLReadonlyType = l.readonlyType;
const constInitToLReadonlyNestedType = l.nested.readonlyNestedType;
const constInitToLReadonlyTypeWithTypeQuery: typeof l.readonlyType = l.readonlyType;
const constInitToLReadonlyNestedTypeWithTypeQuery: typeof l.nested.readonlyNestedType = l.nested.readonlyNestedType;
const constInitToLReadonlyTypeWithIndexedAccess: L["readonlyType"] = l.readonlyType;
const constInitToLReadonlyNestedTypeWithIndexedAccess: L["nested"]["readonlyNestedType"] = l.nested.readonlyNestedType;

// type argument inference
const promiseForConstCall = Promise.resolve(constCall);
const arrayOfConstCall = [constCall];

// unique symbol widening in expressions
declare const s: unique symbol;
declare namespace N { const s: unique symbol; }
declare const o: { [s]: "a", [N.s]: "b" };
declare function f<T>(x: T): T;
declare function g(x: typeof s): void;
declare function g(x: typeof N.s): void;

// widening positions

// argument inference
f(s);
f(N.s);
f(N["s"]);

// array literal elements
[s];
[N.s];
[N["s"]];

// property assignments/methods
const o2 = {
a: s,
b: N.s,
c: N["s"],

method1() { return s; },
async method2() { return s; },
async * method3() { yield s; },
* method4() { yield s; },
method5(p = s) { return p; },
};

// property initializers
class C0 {
static readonly a = s;
static readonly b = N.s;
static readonly c = N["s"];

static d = s;
static e = N.s;
static f = N["s"];

readonly a = s;
readonly b = N.s;
readonly c = N["s"];

d = s;
e = N.s;
f = N["s"];

method1() { return s; }
async method2() { return s; }
async * method3() { yield s; }
* method4() { yield s; }
method5(p = s) { return p; }
}

// non-widening positions

// element access
o[s];
o[N.s];
o[N["s"]];

// arguments (no-inference)
f<typeof s>(s);
f<typeof N.s>(N.s);
f<typeof N.s>(N["s"]);
g(s);
g(N.s);
g(N["s"]);

// falsy expressions
s || "";
N.s || "";
N["s"] || "";

// conditionals
Math.random() * 2 ? s : "a";
Math.random() * 2 ? N.s : "a";
Math.random() * 2 ? N["s"] : "a";

// computed property names
({
[s]: "a",
[N.s]: "b",
});

class C1 {
static [s]: "a";
static [N.s]: "b";

[s]: "a";
[N.s]: "b";
}

// contextual types

interface Context {
method1(): typeof s;
method2(): Promise<typeof s>;
method3(): AsyncIterableIterator<typeof s>;
method4(): IterableIterator<typeof s>;
method5(p?: typeof s): typeof s;
}

const o3: Context = {
method1() {
return s; // return type should not widen due to contextual type
},
async method2() {
return s; // return type should not widen due to contextual type
},
async * method3() {
yield s; // yield type should not widen due to contextual type
},
* method4() {
yield s; // yield type should not widen due to contextual type
},
method5(p = s) { // parameter should not widen due to contextual type
return p;
},
};

// allowed when not emitting declarations

const o4 = {
method1(p: typeof s): typeof s {
return p;
},
method2(p: I["readonlyType"]): I["readonlyType"] {
return p;
}
};

const ce0 = class {
method1(p: typeof s): typeof s {
return p;
}
method2(p: I["readonlyType"]): I["readonlyType"] {
return p;
}
};

function funcInferredReturnType(obj: { method(p: typeof s): void }) {
return obj;
}

// https://github.com/microsoft/TypeScript/issues/61070
const bar = Symbol('bar') satisfies symbol;
let bar2 = Symbol('bar2') satisfies symbol;

const testErrorMessage1 = Symbol() as string;
~~~~~~~~~~~~~~~~~~
!!! error TS2352: Conversion of type 'symbol' to type 'string' may be a mistake because neither type sufficiently overlaps with the other. If this was intentional, convert the expression to 'unknown' first.

10 changes: 10 additions & 0 deletions tests/baselines/reference/uniqueSymbols.js
Original file line number Diff line number Diff line change
Expand Up @@ -267,6 +267,12 @@ const ce0 = class {
function funcInferredReturnType(obj: { method(p: typeof s): void }) {
return obj;
}

// https://github.com/microsoft/TypeScript/issues/61070
const bar = Symbol('bar') satisfies symbol;
let bar2 = Symbol('bar2') satisfies symbol;

const testErrorMessage1 = Symbol() as string;


//// [uniqueSymbols.js]
Expand Down Expand Up @@ -458,3 +464,7 @@ const ce0 = class {
function funcInferredReturnType(obj) {
return obj;
}
// https://github.com/microsoft/TypeScript/issues/61070
const bar = Symbol('bar');
let bar2 = Symbol('bar2');
const testErrorMessage1 = Symbol();
13 changes: 13 additions & 0 deletions tests/baselines/reference/uniqueSymbols.symbols
Original file line number Diff line number Diff line change
Expand Up @@ -863,3 +863,16 @@ function funcInferredReturnType(obj: { method(p: typeof s): void }) {
>obj : Symbol(obj, Decl(uniqueSymbols.ts, 263, 32))
}

// https://github.com/microsoft/TypeScript/issues/61070
const bar = Symbol('bar') satisfies symbol;
>bar : Symbol(bar, Decl(uniqueSymbols.ts, 268, 5))
>Symbol : Symbol(Symbol, Decl(lib.es5.d.ts, --, --), Decl(lib.es2015.symbol.d.ts, --, --), Decl(lib.es2015.symbol.wellknown.d.ts, --, --), Decl(lib.es2019.symbol.d.ts, --, --))

let bar2 = Symbol('bar2') satisfies symbol;
>bar2 : Symbol(bar2, Decl(uniqueSymbols.ts, 269, 3))
>Symbol : Symbol(Symbol, Decl(lib.es5.d.ts, --, --), Decl(lib.es2015.symbol.d.ts, --, --), Decl(lib.es2015.symbol.wellknown.d.ts, --, --), Decl(lib.es2019.symbol.d.ts, --, --))

const testErrorMessage1 = Symbol() as string;
>testErrorMessage1 : Symbol(testErrorMessage1, Decl(uniqueSymbols.ts, 271, 5))
>Symbol : Symbol(Symbol, Decl(lib.es5.d.ts, --, --), Decl(lib.es2015.symbol.d.ts, --, --), Decl(lib.es2015.symbol.wellknown.d.ts, --, --), Decl(lib.es2019.symbol.d.ts, --, --))

Loading