TypeScript Version: 3.6.0-dev.20190803
Search Terms: assigning keyof generic key
Code
type A = { a: number }
type B = { b: number }
interface Foo {
a: A
b: B
}
declare let target: Foo
declare let source: Foo
declare let key: keyof Foo
target[key] = source[key]
Expected behavior:
I expect it to allow the assignment. Since typeof target and typeof source are the same, I expect typeof target[key] to always be the same as typeof source[key].
Actual behavior:
error TS2322: Type 'A | B' is not assignable to type 'A & B'.
Playground Link: https://www.typescriptlang.org/play/#code/C4TwDgpgBAglC8UDeUCGAuKA7ArgWwCMIAnKAXwChRIoAhBZKAzXQk8iigSy2BIDNUAY2gAxAPbjkFKLLSYYMuczoVKFACYQhAG1TFoOiMCjB9Ac2OYJ4zdr0GoRkwGdxOYiOuS7u-YeMoAGsIEEwQkHF+KBtOM2JLYABtCIBdBjcPERTQ1IogA
Related Issues: #31665 (but it was closed and the comments seem to indicate that this should work 🤔) ping @RyanCavanaugh
See #30769 and #31445. The compiler only sees the types when checking the assignment and doesn't realize that [key] accesses the same property on both sides, and therefore can't prove the assignment is safe. As far as TS is concerned you might be doing target.b = source.a (or vice versa).
This used to work prior to #30769 (TS 3.5), but only as a consequence of being unsound in general:
type A = { a: number }
type B = { b: number }
interface Foo {
a: A
b: B
}
declare let target: Foo
declare let source: Foo
declare let k1: keyof Foo
declare let k2: keyof Foo
// this was incorrectly allowed before TS 3.5
target[k1] = source[k2]
I think the title is slightly misleading here: the key in your example is not generic and #30769 specifically does not affect assignments where the key is generic.
function assignProp<K extends keyof Foo>(target: Foo, source: Foo, key: K) {
target[key] = source[key]; // no error
}
In a sense keyof itself could be seen as generic, insofar as it’s a type which is parameterized on another type. But you’re right that it’s not “a generic key” in the sense that we would normally use the term. :smile:
That said, I think we should revisit since it seems to be coming up reasonably often. Special-casing a[k] = b[k] for identical identifiers k would fix 95% of these and in fact it's hard to imagine a sound assignment where the ks differed
Yes, please consider special case handling. In our project we have to stick on 3.4.5, because TypeScript does not get the following pattern anymore:
interface O {
a?: (object|string);
b?: string;
}
function B (p?: ('a'|'b')): void {
var o: O = {};
if (p) {
o[p] = o[p] || {};
}
}
CC @medns
Approved for the case where
t[x] = s[x]
where x is an "identical reference" (the same function used for CFA). When this happens, relate the assignment using the old "union target" rules about the left-hand side instead of the newer stricter "intersection target" put in place. This would be a special case in checkBinaryExpression. Ping me for clarification if needed.
related to explanation from @RyanCavanaugh
Your code makes sense, but it is a different case from the one I submitted.
You proposed general case and x[g] is not assignable to x[f]. However its a bit tricky and special case should be considered. So, if the indexer is the same type than operation should be allowed.
The type of indexer is not just its declared type, perhaps it also depends on exact instance. This is not currently checked with compiler, and this raise error for correct code , please check the code.
type SimpleType = { a: string, b: number }
let x,y: SimpleType = { a: "", b: 9 }
let fields: (keyof SimpleType)[] = ["a", "b"]
for (const g of fields) {
for (const f of fields) {
x[f] = x[g] //OK (Error -> and should not be allowed; type are not same: each one is string|number )
y[f] = x[f] //Error -> but is OK (types are the same: each one is string or each one is number)
x[f] = x[f] //Error -> but is OK (types are the same)
let h = (Math.random() > 0.5) ? fields[0] : fields[1]
y[h] = x[x] //Should be an error
let j = f
y[j]=x[f] //Would be nice if the complier let this compile to
}
}
Proposal:
1) add additional check while compiling that checks both type and variable in indexer
2) add special keyword (to simplify and limit compiling effort)
for (const f of /*keyword_of_mapped_field*/ fields)
and implement additional indexer instance check
3) add special keyword (to simplify and limit compiling effort)
y[/*keyword_of_mapped_indexer(*/f/*)*/]= y[/*keyword_of_mapped_indexer(*/g/*)*/]
and implement additional indexer instance check
Most helpful comment
31445 is the canonical one for this problem.
That said, I think we should revisit since it seems to be coming up reasonably often. Special-casing
a[k] = b[k]for identical identifierskwould fix 95% of these and in fact it's hard to imagine a sound assignment where theks differed