I noticed that recently a lot of new APIs are added to lib.d.ts. Of cause this is a good thing, however my code breaks badly because of such update.
Previously I had added a lot of declarations in my code for HTML5 APIs that were missing in lib.d.ts . Now some of them are added to lib.d.ts, and I get quite a few "duplicated identifier" errors. For example, I had added this piece to play with fullscreen:
interface Document {
//...
fullscreenEnabled: boolean;
mozFullScreenEnabled: boolean;
webkitFullscreenEnabled: boolean;
}
Now fullscreenEnabled and webkitFullscreenEnabled are available in lib.d.ts, and my code breaks, and I have to remove them. However, you are still missing mozFullScreenEnabled right? -- well, waiting for the next break.
Web platforms are constantly evolving, it is not practical to expect lib.d.ts always up to date, so developers just have to write such compensating declarations from time to time. Can TS avoid such breaks? I have a few ideas:
lib.d.ts overridable. When tsc find conflictions between lib.d.ts and client codes, it should pickup declarations in client codes. This is because declarations lib.d.ts may contain bugs, developers should be allowed to workaround them. E.g. MutationObserver's constructor was missing a parameter for a long time. Again, optional warnings are welcomed.What do you think?
Interfaces are open-ended... Once you strip out the collisions, you will only be left with the "extensions" to the interface you are making. For example, this is perfectly valid TypeScript:
interface Foo {
bar: boolean;
}
interface Foo {
qat: boolean;
}
var foo: Foo = {
bar: false,
qat: true
};
Interfaces are open-ended -- I known.
The problem is: why should duplicated members in multiple parts of an interface be treated as error? I think duplications are perfectly legal if an interface is developed/maintained by multiple parties independently.
Web APIs is a good example. Different browser vendors implement their own set of APIs. Largely the same, but have many small differences, e.g. vendor-prefixed APIs . Currently, TS's lib.d.ts is based on IE's API, this is obviously limited. Ideally, every browser vendors should provide their own version of lib.d.ts, and developers can include some or all of them to achieve cross-platform-ness. An imaginary client code looks like:
///<reference path='lib.ms.d.ts'/>
///<reference path='lib.webkit.d.ts'/>
///<reference path='lib.blink.d.ts'/>
///<reference path='lib.moz.d.ts'/>
var requestFullscreen = document.requestFullscreen ||
document.mozRequestFullScreendocument || document.webkitRequestFullscreen ||
document.msRequestFullscreen;
//...
But obviously this can't be done with current TS, because those lib.nnn.d.ts files contain large amount of duplicated members of interfaces.
This is can be a giant pain. We should try to figure something out.
Make lib.d.ts overridable. When tsc find conflictions between lib.d.ts and client codes, it should pickup declarations in client codes.
FWIW this is allowed when compiling with --nolib flag.
Feel like recommending that lib.d.ts be removed from core and spinned into an external project maintained by MS + community.
FWIW this is allowed when compiling with --nolib flag.
No, I just want to override the conflicting pieces, not lib.d.ts entirely. Maintaining a whole lib.d.ts myself is not practical (and there are lib.d.ts used by IDEs).
Feel like recommending that lib.d.ts be removed from core and spinned into an external project maintained by MS + community.
Good idea. However it is still nice to allow duplication/overriding of declarations, because there is still no guarantee that individual parties that maintaining those declarations keep in sync with each other.
Another use case for this: if a project wants to target both ES5 and ES6, they have to include polyfill declarations for ES5 emit (eg. Promise) but change the build to remove these for ES6 emit. It would be easier if both es6-promise.d.ts and lib.es6.d.ts could be present to the compiler, so long as they don't conflict.
There should be no harm in multiple declarations of the same interface as long as there are no conflicts.
Another option would be conditional compile, something along these lines:
...
...
This needs to be resolved soon, it is getting really painful to target both es-5 and es-6
This needs to be resolved soon, it is getting really painful to target both es-5 and es-6
What is wrong with just including lib.es6.d.ts in your project? the library is a super set of ES5 one, and if your project targets both, and you have the correct pollyfils, then you should be safe.
I have two different node packages defining Iterable
How about these who will add *.d.ts of my package? I can not require them do add lib.es6.d.ts as well.
What is wrong with ignoring duplicates?
This issue also affects DefinitelyTyped potentially. Currently there are definations for webrtc, web speech, web midi, and firefox/chrome specific apis, etc, which will confilict with future TS releases. It is even harder for web developers to resolve such conflicts. One can't simply drop DefinitelyTyped because lib.d.ts doesn't contains all vendor prefixed APIs.
I think ideally switching ES5/ES6 targets should not involve changing lib.d.ts, and "compatibility note" suggested in #3250 is a better alternative. There are some ES7 APIs to be added to TS soon, do we want yet another lib.es7.d.ts?
Write up proposal for decoupled libraries
Added proposal at https://github.com/Microsoft/TypeScript/issues/4423
@ENikS Proposal:
Allow duplicated members in multiple interface declarations, as long as they are identical. Optional compiling warnings are acceptable, to assist developers to remove redundant codes at convenient time.
This should not generate any errors, example:
shim.d.ts
interface Symbol {
toString(): string;
valueOf(): Object;
[Symbol.toStringTag]: string;
}
es6.d.ts
...
interface Symbol {
/** Returns a string representation of an object. */
toString(): string;
/** Returns the primitive value of the specified object. */
valueOf(): Object;
[Symbol.toStringTag]: string;
}
...
Make lib.d.ts overridable. When tsc find conflictions between lib.d.ts and client codes, it should pickup declarations in client codes. This is because declarations lib.d.ts may contain bugs, developers should be allowed to workaround them. E.g. MutationObserver's constructor was missing a parameter for a long time. Again, optional warnings are welcomed.
Wanted to add clarification for above argument. As @kitsonk explained, allowing duplicate declarations should resolve this issue as well. Consider following:
I'm just getting started with TypeScript, and targeting ES6 because I want to use async/await. When I include the typings for whatwg-fetch, it references es6-promise.d.ts, and I get Duplicate identifier 'Promise' (x4).
Is there a way to fix this without manually updating the typings?
I tried to exclude es6-promise.d.ts using tsconfig, but it's still included because it's referenced by the whatwg-fetch typing (and from within tsd.d.ts).
Should the author of whatwg-fetch remove the reference, or should this be fixed by TypeScript itself? I imagine that this will continue to happen with other typings.
I don't really know what I'm talking about, but in JS, if you define the same function twice, the last one will be invoked. Couldn't TS type definitions behave the same way, i.e. the last loaded definition is used? ... this probably makes no sense, but I'm just trying to find a solution.
+1
I have the same problem as @glen-84. It would be nice to see a fix or recommendation how to handle this.
Commenting to say that I have a similar issue to @donaldpipowitch and @glen-84 regarding Duplicate definitions of Promise when targeting ES6 with the intention to use async/await.
Watching to see if there is any resolution to this in the future.
Big +1 on this. Much painful.
We have the same issue, angular 2 just won't compile to ES6 without duplicated declaration issues, if we use the --nolib option we get missing declarations issues. Angular 2 decided to don't address this problem but rather delegate to typescript for the solution, it's an extreme pain, what can we do other than manually editing source files which is not viable in the long term?
For those with the Promise issue, one work-around is to use Typings instead of TSD. Typings will not install referenced type definitions (see here), and you can manually install the "dependencies" that you need (excluding the unwanted definitions of Promise).
Not ideal, but there aren't many other options at the moment.
+1 here as well.
I have bluebird and mssql which are both grabbing the Promise issue. For now, I have to manually approve Codeship builds since I'm not sure what's compiler noise and what's an actual error.
+1
I'm not sure I fully understand the issue here, but isn't the real problem that the definition files aren't using "ghost modules" hence "polluting the global namespace with many interfaces"?
@ryansmith94 I'm brand new to Typescript, but it seems like the definition file is shimming by using es6-promise even though we have Promise built in now (I'm targeting ES6).
Possibly @lindo-jmm, but I have the following error because of Bluebird which isn't a shim to my knowledge.
typings/bluebird/bluebird.d.ts(19,15): error TS2300: Duplicate identifier 'Promise'.
typings/bluebird/bluebird.d.ts(19,37): error TS2503: Cannot find namespace 'Promise'.
+1
We have a special typing for Webpack's require function and with Angular2 (beta) node.d.ts (recently updated typings) we're now getting TS2403.... errors - the open-ended suggestion by @kitsonk worked out fine but looks a bit hacky and would make IDE confused (too many intellisense suggestions)
+1 (This is especially a pain in Angular2)
suggested workaround for this? https://github.com/DefinitelyTyped/DefinitelyTyped/issues/5015#issuecomment-172297350 maybe?
As one of the possible workarounds it it OK but it does not really solve the big issue...
A massive +1, you import angular2 into a project, and BOOM! 1221 Error Lines of TS2300: Duplicate identifier due to node.d.ts and es6-shim.d.ts
As we start working with more and more modules in TypeScript this is enviable. I actually think it's good practice for each module to have all the typings that it works with, but I would propose Cascading Overwriting to solve the issue of duplicated typings.
Priority from bottom to top, :
tsd.json filetsd.json filetsd.json fileExamples:
es6-shim.d.ts in typescript has priority 0
es6-shim.d.ts in angular2 has priority 1
es6-shim.d.ts in es6-shim module has priority 3
es6-shim.d.ts in project directory has priority 5
@monteirocode I would not like such overwriting, it likely goes against TypeScript principles of predictability and obviousness, thus you have strong typing but don't know where the definitions came from without an extra exploration.
Objectively, I'm for ignorance of identical definitions at the first place (https://github.com/Microsoft/TypeScript/issues/3215#issuecomment-134719471). It just fixes a bunch of existing issues for many projects and doesn't create new problems or breaking changes (does it?). I would be glad to see this implemented ASAP. IMHO, only after that any extended behavior (type unions, overwriting, etc.) may be discussed.
Hi there, I've made a change in Angular 2 to fix the BOOM! of Duplicate identifier. Will be available in beta.4. See https://github.com/angular/angular/issues/5807
I think it's a burden for users to be careful that each typing is only installed once (eg. installing es6-shim.d.ts and es6-promise.d.ts) so I hope there will still be a resolution on this issue.
I have another use case:
There is a third party library that has started creating their own type definitions - unfortunately many of the interface members are just stubbed with type any, which is nearly useless.
Even with the proposal here, I cannot override the provided type definition with the type I know to be correct. The type definition itself can be updated, but in the next version you'll have to start over.
Reference Type directives are designed to solve exactly this problem by providing a canonical lookup location for things which mess with the global scope.
Combined with the library decoupling work, I think we're addressing all the scenarios here as best as is feasible. There will need to be some level of restructuring of code that found itself in this state, but fundamentally it is solving the problem where two people both want to declare something in the global scope.
@RyanCavanaugh Do you mean Library include directives? It is a very useful feature, but I'm not sure how "duplicated identifer" is resolved. Is it still an error to declare an identical member in a global interface in client code?
It is a very useful feature, but I'm not sure how "duplicated identifer" is resolved
It doesn't. But it allows library declaration authors to not pollute the global namespace. Thus decreases the likelihood of global namespace collision :rose:
It solves it because the offending declaration files can be rewritten to be included via library directives, which will only happen once
I want to use some DOM4 APIs and installed dom4 typings (typings install -A dom4). What will happen when some DOM4 APIs are added to lib.d.ts in future? E.g.
interface ParentNode {
children: HTMLCollection;
}
I also note that duplicate method in interface is not an error, but duplicate property is.
E.g.
interface Element {
innerHTML: string; // error: duplicate identifer
getAttribute(name?: string): string; // OK
getAttribute: (name?: string) => string; // error: duplicate identifer
}
This is a strange divergence to me.
facing same issue in beta17
This is really annoying when trying to compile to es6 - and I am not talking about Angular ,it is valid for any TS project (in my case electron application).
If any of your dependencies has types/es6-promise as dependency you get the duplicated Promise error mentioned above.
Now your options are (as far as I can see):
@kirilpopov the other option is to manually exclude whatever the type file that is providing the typing in the tsconfig.json.
Most helpful comment
We have the same issue, angular 2 just won't compile to ES6 without duplicated declaration issues, if we use the --nolib option we get missing declarations issues. Angular 2 decided to don't address this problem but rather delegate to typescript for the solution, it's an extreme pain, what can we do other than manually editing source files which is not viable in the long term?