microsoft / TypeScript

TypeScript is a superset of JavaScript that compiles to clean JavaScript output.
https://www.typescriptlang.org
Apache License 2.0
101.28k stars 12.52k forks source link

Array.isArray() : a possible fix ? + discussion on type guards (frozen keyword + better type deduction) #60177

Open denis-migdal opened 1 month ago

denis-migdal commented 1 month ago

πŸ” Search Terms

in:title frozen in:title freeze

I searched for issues on Array.isArray() and found a lot of them, too much to list them all. 3 weeks ago I suggested something that could lead to a possible fix on an existing issue.

βœ… Viability Checklist

⭐ Suggestion

The type guard for Array.isArray() is currently erroneous and the fix potentially quite complex. In retrospect I think the potential fix I suggested previously is more a "workaround" (still, you can get a look at it), and that there are issues on type guards.

I'd like to discuss here what should be the behavior of Array.isArray() on different cases, and to discuss possibilities to simplify the current "workaround" by improving type guards behavior.

Summary :

First, for the sake of simplicity, let's assume :

function isArray(a: unknown): a is unknown[];

type A = typeof a;
Is   <A, unknown[]> // the type of a if isArray() is TRUE
IsNot<A, unknown[]> // the type of a if isArray() is FALSE

In the general case :

Is   <T, U> = T&U;
IsNot<T, U> = Exclude<T,U>;

Of course, type guards need to do more than that, and are currently doing more, but not enough.

Union:

Is<T1|T2, U> = Is<T1, U> | Is<T2, U>;

Currently, Type guards and & seem to behave as expected. Currently, (T1|T1)&U is distributed as (T1&U) | (T2&U) to remove some never then factorized when the type is printed.

Child class :

Is<T extends U, U> = T;

Currently, Type guards behave as expected, but & doesn't (but not an issue).

class A<T> extends Array<T> { /* ... */ }

type T = A<any> & Array<any>; // A<any> expected, got A<any> & any[].

Base type

Is<T, U extends T> = U;

Currently, Type guards behave as expected, but & doesn't (but not an issue).

interface A {
        get length(): number;
    }

type C = A & Array<any>; // expected Array<any>, got Array<any> & A.

Readonly:

There is 2 ways to see readonly :

For TS, it is saw as a partial interface, therefore : readonly T & T = T. Which is quite confusing as, in practice, we mainly use it as a constraint, but this is a design choice, why not.

The issue is that the type of Μ€Object.freeze([]) is also a readonly [], when this is not a partial interface, BUT a constraint. This is an inconsistency in the design, which could be solved with e.g. a frozen keyword : frozen number[] which would set some properties/methods as never instead of simply removing them :

    type A = {
        a: 3,
        b: 4
    }

    type Excl<T, keys> = {
        [K in keyof T]: K extends keys ? never : T[K]
    }

    type B = Excl<A, "b">;
   // or type B = number [] & { push: never };

    type C = B&A;
    let c = f<C>();
    c.b // never

Currently, Array.isArray(readonly T[]), assert T as being any[], which is wrong for 2 reasons :

  1. the generic type information is lost.
  2. the readonly information is lost.

I argue that as Array.isArray(Object.freeze([])) returns true, so we shouldn't remove the readonly keyword. But should it be at the type guard level, or at the Array.isArray() call signature level ?

On one side readonly is only a partial interface, and on the other side, it is often used as a constraint (a frozen keyword would solve this). On another side, readonly is at the type level, when the type guard function is based on the value during execution.

Therefore, without frozen there is 4 solutions :

  1. Add readonly at the Array.isArray() level, and require other devs to do so for their type guards.
  2. Handle readonly at the type guard level, with Is<readonly T, U> = readonly (T&U), which would be ambiguous as readonly isn't a constraint, but a partial interface.
  3. In type guards, if T extends readonly U, makes U implicitly readonly. Is<T extends readonly U, U> = T & readonly U otherwise Is<T,U> = T&U, which might also be confusing.
  4. Assume that type guards offer information on the runtime value, but not on the desired TS type, i.e. : Is<T, U extends T> = T and requires an explicit cast to get a U (which would always be legit), which would be horrible.

And here the good stuff, with generics...

Base type + generics

    interface A {
        push(...args: number[]): void;
        pop(): number|undefined;
    }

    type X = Array<number> extends A ? true : false; // true
    let a = f<A>();

    if(Array.isArray(a) )
        a; // any[] <- should be number[]

We should try to assert the generic types :

// with U<number> extends T;
Is<T, U<unknown>> = Is<Partial<U<number>>, U<unknown>> = U<unknown&number> = U<number>; 

I think a type deduction is technically possible in lot of cases, an would simplify lot of type guards using generics. The issue is to assert when the following step would be legal in a type guard :

Is<Partial<U<number>>, U<unknown>> = U<unknown&number>

Maybe if, and only if, U<unknown&number> extends U<unknown> ?

We could even be more generic :

// with U<number> extends Pick<T, keyof U>;
Is<T, U<unknown>> = Is<Partial<U<number>>, U<unknown>> & T = U<unknown&number> & T = U<number> & T; 

When we can't deduce, I suggest:

This issue also occurs with readonly unknown[], as it can be seen as a base type of unknown[].

πŸ“ƒ Motivating Example

This would lead to more precise type deduction in type guards.

πŸ’» Use Cases

  1. What do you want to use this for?

Deduce type more precise types.

  1. What shortcomings exist with current approaches?

Deduced types are incorrect/not precised.

  1. What workarounds are you using in the meantime?

Complex type guards.

denis-migdal commented 1 month ago

I'll discuss a little the behavior of the "frozen" keyword that could help to solve issues with readonly (I can open a new issue if necessary).

Let assume frozen T = Freeze<T>.

1) the frozen state needs to be contagious : Freeze<T> & U = Freeze<T&U>. It can be implemented :

2) we need to make some call signatures non callable. It can be implemented :

3) We need to mark the call signature so that they will be callable in the general case, but non callable when the type is frozen. It can be implemented :

PoC : Playground Link

denis-migdal commented 1 month ago

Update: it seems TS is already able to assert generics types of base types:

interface A {
    push(...args: number[]): number;
    pop(): number|undefined;
}

type PickMatching<V, T> = { [K in keyof T]: (T&V)[K] };
type Z =  A extends PickMatching<Array<infer T>, A> ? T[] : never; // number[]

interface A2 {
    [key: number]: number;
    push(...args: number[]): number;
    pop(): number|undefined;
}
type Z2 =  A2 extends PickMatching<Array<infer T>, A> ? T[] : never; // number[]

I updated my Array.isArray() workaround to include it.

Then, for type guards, should it be done by TS itself, or by devs in the function signature ?