-
Notifications
You must be signed in to change notification settings - Fork 261
Unable to create definition for es6 class exported as default (facebook/dataloader) #670
Comments
You should do Since it is doing import DataLoader form 'dataloader'
// js
const DataLoader = require('dataloader').default |
@unional thanks for the help! But I am still not able to make it work. So far, I have a file with: /*
dataloader
*/
// ... types omitted
export default class DataLoader<K, V> {
constructor(batchLoadFn: BatchLoadFn<K, V>, options?: Options<K, V>);
load(key: K): Promise<V>;
loadMany(keys: K[]): Promise<V[]>;
// ... methods omitted
} If I install it with 'typings install dataloader.d.ts --save', then typings wraps it with a "declare module 'dataloader' {..." what helps tsc find the module when compiling "import DataLoader from 'dataloader';" but in execution fails with "TypeError: dataloader_1.default is not a constructor". What does not make sense to me, as the module seems to export the DataLoader class as default: https://npmcdn.com/dataloader@1.2.0 If I just add the bare file to my index.d.ts to avoid having the "declare module 'dataloader' {...": /// <reference path="globals/ambient/index.d.ts" />
/// <reference path="globals/mocha/index.d.ts" />
/// <reference path="globals/node/index.d.ts" />
/// <reference path="globals/request/index.d.ts" />
/// <reference path="globals/should/index.d.ts" />
/// <reference path="globals/sinon/index.d.ts" />
/// <reference path="modules/form-data/index.d.ts" />
/// <reference path="modules/graphql/index.d.ts" />
/// <reference path="../dataloader.d.ts" /> then I get the error in tsc: "error TS2307: Cannot find module 'dataloader'." what somehow makes sense too because now there is no TS module defined, right?. Many thanks for your help!! |
IIRC, the last line: If that is correct, then it actually only exports in commonjs format and you have to use it as import dataloader = require('dataloader') I might be wrong though. |
with
It seems that tsc only recognises the type information (DataLoader as a class) when I use Any other ideas? |
@unional I cloned 'facebook/dataloader' and built it by default (babel5) what generates exactly the same dist file with the Do you know what is the change between babel5 and babel6 that has originated this problem? My definition now looks a bit better but it is still weird and feels like a complete workaround: // ... types omitted
interface IDataLoader<K, V> {
new (batchLoadFn: BatchLoadFn<K, V>, options?: Options<K, V>);
// ... definitions omitted
}
declare const DataLoader: IDataLoader<any, any>;
declare module 'dataloader' {
export = DataLoader;
} I have to import it like this: import DataLoader from 'dataloader'; And use it like this: const dataloader: IDataLoader = new DataLoader(...); Am I still missing something here? Thanks a lot for the feedback!! |
Just for the record: The right way of importing a package like facebook/dataloader, which is a es6 class exported as default and which happens to be built with babel5 (as per version 1.2.0) and therefor has a import * as DataLoader from 'dataloader'; // commonjs module
// or
import DataLoader = require('dataloader'); // commonjs module
// but NOT -- WRONG!!
import DataLoader from 'dataloader'; // es6 module
// and sure NOT -- WRONG!!
import { DataLoader } from 'dataloader'; // named import of something which does not exist Usage in code: const dataloader: IDataLoader = new DataLoader(...); // IDataLoader comes from the type definition, so no need to import it And the type definition should look as shown here @blakeembrey any feedback on this issue before I close it? Am I completely lost here? or should I send a PR with an example for the documentation to help others falling on this trap? cheers, |
@almilo Nothing major. I recommend checking out https://www.typescriptlang.org/docs/handbook/modules.html (specifically |
Finally adjusted my definition as: declare class DataLoader<K, V> {
constructor(batchLoadFn: BatchLoadFn<K, V>, options?: Options<K, V>);
// ... definitions omitted
}
declare module 'dataloader' {
export = DataLoader;
} and the usage: import DataLoader = require('dataloader'); // it really only works if import = notation is used
const dataloader = new DataLoader<Object, Object>(...); Note 1: typescript@1.8.10 Thanks for the hints! This was neither an easy learning, nor I can say that I love the solution indeed :( |
If you write the definition using external module syntax (just |
Prerequisites
Description
I am trying to write a definition for facebook/dataloader in oder to use it in a TS project.
DataLoader seems to be a es6 class exported as default.
After reading the examples and other articles and issues, still cannot find the way to 'type' the default export and import it as any other class from TypeScript.
I am using TS 1.8.10 with this configuration.
I would like to write a definition like:
and use it like:
What is not working as it seems that the import to use must be of the form:
which seems to disallow the import of the class.
The question is 'how to write such a definition'? (for a es6 class transpiled into es5 as es6 module but exported as default).
Although probably not a problem with typings itself, I think it would be great to add this case to the examples.
Thanks!!
Alberto
The text was updated successfully, but these errors were encountered: