typescript: validate excess keys on value, returned from function

后端 未结 1 1983
梦谈多话
梦谈多话 2021-01-27 17:15

Let say I\'m doing this:

type Keys = \'a\' | \'b\' | \'c\'
type Rec = { [K in Keys]?: number }
let rec: Rec = { a: 1, d: 4 }

It results in:

<
相关标签:
1条回答
  • 2021-01-27 17:37

    Note that {a: 1, d: 4} is of the Rec type. Object types in TypeScript generally allow excess properties and are not exact. There are good reasons for this having to do with subtyping and assignability. For example:

    class Foo {a: string = ""}
    class Bar extends Foo {b: number = 123}
    console.log(new Bar() instanceof Foo); // true
    

    Note that every Bar is a Foo, which means that you can't say "all Foo objects only have an a property" without preventing class or interface inheritance and extension. And since interface works the same way, and since TypeScript's type system is structural and not nominal, you don't even have to declare a Bar type for it to exist:

    interface Foo2 {a: string};
    // interface Bar2 extends Foo2 {b: number};
    const bar2 = {a: "", b: 123 };
    const foo2: Foo2 = bar2; // okay
    

    So for better or worse we are stuck with a type system whereby extra properties do not break type compatibility.


    Of course, this can be a source of errors. So in the case where you are explicitly assigning a brand new object literal to a place that expects a particular object type, there are excess property checks that behave as if the type were exact. These checks only happen in particular circumstances, as in your first example:

    let rec: Rec = { a: 1, d: 4 }; // excess property warning
    

    But return values from functions are not currently one of these circumstances. The return value's type gets widened before any excess property checks can happen. There is a quite old open GitHub issue, microsoft/TypeScript#241 which suggests that this should be changed so that return values from functions not be widened this way, and there's even an implementation of a potential fix at microsoft/TypeScript#40311 but I'm not sure if and when it will make it into the language. Still, there's a chance you'll see a change here in an upcoming TS version.


    There aren't any perfect ways to suppress excess properties in general. My advice is to just accept that objects may have excess keys and ensure that any code you write would not break if this is the case. You can do things which discourage excess properties, such as these:

    // annotate return type explicitly
    const fn2: Func = (): Rec => ({ a: 1, d: 4 }) // excess property warning
    
    // use a generic type that gets mad about excess properties
    const asFunc = <T extends Rec & Record<Exclude<keyof T, keyof Rec>, never>>(
        cb: () => T
    ): Func => cb;
    const fn3 = asFunc(() => ({ a: 1, d: 4 })); // error! number is not never
    

    But they are more complicated and easily broken, since nothing whatsoever will stop you from doing this no matter how much you try to safeguard your Func type:

    const someBadFunc = () => ({ a: 1, d: 4 });
    const cannotPreventThis: Rec = someBadFunc();
    

    Writing code that anticipates extra properties usually involves holding onto an array of known keys. So don't do this:

    function extraKeysBad(rec: Rec) {
        for (const k in rec) { 
            const v = rec[k as keyof Rec];  // bad assumption that k is keyof Rec 
            console.log(k + ": " + v?.toFixed(2))
        }
    }
    
    const extraKeys = {a: 1, b: 2, d: "four"};
    extraKeysBad(extraKeys); // a: 1.00, b: 2.00, RUNTIME ERROR! v.toFixed not a function
    

    Do this instead:

    function extraKeysOkay(rec: Rec) {
        for (const k of ["a", "b", "c"] as const) {
            const v = rec[k];
            console.log(k + ": " + v?.toFixed(2))
        }
    }
    
    extraKeysOkay(extraKeys); // a: 1.00, b: 2.00, c: undefined
    

    Playground link to code

    0 讨论(0)
提交回复
热议问题