-
Notifications
You must be signed in to change notification settings - Fork 12.8k
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
Why does { foo: T } not satisfy { [key:string]: T }? #6041
Comments
After browsing the issues a bit deeper, this may be the same thing as #5683. |
What is the error you get? It looks to me that you'll get a different type inferred for the values of |
@MrHen Can you post a repro that shows the exact issue? |
@sandersn, @ahejlsberg Here is a Playground link. The error I'm used to seeing is:
Turns out an important difference is whether I explicitly use
I suppose I can use the |
Here is another (simplified) example of what I ran into when using Angular's $http (Playground link): interface HttpHeaders {
[key: string]: string
}
interface HttpService {
<T>(path: string, method: string, headers: HttpHeaders): PromiseLike<T>;
}
let httpService: HttpService;
//this works:
httpService('', '', {
'Content-Type': 'application/x-www-form-urlencoded'
});
//this does not:
const headers = {
'Content-Type': 'application/x-www-form-urlencoded'
};
httpService('', '', headers); It results in the same error:
|
@RyanCavanaugh That's a good explanation for what happens but it's still kind of irritating because of how many libraries take dictionary-esque parameters. From the SO example:
It is really annoying to add explicit index signatures to variables like
But this gets old:
Especially for libraries like async where the types can get weird. |
I don't know what the answer is, but it's clear this is becoming really annoying as definition file authors provide increasingly precise type definitions (I think two years ago, we would have seen someone write |
Proposing the following change: If we are trying to assign Pre-buttal: What about aliasing? Well, we already have this hole, which is basically equivalent: let a = { x: 4, y: 'oops' };
let b: { x: number } = a;
let c: { x: number; y?: number } = b;
c.y.toFixed(); // Runtime error Informally, we can choose to think about the indexer master...RyanCavanaugh:inferStringIndexers#diff-8b210ad15fe8058f8cadce2a5a0c71ccR1 |
@RyanCavanaugh What about cases like this? let x: {}; // x can be any object
x = { a: 10, b: true };
let stringMap: { [x: number]: string } = x; Effectively the |
@ahejlsberg I see that as being entirely analogous to this, which we have always allowed let x: {};
x = { n: 3 };
let y: { n?: string } = x; |
👍 for the proposal. That should address my particular issue. |
I ran into this, again. Let's say I want to say "You can provide me an object, but all properties of that object must be functions of one argument": interface Foo {
[x: string]: (n: any) => void;
} Naturally I implement this with an internal module: namespace MyFoo {
export function fn1(x: number) { }
export function fn2(x: string) { }
export function fn3(x: {}, y: {}) { } // oops
}
// Desired: error about fn3 having an invalid type
// Actual: complaint about the index signature being missing
let x: Foo = MyFoo; So I try to fix it by module merging, which doesn't work // Does nothing useful
interface MyFoo {
[x: string]: (n: any) => void;
} Then I try to fix it by declaring an index signature, which doesn't work either module MyFoo {
// Syntax error
[x: string]: (n: any) => void;
} As far as I can tell, there's no way to add the index signature to the namespace. So in this case the namespace is worse than an object literal, because I could at least contextually type the object literal to get the signature in. How about having a notion of an optional index signature which doesn't require the index signature to be declared, but does enforce that declared properties are assignable to the index signature type? That would have only the same holes as our existing optional property rules. |
That seems good. One context to consider is that libraries like async or lodash like to take "collections" which can be either Objects or Arrays with the expectation that all elements match a particular type. What they really want is a way to say, "all properties resolve to T" and "takes Objects and/or Arrays." E.g., async.parallel and lodash.map.
The typings files for these will often have to jump through all sorts of hoops. Here is
For completeness, here are the helper interfaces:
If I am understanding your suggestion, it would alter the
But it would be nice to make the index type optional, too. The maintainers for the lodash or async typings may care but I don't. All I want is either of these:
So I can do this:
Or this (which is the example that I originally filed):
|
@RyanCavanaugh You made this comment:
Thinking about it some more, I think you're right. I'll experiment a bit with changing our assignability rules such that a type S is assignable to a type T with an index signature if all known properties in S are assignable to T's index signature. |
Correct me If I'm wrong but I think this convenient transformation was only added at the value level, not the type level. I still can't do : interface Bla<T extends { [key: string]: number }> {
}
interface Options {
someKey: 33
} Then later use |
This doesn't work either interface Foo {
foo: string;
}
const foo: {[key: string]: any} = {foo: 'bar'};
const test: Foo = foo; |
Various libraries I use want to see dictionary "loops" that take some number of key/value pairs and operate on them. E.g.:
The typings file for async likes to define helper dictionary types like this:
Periodically, I need to create these key/value pairs conditionally and on the fly but the compiler yells at me if I try to send it in without explicitly typing my temporary object:
block
now technically satisfies the{ [key: string]: T; }
because it is an object with two keys, both strings. TypeScript will still yell at me for it, though, unless I explicit add a similar dictionary type toblock
. I, therefore, find myself putting this dictionary type all over the place just to satisfy what seems to be an arbitrary distinction between what is or is not akey:string
.This comes up enough that I find the type bloat irritating and it also makes it a requirement to only access
block.foo
asblock['foo']
.The text was updated successfully, but these errors were encountered: