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

Potentially-undefined value not flagged when destructuring union of empty tuple and array #61424

Open
bthall16 opened this issue Mar 14, 2025 · 4 comments
Labels
Bug A bug in TypeScript Help Wanted You can do this
Milestone

Comments

@bthall16
Copy link

bthall16 commented Mar 14, 2025

πŸ”Ž Search Terms

"tuple destructuring", "tuple narrowing"

πŸ•— Version & Regression Information

This bug appears to occur as far back as TS 3.3 up to the current nightly build, as tested in the Playground.

⏯ Playground Link

https://www.typescriptlang.org/play/?ts=5.8.2#code/C4TwDgpgBAggTnKBeWCCGIA8A7ArgWwCMI4A+KAHygG0BdSmgZ2DgEtsBzWgbgCheAJhADGAGzRxowgPbZmUCXABcqOH14y5wGmnopFfNADpg0gMot2HABQBKblAD0jqAFEAHpGHAIAlSThpOCNYb1w0URVsaSgAoJDAGXJ+TXlCZAUEagAGHl5CE3NLTjsHZzdPER8-WIR40OBwyJrA4KhAUHJ+XigoIA

πŸ’» Code

type Arr = Array<number> | [] | [string];

declare const arr: Arr;

const [a] = arr;
a.toString(); // Expected: error. Actual: no error. ❌

const b = arr[0];
b.toString(); // Expected: error. Actual: error. βœ…

.d.ts from Playground:

type Arr = Array<number> | [] | [string];
declare const arr: Arr;
declare const a: string | number | undefined;
declare const b: string | number | undefined;

πŸ™ Actual behavior

When destructuring a value whose type is an array, an empty tuple, and a non-empty tuple, the destructured value isn't flagged as possibly undefined which caused a runtime error. The value is correctly flagged as possibly undefined when accessing it by index.

πŸ™‚ Expected behavior

When destructuring a value whose type is an array, an empty tuple, and a non-empty tuple, the destructured value should be flagged as potentially being undefined.

Additional information about the issue

This seems related to #55661 except, in this case, the union contains a mix of tuples and an array.

@MartinJohns
Copy link
Contributor

You need to enable noUncheckedIndexedAccess. That's what the flag is there for.

@RyanCavanaugh
Copy link
Member

Still weird that the destructuring behavior is different though.

@RyanCavanaugh RyanCavanaugh added Bug A bug in TypeScript Help Wanted You can do this labels Mar 14, 2025
@RyanCavanaugh RyanCavanaugh added this to the Backlog milestone Mar 14, 2025
@bthall16
Copy link
Author

bthall16 commented Mar 14, 2025

Even with noUncheckedIndexAccess off, the index access in the snippet works correctly: b could be undefined and TS correctly raises an error when I try to call toString. It's the destructuring behavior that seems incorrect since there's no error trying to call toString when there should be.

EDIT: I suppose it's not accurate for me to say the index access works "correctly" when noUncheckedIndexAccess is off (as in the snippet). Rather, it works "intuitively" but I'm not sure what "correct" behavior would look like here since arrays and tuples would individually behave differently.

@Andarist
Copy link
Contributor

An extra bug related to this:

type Arr = Array<number> | [] | [string];

declare const arr: Arr;

export const [a] = arr;

type Mix = Iterable<number> | [] | [string];

declare const mix: Mix;

export const [b] = mix;

Actual .d.ts emit:

export declare const a: string | number | undefined;
export declare const b: string | number;

Expected .d.ts emit:

export declare const a: string | number | undefined;
export declare const b: string | number | undefined;

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Bug A bug in TypeScript Help Wanted You can do this
Projects
None yet
Development

No branches or pull requests

4 participants