From 136dfa02e987ad09f3ef026d02f366f405b65ff3 Mon Sep 17 00:00:00 2001 From: Aliaksei Tuzik Date: Sun, 12 Dec 2021 10:41:08 +0400 Subject: [PATCH 1/2] chore: update flow-bin dependency In order to support specifying "this" type for a function --- .flowconfig | 1 - package.json | 2 +- src/__tests__/abuse.test.js | 27 ++++++++++++++------------- src/__tests__/dataloader.test.js | 1 + src/index.js | 7 +++++-- yarn.lock | 8 ++++---- 6 files changed, 25 insertions(+), 21 deletions(-) diff --git a/.flowconfig b/.flowconfig index 6208195..be9f3bd 100644 --- a/.flowconfig +++ b/.flowconfig @@ -10,5 +10,4 @@ [libs] [options] -suppress_comment=\\(.\\|\n\\)*\\$FlowExpectError include_warnings=true diff --git a/package.json b/package.json index 2a4c66a..433cf6b 100644 --- a/package.json +++ b/package.json @@ -47,7 +47,7 @@ "babel-eslint": "10.0.3", "coveralls": "3.0.7", "eslint": "6.6.0", - "flow-bin": "0.112.0", + "flow-bin": "0.167.0", "jest": "24.9.0", "sane": "4.1.0" } diff --git a/src/__tests__/abuse.test.js b/src/__tests__/abuse.test.js index 9300e80..c905c97 100644 --- a/src/__tests__/abuse.test.js +++ b/src/__tests__/abuse.test.js @@ -13,7 +13,7 @@ describe('Provides descriptive error messages for API abuse', () => { it('Loader creation requires a function', () => { expect(() => { - // $FlowExpectError + // $FlowExpectedError[incompatible-call] new DataLoader(); // eslint-disable-line no-new }).toThrow( 'DataLoader must be constructed with a function which accepts ' + @@ -21,7 +21,7 @@ describe('Provides descriptive error messages for API abuse', () => { ); expect(() => { - // $FlowExpectError + // $FlowExpectedError[prop-missing] new DataLoader({}); // eslint-disable-line no-new }).toThrow( 'DataLoader must be constructed with a function which accepts ' + @@ -33,7 +33,7 @@ describe('Provides descriptive error messages for API abuse', () => { const idLoader = new DataLoader(async keys => keys); expect(() => { - // $FlowExpectError + // $FlowExpectedError[incompatible-call] idLoader.load(); }).toThrow( 'The loader.load() function must be called with a value, ' + @@ -41,7 +41,7 @@ describe('Provides descriptive error messages for API abuse', () => { ); expect(() => { - // $FlowExpectError + // $FlowExpectedError[incompatible-call] idLoader.load(null); }).toThrow( 'The loader.load() function must be called with a value, ' + @@ -58,7 +58,7 @@ describe('Provides descriptive error messages for API abuse', () => { const idLoader = new DataLoader(async keys => keys); expect(() => { - // $FlowExpectError + // $FlowExpectedError[incompatible-call] idLoader.loadMany(); }).toThrow( 'The loader.loadMany() function must be called with Array ' + @@ -66,7 +66,8 @@ describe('Provides descriptive error messages for API abuse', () => { ); expect(() => { - // $FlowExpectError + // $FlowExpectedError[incompatible-call] + // $FlowExpectedError[extra-arg] idLoader.loadMany(1, 2, 3); }).toThrow( 'The loader.loadMany() function must be called with Array ' + @@ -80,7 +81,7 @@ describe('Provides descriptive error messages for API abuse', () => { }); it('Batch function must return a Promise, not null', async () => { - // $FlowExpectError + // $FlowExpectedError[incompatible-call] const badLoader = new DataLoader(() => null); let caughtError; @@ -99,7 +100,7 @@ describe('Provides descriptive error messages for API abuse', () => { it('Batch function must return a Promise, not a value', async () => { // Note: this is returning the keys directly, rather than a promise to keys. - // $FlowExpectError + // $FlowExpectedError[incompatible-call] const badLoader = new DataLoader(keys => keys); let caughtError; @@ -118,7 +119,7 @@ describe('Provides descriptive error messages for API abuse', () => { it('Batch function must return a Promise of an Array, not null', async () => { // Note: this resolves to undefined - // $FlowExpectError + // $FlowExpectedError[incompatible-call] const badLoader = new DataLoader(async () => null); let caughtError; @@ -162,9 +163,9 @@ describe('Provides descriptive error messages for API abuse', () => { } expect(() => { - // $FlowExpectError const incompleteMap = new IncompleteMap(); const options = { cacheMap: incompleteMap }; + // $FlowExpectedError[incompatible-call] new DataLoader(async keys => keys, options); // eslint-disable-line no-new }).toThrow( 'Custom cacheMap missing methods: set, delete, clear' @@ -173,7 +174,7 @@ describe('Provides descriptive error messages for API abuse', () => { it('Requires a number for maxBatchSize', () => { expect(() => - // $FlowExpectError + // $FlowExpectedError[incompatible-call] new DataLoader(async keys => keys, { maxBatchSize: null }) ).toThrow('maxBatchSize must be a positive number: null'); }); @@ -186,14 +187,14 @@ describe('Provides descriptive error messages for API abuse', () => { it('Requires a function for cacheKeyFn', () => { expect(() => - // $FlowExpectError + // $FlowExpectedError[incompatible-call] new DataLoader(async keys => keys, { cacheKeyFn: null }) ).toThrow('cacheKeyFn must be a function: null'); }); it('Requires a function for batchScheduleFn', () => { expect(() => - // $FlowExpectError + // $FlowExpectedError[incompatible-call] new DataLoader(async keys => keys, { batchScheduleFn: null }) ).toThrow('batchScheduleFn must be a function: null'); }); diff --git a/src/__tests__/dataloader.test.js b/src/__tests__/dataloader.test.js index 47b2059..4b050b5 100644 --- a/src/__tests__/dataloader.test.js +++ b/src/__tests__/dataloader.test.js @@ -50,6 +50,7 @@ describe('Primary API', () => { let that; const loader = new DataLoader(async keys => keys, { cacheKeyFn(key) { + // $FlowIgnore[object-this-reference] that = this; return key; } diff --git a/src/index.js b/src/index.js index 6250df9..9987819 100644 --- a/src/index.js +++ b/src/index.js @@ -24,12 +24,12 @@ export type Options = { }; // If a custom cache is provided, it must be of this type (a subset of ES6 Map). -export type CacheMap = { +export interface CacheMap { get(key: K): V | void; set(key: K, value: V): any; delete(key: K): any; clear(): any; -}; +} /** * A `DataLoader` creates a public API for loading data from a particular @@ -303,6 +303,7 @@ function dispatchBatch( var batchPromise = loader._batchLoadFn(batch.keys); // Assert the expected response from batchLoadFn + // $FlowIgnore[method-unbinding] if (!batchPromise || typeof batchPromise.then !== 'function') { return failedDispatch(loader, batch, new TypeError( 'DataLoader must be constructed with a function which accepts ' + @@ -435,6 +436,7 @@ function getValidCacheMap( if (cacheMap !== null) { var cacheFunctions = [ 'get', 'set', 'delete', 'clear' ]; var missingFunctions = cacheFunctions + // $FlowIgnore[prop-missing] .filter(fnName => cacheMap && typeof cacheMap[fnName] !== 'function'); if (missingFunctions.length !== 0) { throw new TypeError( @@ -452,6 +454,7 @@ function isArrayLike(x: mixed): boolean { x !== null && typeof x.length === 'number' && (x.length === 0 || + // $FlowIgnore[method-unbinding] (x.length > 0 && Object.prototype.hasOwnProperty.call(x, x.length - 1))) ); } diff --git a/yarn.lock b/yarn.lock index 056aa62..6f703ad 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2010,10 +2010,10 @@ flatted@^2.0.0: resolved "https://registry.yarnpkg.com/flatted/-/flatted-2.0.1.tgz#69e57caa8f0eacbc281d2e2cb458d46fdb449e08" integrity sha512-a1hQMktqW9Nmqr5aktAux3JMNqaucxGcjtjWnZLHX7yyPCmlSV3M54nGYbqT8K+0GhF3NBgmJCc3ma+WOgX8Jg== -flow-bin@0.112.0: - version "0.112.0" - resolved "https://registry.yarnpkg.com/flow-bin/-/flow-bin-0.112.0.tgz#6a21c31937c4a2f23a750056a364c598a95ea216" - integrity sha512-vdcuKv0UU55vjv0e2EVh1ZxlU+TSNT19SkE+6gT1vYzTKtzYE6dLuAmBIiS3Rg2N9D9HOI6TKSyl53zPtqZLrA== +flow-bin@0.167.0: + version "0.167.0" + resolved "https://registry.yarnpkg.com/flow-bin/-/flow-bin-0.167.0.tgz#65f42d962707b015f6165b2f3536c4a8701fa929" + integrity sha512-/J64i+BN1Y8xzpdzRsML/Mij/k2G1d0UGUnMBUxN8ALrClPTQOxsOgDlh9wpEzNM/5yY7iu6eQYw8TnLbw6H5Q== for-in@^1.0.2: version "1.0.2" From b9419ffa9ae917e9baaab426333b34796b607008 Mon Sep 17 00:00:00 2001 From: Aliaksei Tuzik Date: Sun, 12 Dec 2021 11:46:03 +0400 Subject: [PATCH 2/2] feat(typings): allow to reference the loader instance in the batch function --- src/__tests__/dataloader.test.js | 2 +- src/index.d.ts | 6 +++--- src/index.js | 8 ++++---- 3 files changed, 8 insertions(+), 8 deletions(-) diff --git a/src/__tests__/dataloader.test.js b/src/__tests__/dataloader.test.js index 4b050b5..d786150 100644 --- a/src/__tests__/dataloader.test.js +++ b/src/__tests__/dataloader.test.js @@ -34,7 +34,7 @@ describe('Primary API', () => { }); it('references the loader as "this" in the batch function', async () => { - let that; + let that: DataLoader; const loader = new DataLoader(async function (keys) { that = this; return keys; diff --git a/src/index.d.ts b/src/index.d.ts index 71c2bdb..b0a3868 100644 --- a/src/index.d.ts +++ b/src/index.d.ts @@ -17,7 +17,7 @@ */ declare class DataLoader { - constructor(batchLoadFn: DataLoader.BatchLoadFn, options?: DataLoader.Options); + constructor(batchLoadFn: DataLoader.BatchLoadFn, options?: DataLoader.Options); /** * Loads a key, returning a `Promise` for the value represented by that key. @@ -70,8 +70,8 @@ declare namespace DataLoader { // A Function, which when given an Array of keys, returns a Promise of an Array // of values or Errors. - export type BatchLoadFn = - (keys: ReadonlyArray) => PromiseLike>; + export type BatchLoadFn = + (this: DataLoader, keys: ReadonlyArray) => PromiseLike>; // Optionally turn off batching or caching or provide a cache key function or a // custom cache instance. diff --git a/src/index.js b/src/index.js index 9987819..e73f393 100644 --- a/src/index.js +++ b/src/index.js @@ -9,8 +9,8 @@ // A Function, which when given an Array of keys, returns a Promise of an Array // of values or Errors. -export type BatchLoadFn = - (keys: $ReadOnlyArray) => Promise<$ReadOnlyArray>; +export type BatchLoadFn = + (this: DataLoader, keys: $ReadOnlyArray) => Promise<$ReadOnlyArray>; // Optionally turn off batching or caching or provide a cache key function or a // custom cache instance. @@ -43,7 +43,7 @@ export interface CacheMap { */ class DataLoader { constructor( - batchLoadFn: BatchLoadFn, + batchLoadFn: BatchLoadFn, options?: Options ) { if (typeof batchLoadFn !== 'function') { @@ -61,7 +61,7 @@ class DataLoader { } // Private - _batchLoadFn: BatchLoadFn; + _batchLoadFn: BatchLoadFn; _maxBatchSize: number; _batchScheduleFn: (() => void) => void; _cacheKeyFn: K => C;