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

Private field check narrows generic class too far #46668

Open
molisani opened this issue Nov 3, 2021 · 1 comment · May be fixed by #58028
Open

Private field check narrows generic class too far #46668

molisani opened this issue Nov 3, 2021 · 1 comment · May be fixed by #58028
Assignees
Labels
Bug A bug in TypeScript Fix Available A PR has been opened for this issue Rescheduled This issue was previously scheduled to an earlier milestone

Comments

@molisani
Copy link
Contributor

molisani commented Nov 3, 2021

Bug Report

Discovered a potential issue with #44648 in regards to how classes with type parameters are handled. The presence of the brand indicates that the object is an instance of this class, but doesn't say anything about the type arguments. For instanceof the current behavior is to pass any for all type arguments (see #17473 for that issue) but the default here should be that the type matches the constraint for that corresponding type parameter.

🔎 Search Terms

ts4.5-beta, private fields, brand checks, generic, type parameter, narrowing

🕗 Version & Regression Information

TypeScript Version: ts4.5-beta

  • I was unable to test this on prior versions because the feature didn't exist yet.

⏯ Playground Link

Playground link with relevant code

💻 Code

type Constraint = { foo: string } | { bar: number };

class MyClass<T extends Constraint> {
    #brand = true;
    value: T;
    constructor(value: T) {
        this.value = value;
    }

    copyValueFrom(x: object) {
        if (#brand in x) {
            this.value = x.value; 
        }
    }
}

const wrappedFoo = new MyClass({ foo: "foo" });
const wrappedBar = new MyClass({ bar: 123 });
wrappedFoo.copyValueFrom(wrappedBar);
console.log(wrappedFoo.value.foo); // type expects foo, value is bar

🙁 Actual behavior

Assignment in copyValueFrom proceeded without error, results in mismatch between type and value.

🙂 Expected behavior

In copyValueFrom, expected brand to narrow x to MyClass<Constraint> and report error Type 'Constraint' is not assignable to type 'T'. when assigning this.value = x.value.

@andrewbranch andrewbranch added the Bug A bug in TypeScript label Nov 3, 2021
@andrewbranch andrewbranch added this to the TypeScript 4.6.0 milestone Nov 3, 2021
@acutmore
Copy link
Contributor

acutmore commented Nov 4, 2021

Is there anything we can do to also make writing in the other direction safer?

type Constraint = { foo: string } | { bar: number };

class MyClass<T extends Constraint> {
    #brand = true;
    value: T;
    constructor(value: T) {
        this.value = value;
    }

    writeValueTo(x: object) {
        if (#brand in x) {
            // if x.value is `Constraint`, so we can still do an unsafe write
            x.value = this.value; 
        }
    }
}

const wrappedFoo = new MyClass({ foo: "foo" });
const wrappedBar = new MyClass({ bar: 123 });
wrappedFoo.writeValueTo(wrappedBar);
console.log(wrappedBar.value); // type expects bar, value is foo

https://www.typescriptlang.org/play?ts=4.5.0-beta#code/C4TwDgpgBAwg9gOwM7AE4EMCWDhQLxQDeUAZnHAFxQqrYDmUAvlAD5FQBG6qVCArgFsOEVEwDcAWABQ0gCYQAxgBtu0AG7coALyp8EAawRwA7gkkypy9EiRQAsiBgqbAHgAqUCAA9gEBLNt4ZDQsHAA+ImkoaKgAYg4Mf3woND4IcxioDSU0qjcMmIVEGj4FYDhUAAps3Kg3AEpIqUzM4AALTCQAOhroAl6C6MZpKJjjWl8ANXQciDc4Sq8qBydrJBcgmlDgMMbCUZaoLx6ZtOT2zpPZsSgAeluoGaUTWz0kdBJoccxfA6YR5pjCYQaazeYAfT0hhMCEWy0cznWUKMpl27D+mWOvXOHW6AzuDxQmCUSkeJJMEFkf2GFky3ympzmcHBCAgahEcPsCLWLlZ7NQaP2gMO9ygxkQAHJcE8TCk2tAwHBfDhME8QFA3h8vsCADScPi4drqRlQTpQIy4VAQNWPc1skQYmJYk0EC54xk3akAmnSIrBMUYMCQWQAMXIyVZxi5qxslWIZEoUAARAmk0x6uY-SgA+gg5SAEKaAiR6OIuOcbhUACMACYAMzp8zjXPBsNwLr0kGM+aVZt52SF1AZ33FOBKCBdZ50XuB4ODq5pDMElLgaDeSBlWwSrioCV67FmiUJiVQIA

@RyanCavanaugh RyanCavanaugh added the Rescheduled This issue was previously scheduled to an earlier milestone label May 13, 2022
@typescript-bot typescript-bot added the Fix Available A PR has been opened for this issue label Jul 11, 2022
@molisani molisani linked a pull request Apr 1, 2024 that will close this issue
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Bug A bug in TypeScript Fix Available A PR has been opened for this issue Rescheduled This issue was previously scheduled to an earlier milestone
Projects
None yet
6 participants