@rudderstack/integrations-lib 0.2.33 → 0.2.34

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=batch-processing.bench.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"batch-processing.bench.d.ts","sourceRoot":"","sources":["../../src/utils/batch-processing.bench.ts"],"names":[],"mappings":""}
@@ -0,0 +1,96 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ /* eslint-disable no-console */
7
+ const node_perf_hooks_1 = require("node:perf_hooks");
8
+ const lodash_1 = __importDefault(require("lodash"));
9
+ const batch_processing_1 = require("./batch-processing");
10
+ async function bench(fn, label, iterations = 5) {
11
+ let total = 0;
12
+ for (let i = 0; i < iterations; i += 1) {
13
+ const start = node_perf_hooks_1.performance.now();
14
+ // eslint-disable-next-line no-await-in-loop
15
+ await fn();
16
+ total += node_perf_hooks_1.performance.now() - start;
17
+ }
18
+ const avg = total / iterations;
19
+ console.log(`${label}: avg ${avg.toFixed(2)} ms over ${iterations} runs`);
20
+ return avg;
21
+ }
22
+ async function section(title, fn) {
23
+ console.log(`\n=== ${title} ===`);
24
+ await fn();
25
+ }
26
+ async function runBenchmarks() {
27
+ const arr = Array.from({ length: 10_000 }, (_, i) => i);
28
+ console.log(`Running benchmarks on array of length ${arr.length}`);
29
+ await section('map', async () => {
30
+ await bench(() => (0, batch_processing_1.mapInBatches)(arr, (x) => x + 1, { batchSize: 20, yieldThreshold: 10 }), 'mapInBatches { batchSize: 20, yieldThreshold: 10 }');
31
+ await bench(() => (0, batch_processing_1.mapInBatches)(arr, (x) => x + 1, {
32
+ batchSize: 20,
33
+ yieldThreshold: 10,
34
+ sequentialProcessing: false,
35
+ }), 'mapInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }');
36
+ await bench(() => Promise.resolve(lodash_1.default.map(arr, (x) => x + 1)), 'lodash.map');
37
+ await bench(() => Promise.resolve(arr.map((x) => x + 1)), 'Array.prototype.map');
38
+ });
39
+ await section('filter', async () => {
40
+ await bench(() => (0, batch_processing_1.filterInBatches)(arr, (x) => x % 2 === 0, { batchSize: 20, yieldThreshold: 10 }), 'filterInBatches { batchSize: 20, yieldThreshold: 10 }');
41
+ await bench(() => (0, batch_processing_1.filterInBatches)(arr, (x) => x % 2 === 0, {
42
+ batchSize: 20,
43
+ yieldThreshold: 10,
44
+ sequentialProcessing: false,
45
+ }), 'filterInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }');
46
+ await bench(() => Promise.resolve(lodash_1.default.filter(arr, (x) => x % 2 === 0)), 'lodash.filter');
47
+ await bench(() => Promise.resolve(arr.filter((x) => x % 2 === 0)), 'Array.prototype.filter');
48
+ });
49
+ await section('groupBy', async () => {
50
+ await bench(() => (0, batch_processing_1.groupByInBatches)(arr, (x) => (x % 2 === 0 ? 'even' : 'odd'), {
51
+ batchSize: 20,
52
+ yieldThreshold: 10,
53
+ }), 'groupByInBatches { batchSize: 20, yieldThreshold: 10 }');
54
+ await bench(() => (0, batch_processing_1.groupByInBatches)(arr, (x) => (x % 2 === 0 ? 'even' : 'odd'), {
55
+ batchSize: 20,
56
+ yieldThreshold: 10,
57
+ sequentialProcessing: false,
58
+ }), 'groupByInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }');
59
+ await bench(() => Promise.resolve(lodash_1.default.groupBy(arr, (x) => (x % 2 === 0 ? 'even' : 'odd'))), 'lodash.groupBy');
60
+ });
61
+ await section('reduce', async () => {
62
+ await bench(() => (0, batch_processing_1.reduceInBatches)(arr, (acc, x) => acc + x, 0, { batchSize: 20, yieldThreshold: 10 }), 'reduceInBatches { batchSize: 20, yieldThreshold: 10 }');
63
+ await bench(() => Promise.resolve(lodash_1.default.reduce(arr, (acc, x) => acc + x, 0)), 'lodash.reduce');
64
+ await bench(() => Promise.resolve(arr.reduce((acc, x) => acc + x, 0)), 'Array.prototype.reduce');
65
+ });
66
+ await section('flatMap', async () => {
67
+ await bench(() => (0, batch_processing_1.flatMapInBatches)(arr, (x) => [x, x], { batchSize: 20, yieldThreshold: 10 }), 'flatMapInBatches { batchSize: 20, yieldThreshold: 10 }');
68
+ await bench(() => (0, batch_processing_1.flatMapInBatches)(arr, (x) => [x, x], {
69
+ batchSize: 20,
70
+ yieldThreshold: 10,
71
+ sequentialProcessing: false,
72
+ }), 'flatMapInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }');
73
+ await bench(() => Promise.resolve(lodash_1.default.flatMap(arr, (x) => [x, x])), 'lodash.flatMap');
74
+ await bench(() => Promise.resolve(arr.flatMap
75
+ ? arr.flatMap((x) => [x, x])
76
+ : [].concat(...arr.map((x) => [x, x]))), 'Array.prototype.flatMap');
77
+ });
78
+ await section('forEach', async () => {
79
+ await bench(() => (0, batch_processing_1.forEachInBatches)(arr, () => { }, { batchSize: 20, yieldThreshold: 10 }), 'forEachInBatches { batchSize: 20, yieldThreshold: 10 }');
80
+ await bench(() => (0, batch_processing_1.forEachInBatches)(arr, () => { }, {
81
+ batchSize: 20,
82
+ yieldThreshold: 10,
83
+ sequentialProcessing: false,
84
+ }), 'forEachInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }');
85
+ await bench(() => Promise.resolve(arr.forEach(() => { })), 'Array.prototype.forEach');
86
+ await bench(() => Promise.resolve(lodash_1.default.forEach(arr, () => { })), 'lodash.forEach');
87
+ });
88
+ }
89
+ // to run benchmarks use: npx ts-node src/utils/batch-processing.bench.ts
90
+ if (require.main === module) {
91
+ runBenchmarks().catch((err) => {
92
+ console.error(err);
93
+ process.exit(1);
94
+ });
95
+ }
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"batch-processing.bench.js","sourceRoot":"","sources":["../../src/utils/batch-processing.bench.ts"],"names":[],"mappings":";;;;;AAAA,+BAA+B;AAC/B,qDAA8C;AAC9C,oDAA4B;AAC5B,yDAO4B;AAE5B,KAAK,UAAU,KAAK,CAClB,EAAoC,EACpC,KAAa,EACb,UAAU,GAAG,CAAC;IAEd,IAAI,KAAK,GAAG,CAAC,CAAC;IACd,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC;QACvC,MAAM,KAAK,GAAG,6BAAW,CAAC,GAAG,EAAE,CAAC;QAChC,4CAA4C;QAC5C,MAAM,EAAE,EAAE,CAAC;QACX,KAAK,IAAI,6BAAW,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC;IACrC,CAAC;IACD,MAAM,GAAG,GAAG,KAAK,GAAG,UAAU,CAAC;IAC/B,OAAO,CAAC,GAAG,CAAC,GAAG,KAAK,SAAS,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,YAAY,UAAU,OAAO,CAAC,CAAC;IAC1E,OAAO,GAAG,CAAC;AACb,CAAC;AAED,KAAK,UAAU,OAAO,CAAC,KAAa,EAAE,EAAuB;IAC3D,OAAO,CAAC,GAAG,CAAC,SAAS,KAAK,MAAM,CAAC,CAAC;IAClC,MAAM,EAAE,EAAE,CAAC;AACb,CAAC;AAED,KAAK,UAAU,aAAa;IAC1B,MAAM,GAAG,GAAG,KAAK,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;IAExD,OAAO,CAAC,GAAG,CAAC,yCAAyC,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;IAEnE,MAAM,OAAO,CAAC,KAAK,EAAE,KAAK,IAAI,EAAE;QAC9B,MAAM,KAAK,CACT,GAAG,EAAE,CAAC,IAAA,+BAAY,EAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC,EAC5E,oDAAoD,CACrD,CAAC;QACF,MAAM,KAAK,CACT,GAAG,EAAE,CACH,IAAA,+BAAY,EAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,EAAE;YAC9B,SAAS,EAAE,EAAE;YACb,cAAc,EAAE,EAAE;YAClB,oBAAoB,EAAE,KAAK;SAC5B,CAAC,EACJ,iFAAiF,CAClF,CAAC;QACF,MAAM,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,gBAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;QAChF,MAAM,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,qBAAqB,CAAC,CAAC;IACnF,CAAC,CAAC,CAAC;IAEH,MAAM,OAAO,CAAC,QAAQ,EAAE,KAAK,IAAI,EAAE;QACjC,MAAM,KAAK,CACT,GAAG,EAAE,CAAC,IAAA,kCAAe,EAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC,EACrF,uDAAuD,CACxD,CAAC;QACF,MAAM,KAAK,CACT,GAAG,EAAE,CACH,IAAA,kCAAe,EAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE;YACvC,SAAS,EAAE,EAAE;YACb,cAAc,EAAE,EAAE;YAClB,oBAAoB,EAAE,KAAK;SAC5B,CAAC,EACJ,oFAAoF,CACrF,CAAC;QACF,MAAM,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,gBAAM,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,eAAe,CAAC,CAAC;QAC5F,MAAM,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,wBAAwB,CAAC,CAAC;IAC/F,CAAC,CAAC,CAAC;IAEH,MAAM,OAAO,CAAC,SAAS,EAAE,KAAK,IAAI,EAAE;QAClC,MAAM,KAAK,CACT,GAAG,EAAE,CACH,IAAA,mCAAgB,EAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,EAAE;YAC3D,SAAS,EAAE,EAAE;YACb,cAAc,EAAE,EAAE;SACnB,CAAC,EACJ,wDAAwD,CACzD,CAAC;QACF,MAAM,KAAK,CACT,GAAG,EAAE,CACH,IAAA,mCAAgB,EAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,EAAE;YAC3D,SAAS,EAAE,EAAE;YACb,cAAc,EAAE,EAAE;YAClB,oBAAoB,EAAE,KAAK;SAC5B,CAAC,EACJ,qFAAqF,CACtF,CAAC;QACF,MAAM,KAAK,CACT,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,gBAAM,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EACjF,gBAAgB,CACjB,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,MAAM,OAAO,CAAC,QAAQ,EAAE,KAAK,IAAI,EAAE;QACjC,MAAM,KAAK,CACT,GAAG,EAAE,CAAC,IAAA,kCAAe,EAAC,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC,EACzF,uDAAuD,CACxD,CAAC;QACF,MAAM,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,gBAAM,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,eAAe,CAAC,CAAC;QAChG,MAAM,KAAK,CACT,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EACzD,wBAAwB,CACzB,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,MAAM,OAAO,CAAC,SAAS,EAAE,KAAK,IAAI,EAAE;QAClC,MAAM,KAAK,CACT,GAAG,EAAE,CAAC,IAAA,mCAAgB,EAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC,EACjF,wDAAwD,CACzD,CAAC;QACF,MAAM,KAAK,CACT,GAAG,EAAE,CACH,IAAA,mCAAgB,EAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE;YACnC,SAAS,EAAE,EAAE;YACb,cAAc,EAAE,EAAE;YAClB,oBAAoB,EAAE,KAAK;SAC5B,CAAC,EACJ,qFAAqF,CACtF,CAAC;QACF,MAAM,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,gBAAM,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,gBAAgB,CAAC,CAAC;QACzF,MAAM,KAAK,CACT,GAAG,EAAE,CACH,OAAO,CAAC,OAAO,CACb,GAAG,CAAC,OAAO;YACT,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;YAC5B,CAAC,CAAE,EAAe,CAAC,MAAM,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CACvD,EACH,yBAAyB,CAC1B,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,MAAM,OAAO,CAAC,SAAS,EAAE,KAAK,IAAI,EAAE;QAClC,MAAM,KAAK,CACT,GAAG,EAAE,CAAC,IAAA,mCAAgB,EAAC,GAAG,EAAE,GAAG,EAAE,GAAE,CAAC,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC,EAC5E,wDAAwD,CACzD,CAAC;QACF,MAAM,KAAK,CACT,GAAG,EAAE,CACH,IAAA,mCAAgB,EAAC,GAAG,EAAE,GAAG,EAAE,GAAE,CAAC,EAAE;YAC9B,SAAS,EAAE,EAAE;YACb,cAAc,EAAE,EAAE;YAClB,oBAAoB,EAAE,KAAK;SAC5B,CAAC,EACJ,qFAAqF,CACtF,CAAC;QACF,MAAM,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC,EAAE,yBAAyB,CAAC,CAAC;QACrF,MAAM,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,gBAAM,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC,EAAE,gBAAgB,CAAC,CAAC;IACtF,CAAC,CAAC,CAAC;AACL,CAAC;AAED,yEAAyE;AACzE,IAAI,OAAO,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;IAC5B,aAAa,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;QAC5B,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACnB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC,CAAC,CAAC;AACL,CAAC","sourcesContent":["/* eslint-disable no-console */\nimport { performance } from 'node:perf_hooks';\nimport lodash from 'lodash';\nimport {\n  mapInBatches,\n  filterInBatches,\n  groupByInBatches,\n  reduceInBatches,\n  flatMapInBatches,\n  forEachInBatches,\n} from './batch-processing';\n\nasync function bench(\n  fn: () => Promise<unknown> | unknown,\n  label: string,\n  iterations = 5,\n): Promise<number> {\n  let total = 0;\n  for (let i = 0; i < iterations; i += 1) {\n    const start = performance.now();\n    // eslint-disable-next-line no-await-in-loop\n    await fn();\n    total += performance.now() - start;\n  }\n  const avg = total / iterations;\n  console.log(`${label}: avg ${avg.toFixed(2)} ms over ${iterations} runs`);\n  return avg;\n}\n\nasync function section(title: string, fn: () => Promise<void>) {\n  console.log(`\\n=== ${title} ===`);\n  await fn();\n}\n\nasync function runBenchmarks() {\n  const arr = Array.from({ length: 10_000 }, (_, i) => i);\n\n  console.log(`Running benchmarks on array of length ${arr.length}`);\n\n  await section('map', async () => {\n    await bench(\n      () => mapInBatches(arr, (x) => x + 1, { batchSize: 20, yieldThreshold: 10 }),\n      'mapInBatches { batchSize: 20, yieldThreshold: 10 }',\n    );\n    await bench(\n      () =>\n        mapInBatches(arr, (x) => x + 1, {\n          batchSize: 20,\n          yieldThreshold: 10,\n          sequentialProcessing: false,\n        }),\n      'mapInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }',\n    );\n    await bench(() => Promise.resolve(lodash.map(arr, (x) => x + 1)), 'lodash.map');\n    await bench(() => Promise.resolve(arr.map((x) => x + 1)), 'Array.prototype.map');\n  });\n\n  await section('filter', async () => {\n    await bench(\n      () => filterInBatches(arr, (x) => x % 2 === 0, { batchSize: 20, yieldThreshold: 10 }),\n      'filterInBatches { batchSize: 20, yieldThreshold: 10 }',\n    );\n    await bench(\n      () =>\n        filterInBatches(arr, (x) => x % 2 === 0, {\n          batchSize: 20,\n          yieldThreshold: 10,\n          sequentialProcessing: false,\n        }),\n      'filterInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }',\n    );\n    await bench(() => Promise.resolve(lodash.filter(arr, (x) => x % 2 === 0)), 'lodash.filter');\n    await bench(() => Promise.resolve(arr.filter((x) => x % 2 === 0)), 'Array.prototype.filter');\n  });\n\n  await section('groupBy', async () => {\n    await bench(\n      () =>\n        groupByInBatches(arr, (x) => (x % 2 === 0 ? 'even' : 'odd'), {\n          batchSize: 20,\n          yieldThreshold: 10,\n        }),\n      'groupByInBatches { batchSize: 20, yieldThreshold: 10 }',\n    );\n    await bench(\n      () =>\n        groupByInBatches(arr, (x) => (x % 2 === 0 ? 'even' : 'odd'), {\n          batchSize: 20,\n          yieldThreshold: 10,\n          sequentialProcessing: false,\n        }),\n      'groupByInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }',\n    );\n    await bench(\n      () => Promise.resolve(lodash.groupBy(arr, (x) => (x % 2 === 0 ? 'even' : 'odd'))),\n      'lodash.groupBy',\n    );\n  });\n\n  await section('reduce', async () => {\n    await bench(\n      () => reduceInBatches(arr, (acc, x) => acc + x, 0, { batchSize: 20, yieldThreshold: 10 }),\n      'reduceInBatches { batchSize: 20, yieldThreshold: 10 }',\n    );\n    await bench(() => Promise.resolve(lodash.reduce(arr, (acc, x) => acc + x, 0)), 'lodash.reduce');\n    await bench(\n      () => Promise.resolve(arr.reduce((acc, x) => acc + x, 0)),\n      'Array.prototype.reduce',\n    );\n  });\n\n  await section('flatMap', async () => {\n    await bench(\n      () => flatMapInBatches(arr, (x) => [x, x], { batchSize: 20, yieldThreshold: 10 }),\n      'flatMapInBatches { batchSize: 20, yieldThreshold: 10 }',\n    );\n    await bench(\n      () =>\n        flatMapInBatches(arr, (x) => [x, x], {\n          batchSize: 20,\n          yieldThreshold: 10,\n          sequentialProcessing: false,\n        }),\n      'flatMapInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }',\n    );\n    await bench(() => Promise.resolve(lodash.flatMap(arr, (x) => [x, x])), 'lodash.flatMap');\n    await bench(\n      () =>\n        Promise.resolve(\n          arr.flatMap\n            ? arr.flatMap((x) => [x, x])\n            : ([] as number[]).concat(...arr.map((x) => [x, x])),\n        ),\n      'Array.prototype.flatMap',\n    );\n  });\n\n  await section('forEach', async () => {\n    await bench(\n      () => forEachInBatches(arr, () => {}, { batchSize: 20, yieldThreshold: 10 }),\n      'forEachInBatches { batchSize: 20, yieldThreshold: 10 }',\n    );\n    await bench(\n      () =>\n        forEachInBatches(arr, () => {}, {\n          batchSize: 20,\n          yieldThreshold: 10,\n          sequentialProcessing: false,\n        }),\n      'forEachInBatches { batchSize: 20, yieldThreshold: 10, sequentialProcessing: false }',\n    );\n    await bench(() => Promise.resolve(arr.forEach(() => {})), 'Array.prototype.forEach');\n    await bench(() => Promise.resolve(lodash.forEach(arr, () => {})), 'lodash.forEach');\n  });\n}\n\n// to run benchmarks use: npx ts-node src/utils/batch-processing.bench.ts\nif (require.main === module) {\n  runBenchmarks().catch((err) => {\n    console.error(err);\n    process.exit(1);\n  });\n}\n"]}
@@ -1,15 +1,23 @@
1
1
  /**
2
- * Configuration options for batch processing operations
2
+ * Options for batch processing operations
3
3
  */
4
- export interface BatchProcessingDefaults {
4
+ export interface BatchProcessingOptions {
5
5
  /**
6
- * Default number of items to process in each batch
6
+ * Number of items to process in each batch (default: 10)
7
+ * Must be a positive integer.
7
8
  */
8
9
  batchSize?: number;
9
10
  /**
10
- * Default time threshold in milliseconds before yielding control back to the event loop
11
+ * Time threshold in milliseconds (default: 10) before yielding control back to the event loop.
12
+ * Set to 0 to yield after every batch. Must be a non-negative integer.
11
13
  */
12
14
  yieldThreshold?: number;
15
+ /**
16
+ * Whether to process items sequentially within each batch. When true (default), each item in a batch
17
+ * will be processed one at a time. When false, all items in a batch will be processed concurrently.
18
+ * Consider the implications of concurrency on your processing logic before setting this to false, e.g. race conditions, rate limits, memory pressure, etc.
19
+ */
20
+ sequentialProcessing?: boolean;
13
21
  }
14
22
  /**
15
23
  * Configure global defaults for batch processing operations
@@ -25,11 +33,14 @@ export interface BatchProcessingDefaults {
25
33
  * // Set only batch size
26
34
  * configureBatchProcessingDefaults({ batchSize: 50 });
27
35
  *
36
+ * // Enable concurrent processing within batches
37
+ * configureBatchProcessingDefaults({ sequentialProcessing: false });
38
+ *
28
39
  * // Get current configuration
29
40
  * const currentConfig = configureBatchProcessingDefaults();
30
41
  * ```
31
42
  */
32
- export declare function configureBatchProcessingDefaults(config?: BatchProcessingDefaults): BatchProcessingDefaults;
43
+ export declare function configureBatchProcessingDefaults(config?: BatchProcessingOptions): BatchProcessingOptions;
33
44
  /**
34
45
  * Maps over an array in batches to avoid blocking the event loop.
35
46
  * Processes items in chunks and yields control back to the event loop between batches
@@ -39,26 +50,20 @@ export declare function configureBatchProcessingDefaults(config?: BatchProcessin
39
50
  * @template R - The type of items in the result array
40
51
  * @param items - The array to map over
41
52
  * @param mapFn - The mapping function to apply to each item. Receives the item and its index.
42
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
43
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
53
+ * @param options - Batch processing options
44
54
  * @returns A promise that resolves to the mapped array
45
55
  * @throws {Error} When batchSize is not a positive integer
46
56
  *
47
57
  * @example
48
58
  * ```typescript
49
- * // Synchronous mapping
50
- * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2, 2);
51
- * // Result: [2, 4, 6, 8]
52
- *
53
- * // Asynchronous mapping
54
- * const fetched = await mapInBatches(urls, async (url) => fetch(url), 3);
59
+ * // Synchronous mapping with default options (sequential processing)
60
+ * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2);
55
61
  *
56
- * // With index
57
- * const indexed = await mapInBatches(['a', 'b'], (item, index) => `${item}-${index}`);
58
- * // Result: ['a-0', 'b-1']
62
+ * // With concurrent processing within batches
63
+ * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2, { sequentialProcessing: false });
59
64
  * ```
60
65
  */
61
- export declare function mapInBatches<T, R>(items: T[], mapFn: (item: T, index: number) => R | Promise<R>, batchSize?: number, yieldThreshold?: number): Promise<R[]>;
66
+ export declare function mapInBatches<T, R>(items: T[], mapFn: (item: T, index: number) => R | Promise<R>, options?: BatchProcessingOptions): Promise<R[]>;
62
67
  /**
63
68
  * Filters an array in batches to avoid blocking the event loop.
64
69
  * Processes items in chunks and yields control back to the event loop between batches
@@ -67,29 +72,21 @@ export declare function mapInBatches<T, R>(items: T[], mapFn: (item: T, index: n
67
72
  * @template T - The type of items in the array
68
73
  * @param items - The array to filter
69
74
  * @param predicate - The predicate function to test each item. Receives the item and its index.
70
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
71
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
75
+ * @param options - Batch processing options
72
76
  * @returns A promise that resolves to the filtered array
73
77
  * @throws {Error} When batchSize is not a positive integer
74
78
  *
75
79
  * @example
76
80
  * ```typescript
77
- * // Synchronous filtering
78
- * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, 2);
81
+ * // Synchronous filtering with default options
82
+ * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0);
79
83
  * // Result: [2, 4]
80
84
  *
81
- * // Asynchronous filtering
82
- * const valid = await filterInBatches(urls, async (url) => {
83
- * const response = await fetch(url);
84
- * return response.ok;
85
- * }, 3);
86
- *
87
- * // With index
88
- * const evenIndices = await filterInBatches(['a', 'b', 'c'], (_, index) => index % 2 === 0);
89
- * // Result: ['a', 'c']
85
+ * // With custom batch size
86
+ * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, { batchSize: 2 });
90
87
  * ```
91
88
  */
92
- export declare function filterInBatches<T>(items: T[], predicate: (item: T, index: number) => boolean | Promise<boolean>, batchSize?: number, yieldThreshold?: number): Promise<T[]>;
89
+ export declare function filterInBatches<T>(items: T[], predicate: (item: T, index: number) => boolean | Promise<boolean>, options?: BatchProcessingOptions): Promise<T[]>;
93
90
  /**
94
91
  * Groups an array by a key function in batches to avoid blocking the event loop.
95
92
  * Processes items in chunks and yields control back to the event loop between batches
@@ -99,73 +96,54 @@ export declare function filterInBatches<T>(items: T[], predicate: (item: T, inde
99
96
  * @template K - The type of the grouping key (must extend PropertyKey)
100
97
  * @param items - The array to group
101
98
  * @param keyFn - The function to extract the grouping key from each item. Receives the item and its index.
102
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
103
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
99
+ * @param options - Batch processing options
104
100
  * @returns A promise that resolves to an object with grouped items
105
101
  * @throws {Error} When batchSize is not a positive integer
106
102
  *
107
103
  * @example
108
104
  * ```typescript
109
- * // Group by property
105
+ * // Group by property with default options
110
106
  * const byType = await groupByInBatches(
111
107
  * [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],
112
- * (item) => item.type,
113
- * 2
108
+ * (item) => item.type
114
109
  * );
115
110
  * // Result: {A: [{type: 'A', value: 1}, {type: 'A', value: 3}], B: [{type: 'B', value: 2}]}
116
111
  *
117
- * // Group by computed value
118
- * const byParity = await groupByInBatches([1, 2, 3, 4], (x) => x % 2 === 0 ? 'even' : 'odd');
119
- * // Result: {odd: [1, 3], even: [2, 4]}
120
- *
121
- * // Asynchronous key function
122
- * const byCategory = await groupByInBatches(items, async (item) => {
123
- * return await getCategoryForItem(item);
124
- * });
112
+ * // With custom batch size
113
+ * const byType = await groupByInBatches(
114
+ * [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],
115
+ * (item) => item.type,
116
+ * { batchSize: 2 }
117
+ * );
125
118
  * ```
126
119
  */
127
- export declare function groupByInBatches<T, K extends PropertyKey>(items: T[], keyFn: (item: T, index: number) => K | Promise<K>, batchSize?: number, yieldThreshold?: number): Promise<Record<K, T[]>>;
120
+ export declare function groupByInBatches<T, K extends PropertyKey>(items: T[], keyFn: (item: T, index: number) => K | Promise<K>, options?: BatchProcessingOptions): Promise<Record<K, T[]>>;
128
121
  /**
129
122
  * Reduces an array in batches to avoid blocking the event loop.
130
123
  * Processes items in chunks and yields control back to the event loop between batches
131
- * when the processing time exceeds the threshold.
124
+ * when the processing time exceeds the threshold. Sequential processing is always used for the reducer function,
125
+ * irrespective of the `sequentialProcessing` option.
132
126
  *
133
127
  * @template T - The type of items in the array
134
128
  * @template R - The type of the accumulator/result
135
129
  * @param items - The array to reduce
136
130
  * @param reducer - The reducer function. Receives the accumulator, current item, and index.
137
131
  * @param initialValue - The initial value for the accumulator
138
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
139
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
132
+ * @param options - Batch processing options
140
133
  * @returns A promise that resolves to the reduced value
141
134
  * @throws {Error} When batchSize is not a positive integer
142
135
  *
143
136
  * @example
144
137
  * ```typescript
145
- * // Sum numbers
146
- * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, 2);
138
+ * // Sum numbers with default options
139
+ * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0);
147
140
  * // Result: 10
148
141
  *
149
- * // Concatenate strings
150
- * const joined = await reduceInBatches(['a', 'b', 'c'], (acc, x) => acc + x, '');
151
- * // Result: 'abc'
152
- *
153
- * // Build object with index
154
- * const indexed = await reduceInBatches(
155
- * ['x', 'y'],
156
- * (acc, item, index) => ({ ...acc, [index]: item }),
157
- * {}
158
- * );
159
- * // Result: {0: 'x', 1: 'y'}
160
- *
161
- * // Asynchronous reducer
162
- * const processed = await reduceInBatches(urls, async (acc, url) => {
163
- * const data = await fetch(url).then(r => r.json());
164
- * return [...acc, data];
165
- * }, []);
142
+ * // With custom batch size
143
+ * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, { batchSize: 2 });
166
144
  * ```
167
145
  */
168
- export declare function reduceInBatches<T, R>(items: T[], reducer: (acc: R, item: T, index: number) => R | Promise<R>, initialValue: R, batchSize?: number, yieldThreshold?: number): Promise<R>;
146
+ export declare function reduceInBatches<T, R>(items: T[], reducer: (acc: R, item: T, index: number) => R | Promise<R>, initialValue: R, options?: BatchProcessingOptions): Promise<R>;
169
147
  /**
170
148
  * FlatMaps an array in batches to avoid blocking the event loop.
171
149
  * Processes items in chunks, flattens the results, and yields control back to the event loop between batches
@@ -175,31 +153,45 @@ export declare function reduceInBatches<T, R>(items: T[], reducer: (acc: R, item
175
153
  * @template R - The type of items in the flattened result array
176
154
  * @param items - The array to flatMap over
177
155
  * @param mapFn - The mapping function that returns an array for each item. Receives the item and its index.
178
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
179
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
156
+ * @param options - Batch processing options
180
157
  * @returns A promise that resolves to the flattened mapped array
181
158
  * @throws {Error} When batchSize is not a positive integer
182
159
  *
183
160
  * @example
184
161
  * ```typescript
185
- * // Duplicate each item
186
- * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x], 2);
162
+ * // Duplicate each item with default options
163
+ * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x]);
187
164
  * // Result: [1, 1, 2, 2, 3, 3]
188
165
  *
189
- * // Variable length results
190
- * const repeated = await flatMapInBatches([1, 2, 3], (x) => Array(x).fill(x));
191
- * // Result: [1, 2, 2, 3, 3, 3]
166
+ * // With custom batch size
167
+ * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x], { batchSize: 2 });
168
+ * ```
169
+ */
170
+ export declare function flatMapInBatches<T, R>(items: T[], mapFn: (item: T, index: number) => R[] | Promise<R[]>, options?: BatchProcessingOptions): Promise<R[]>;
171
+ /**
172
+ * forEach over an array in batches to avoid blocking the event loop.
173
+ * Processes items in chunks and yields control back to the event loop between batches
174
+ * when the processing time exceeds the threshold.
192
175
  *
193
- * // Split strings into characters
194
- * const chars = await flatMapInBatches(['ab', 'cd'], (str) => str.split(''));
195
- * // Result: ['a', 'b', 'c', 'd']
176
+ * @template T - The type of items in the input array
177
+ * @param items - The array to iterate over
178
+ * @param fn - The function to apply to each item. Receives the item and its index. Can be async.
179
+ * @param options - Batch processing options
180
+ * @returns A promise that resolves when all items have been processed
181
+ * @throws {Error} When batchSize is not a positive integer
196
182
  *
197
- * // Asynchronous mapping
198
- * const expanded = await flatMapInBatches(categories, async (category) => {
199
- * const items = await fetchItemsForCategory(category);
200
- * return items;
183
+ * @example
184
+ * ```typescript
185
+ * // Process items in batches with default options
186
+ * await forEachInBatches([1, 2, 3, 4], async (x) => {
187
+ * await doSomething(x);
201
188
  * });
189
+ *
190
+ * // With custom batch size
191
+ * await forEachInBatches([1, 2, 3, 4], async (x) => {
192
+ * await doSomething(x);
193
+ * }, { batchSize: 2 });
202
194
  * ```
203
195
  */
204
- export declare function flatMapInBatches<T, R>(items: T[], mapFn: (item: T, index: number) => R[] | Promise<R[]>, batchSize?: number, yieldThreshold?: number): Promise<R[]>;
196
+ export declare function forEachInBatches<T>(items: T[], fn: (item: T, index: number) => void | Promise<void>, options?: BatchProcessingOptions): Promise<void>;
205
197
  //# sourceMappingURL=batch-processing.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"batch-processing.d.ts","sourceRoot":"","sources":["../../src/utils/batch-processing.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,MAAM,WAAW,uBAAuB;IACtC;;OAEG;IACH,SAAS,CAAC,EAAE,MAAM,CAAC;IAEnB;;OAEG;IACH,cAAc,CAAC,EAAE,MAAM,CAAC;CACzB;AAMD;;;;;;;;;;;;;;;;;GAiBG;AACH,wBAAgB,gCAAgC,CAC9C,MAAM,CAAC,EAAE,uBAAuB,GAC/B,uBAAuB,CAqBzB;AA0BD;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AACH,wBAAgB,YAAY,CAAC,CAAC,EAAE,CAAC,EAC/B,KAAK,EAAE,CAAC,EAAE,EACV,KAAK,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,EACjD,SAAS,SAAmB,EAC5B,cAAc,SAAwB,GACrC,OAAO,CAAC,CAAC,EAAE,CAAC,CAmBd;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BG;AACH,wBAAgB,eAAe,CAAC,CAAC,EAC/B,KAAK,EAAE,CAAC,EAAE,EACV,SAAS,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,EACjE,SAAS,SAAmB,EAC5B,cAAc,SAAwB,GACrC,OAAO,CAAC,CAAC,EAAE,CAAC,CAqBd;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiCG;AACH,wBAAgB,gBAAgB,CAAC,CAAC,EAAE,CAAC,SAAS,WAAW,EACvD,KAAK,EAAE,CAAC,EAAE,EACV,KAAK,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,EACjD,SAAS,SAAmB,EAC5B,cAAc,SAAwB,GACrC,OAAO,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAiCzB;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuCG;AACH,wBAAgB,eAAe,CAAC,CAAC,EAAE,CAAC,EAClC,KAAK,EAAE,CAAC,EAAE,EACV,OAAO,EAAE,CAAC,GAAG,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,EAC3D,YAAY,EAAE,CAAC,EACf,SAAS,SAAmB,EAC5B,cAAc,SAAwB,GACrC,OAAO,CAAC,CAAC,CAAC,CAsBZ;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkCG;AACH,wBAAgB,gBAAgB,CAAC,CAAC,EAAE,CAAC,EACnC,KAAK,EAAE,CAAC,EAAE,EACV,KAAK,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,CAAC,EAAE,GAAG,OAAO,CAAC,CAAC,EAAE,CAAC,EACrD,SAAS,SAAmB,EAC5B,cAAc,SAAwB,GACrC,OAAO,CAAC,CAAC,EAAE,CAAC,CAoBd"}
1
+ {"version":3,"file":"batch-processing.d.ts","sourceRoot":"","sources":["../../src/utils/batch-processing.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,MAAM,WAAW,sBAAsB;IACrC;;;OAGG;IACH,SAAS,CAAC,EAAE,MAAM,CAAC;IAEnB;;;OAGG;IACH,cAAc,CAAC,EAAE,MAAM,CAAC;IAExB;;;;OAIG;IACH,oBAAoB,CAAC,EAAE,OAAO,CAAC;CAChC;AAOD;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,wBAAgB,gCAAgC,CAC9C,MAAM,CAAC,EAAE,sBAAsB,GAC9B,sBAAsB,CA0BxB;AAoFD;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,wBAAgB,YAAY,CAAC,CAAC,EAAE,CAAC,EAC/B,KAAK,EAAE,CAAC,EAAE,EACV,KAAK,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,EACjD,OAAO,CAAC,EAAE,sBAAsB,GAC/B,OAAO,CAAC,CAAC,EAAE,CAAC,CAqBd;AAED;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,wBAAgB,eAAe,CAAC,CAAC,EAC/B,KAAK,EAAE,CAAC,EAAE,EACV,SAAS,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,EACjE,OAAO,CAAC,EAAE,sBAAsB,GAC/B,OAAO,CAAC,CAAC,EAAE,CAAC,CAuBd;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BG;AACH,wBAAgB,gBAAgB,CAAC,CAAC,EAAE,CAAC,SAAS,WAAW,EACvD,KAAK,EAAE,CAAC,EAAE,EACV,KAAK,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,EACjD,OAAO,CAAC,EAAE,sBAAsB,GAC/B,OAAO,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CA0BzB;AAED;;;;;;;;;;;;;;;;;;;;;;;;GAwBG;AACH,wBAAgB,eAAe,CAAC,CAAC,EAAE,CAAC,EAClC,KAAK,EAAE,CAAC,EAAE,EACV,OAAO,EAAE,CAAC,GAAG,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,EAC3D,YAAY,EAAE,CAAC,EACf,OAAO,CAAC,EAAE,sBAAsB,GAC/B,OAAO,CAAC,CAAC,CAAC,CAcZ;AAED;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,wBAAgB,gBAAgB,CAAC,CAAC,EAAE,CAAC,EACnC,KAAK,EAAE,CAAC,EAAE,EACV,KAAK,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,CAAC,EAAE,GAAG,OAAO,CAAC,CAAC,EAAE,CAAC,EACrD,OAAO,CAAC,EAAE,sBAAsB,GAC/B,OAAO,CAAC,CAAC,EAAE,CAAC,CAuBd;AAED;;;;;;;;;;;;;;;;;;;;;;;;GAwBG;AACH,wBAAgB,gBAAgB,CAAC,CAAC,EAChC,KAAK,EAAE,CAAC,EAAE,EACV,EAAE,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,EACpD,OAAO,CAAC,EAAE,sBAAsB,GAC/B,OAAO,CAAC,IAAI,CAAC,CAef"}
@@ -6,9 +6,11 @@ exports.filterInBatches = filterInBatches;
6
6
  exports.groupByInBatches = groupByInBatches;
7
7
  exports.reduceInBatches = reduceInBatches;
8
8
  exports.flatMapInBatches = flatMapInBatches;
9
+ exports.forEachInBatches = forEachInBatches;
9
10
  // Default configuration values (internal)
10
11
  let defaultBatchSize = 10;
11
12
  let defaultYieldThreshold = 10;
13
+ let defaultSequentialProcessing = true;
12
14
  /**
13
15
  * Configure global defaults for batch processing operations
14
16
  *
@@ -23,6 +25,9 @@ let defaultYieldThreshold = 10;
23
25
  * // Set only batch size
24
26
  * configureBatchProcessingDefaults({ batchSize: 50 });
25
27
  *
28
+ * // Enable concurrent processing within batches
29
+ * configureBatchProcessingDefaults({ sequentialProcessing: false });
30
+ *
26
31
  * // Get current configuration
27
32
  * const currentConfig = configureBatchProcessingDefaults();
28
33
  * ```
@@ -41,10 +46,14 @@ function configureBatchProcessingDefaults(config) {
41
46
  }
42
47
  defaultYieldThreshold = config.yieldThreshold;
43
48
  }
49
+ if (config.sequentialProcessing !== undefined) {
50
+ defaultSequentialProcessing = Boolean(config.sequentialProcessing);
51
+ }
44
52
  }
45
53
  return {
46
54
  batchSize: defaultBatchSize,
47
55
  yieldThreshold: defaultYieldThreshold,
56
+ sequentialProcessing: defaultSequentialProcessing,
48
57
  };
49
58
  }
50
59
  /**
@@ -68,6 +77,57 @@ function defer(startTime, yieldThreshold) {
68
77
  // Otherwise continue immediately
69
78
  return Promise.resolve(false);
70
79
  }
80
+ /**
81
+ * Helper to get batch processing options with defaults applied
82
+ * @param options - User provided options
83
+ * @returns Options with defaults applied
84
+ * @internal
85
+ */
86
+ function getOptions(options) {
87
+ const batchSize = options?.batchSize ?? defaultBatchSize;
88
+ const yieldThreshold = options?.yieldThreshold ?? defaultYieldThreshold;
89
+ const sequentialProcessing = options?.sequentialProcessing ?? defaultSequentialProcessing;
90
+ if (!Number.isInteger(batchSize) || batchSize <= 0) {
91
+ throw new Error('batchSize must be a positive integer');
92
+ }
93
+ if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {
94
+ throw new Error('yieldThreshold must be a non-negative integer');
95
+ }
96
+ return {
97
+ batchSize,
98
+ yieldThreshold,
99
+ sequentialProcessing,
100
+ };
101
+ }
102
+ /**
103
+ * Helper to process an array in batches, yielding as needed.
104
+ * @param items - The array to process
105
+ * @param options - Batch processing options
106
+ * @param batchHandler - Function to handle each batch: (batch, batchStartIndex) => Promise<any>
107
+ * @returns Promise<void>
108
+ */
109
+ async function processInBatches(items, options, batchHandler) {
110
+ let i = 0;
111
+ let startTime = Date.now();
112
+ const { batchSize } = options;
113
+ const n = items.length;
114
+ // Allocate a single batch array and reuses it for all batches
115
+ const batch = Array(batchSize);
116
+ while (i < n) {
117
+ const len = Math.min(batchSize, n - i);
118
+ for (let j = 0; j < len; j += 1) {
119
+ batch[j] = items[i + j];
120
+ }
121
+ batch.length = len;
122
+ // eslint-disable-next-line no-await-in-loop
123
+ await batchHandler(batch, i);
124
+ i += batchSize;
125
+ // eslint-disable-next-line no-await-in-loop
126
+ const didYield = await defer(startTime, options.yieldThreshold);
127
+ if (didYield)
128
+ startTime = Date.now();
129
+ }
130
+ }
71
131
  /**
72
132
  * Maps over an array in batches to avoid blocking the event loop.
73
133
  * Processes items in chunks and yields control back to the event loop between batches
@@ -77,40 +137,40 @@ function defer(startTime, yieldThreshold) {
77
137
  * @template R - The type of items in the result array
78
138
  * @param items - The array to map over
79
139
  * @param mapFn - The mapping function to apply to each item. Receives the item and its index.
80
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
81
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
140
+ * @param options - Batch processing options
82
141
  * @returns A promise that resolves to the mapped array
83
142
  * @throws {Error} When batchSize is not a positive integer
84
143
  *
85
144
  * @example
86
145
  * ```typescript
87
- * // Synchronous mapping
88
- * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2, 2);
89
- * // Result: [2, 4, 6, 8]
90
- *
91
- * // Asynchronous mapping
92
- * const fetched = await mapInBatches(urls, async (url) => fetch(url), 3);
146
+ * // Synchronous mapping with default options (sequential processing)
147
+ * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2);
93
148
  *
94
- * // With index
95
- * const indexed = await mapInBatches(['a', 'b'], (item, index) => `${item}-${index}`);
96
- * // Result: ['a-0', 'b-1']
149
+ * // With concurrent processing within batches
150
+ * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2, { sequentialProcessing: false });
97
151
  * ```
98
152
  */
99
- function mapInBatches(items, mapFn, batchSize = defaultBatchSize, yieldThreshold = defaultYieldThreshold) {
100
- if (!Number.isInteger(batchSize) || batchSize <= 0) {
101
- throw new Error('batchSize must be a positive integer');
102
- }
103
- if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {
104
- throw new Error('yieldThreshold must be a non-negative integer');
105
- }
106
- const process = async (i = 0, acc = [], startTime = Date.now()) => {
107
- if (i >= items.length)
108
- return acc;
109
- const batch = items.slice(i, i + batchSize);
110
- const mapped = await Promise.all(batch.map((item, j) => mapFn(item, i + j)));
111
- return defer(startTime, yieldThreshold).then((didYield) => process(i + batchSize, acc.concat(mapped), didYield ? Date.now() : startTime));
112
- };
113
- return process();
153
+ function mapInBatches(items, mapFn, options) {
154
+ const opts = getOptions(options);
155
+ return (async () => {
156
+ const result = [];
157
+ await processInBatches(items, opts, async (batch, batchStart) => {
158
+ if (opts.sequentialProcessing) {
159
+ // Process items sequentially
160
+ for (let j = 0; j < batch.length; j += 1) {
161
+ // eslint-disable-next-line no-await-in-loop -- sequential processing is required
162
+ const mapped = await mapFn(batch[j], batchStart + j);
163
+ result.push(mapped);
164
+ }
165
+ }
166
+ else {
167
+ // Process items concurrently
168
+ const mapped = await Promise.all(batch.map((item, j) => mapFn(item, batchStart + j)));
169
+ result.push(...mapped);
170
+ }
171
+ });
172
+ return result;
173
+ })();
114
174
  }
115
175
  /**
116
176
  * Filters an array in batches to avoid blocking the event loop.
@@ -120,44 +180,45 @@ function mapInBatches(items, mapFn, batchSize = defaultBatchSize, yieldThreshold
120
180
  * @template T - The type of items in the array
121
181
  * @param items - The array to filter
122
182
  * @param predicate - The predicate function to test each item. Receives the item and its index.
123
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
124
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
183
+ * @param options - Batch processing options
125
184
  * @returns A promise that resolves to the filtered array
126
185
  * @throws {Error} When batchSize is not a positive integer
127
186
  *
128
187
  * @example
129
188
  * ```typescript
130
- * // Synchronous filtering
131
- * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, 2);
189
+ * // Synchronous filtering with default options
190
+ * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0);
132
191
  * // Result: [2, 4]
133
192
  *
134
- * // Asynchronous filtering
135
- * const valid = await filterInBatches(urls, async (url) => {
136
- * const response = await fetch(url);
137
- * return response.ok;
138
- * }, 3);
139
- *
140
- * // With index
141
- * const evenIndices = await filterInBatches(['a', 'b', 'c'], (_, index) => index % 2 === 0);
142
- * // Result: ['a', 'c']
193
+ * // With custom batch size
194
+ * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, { batchSize: 2 });
143
195
  * ```
144
196
  */
145
- function filterInBatches(items, predicate, batchSize = defaultBatchSize, yieldThreshold = defaultYieldThreshold) {
146
- if (!Number.isInteger(batchSize) || batchSize <= 0) {
147
- throw new Error('batchSize must be a positive integer');
148
- }
149
- if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {
150
- throw new Error('yieldThreshold must be a non-negative integer');
151
- }
152
- const process = async (i = 0, acc = [], startTime = Date.now()) => {
153
- if (i >= items.length)
154
- return acc;
155
- const batch = items.slice(i, i + batchSize);
156
- const flags = await Promise.all(batch.map((item, j) => predicate(item, i + j)));
157
- const filtered = batch.filter((_, j) => flags[j]);
158
- return defer(startTime, yieldThreshold).then((didYield) => process(i + batchSize, acc.concat(filtered), didYield ? Date.now() : startTime));
159
- };
160
- return process();
197
+ function filterInBatches(items, predicate, options) {
198
+ const opts = getOptions(options);
199
+ return (async () => {
200
+ const result = [];
201
+ await processInBatches(items, opts, async (batch, batchStart) => {
202
+ if (opts.sequentialProcessing) {
203
+ // Process items sequentially
204
+ for (let j = 0; j < batch.length; j += 1) {
205
+ // eslint-disable-next-line no-await-in-loop -- sequential processing is required
206
+ const passes = await predicate(batch[j], batchStart + j);
207
+ if (passes)
208
+ result.push(batch[j]);
209
+ }
210
+ }
211
+ else {
212
+ // Process items concurrently
213
+ const flags = await Promise.all(batch.map((item, j) => predicate(item, batchStart + j)));
214
+ for (let j = 0; j < batch.length; j += 1) {
215
+ if (flags[j])
216
+ result.push(batch[j]);
217
+ }
218
+ }
219
+ });
220
+ return result;
221
+ })();
161
222
  }
162
223
  /**
163
224
  * Groups an array by a key function in batches to avoid blocking the event loop.
@@ -168,112 +229,94 @@ function filterInBatches(items, predicate, batchSize = defaultBatchSize, yieldTh
168
229
  * @template K - The type of the grouping key (must extend PropertyKey)
169
230
  * @param items - The array to group
170
231
  * @param keyFn - The function to extract the grouping key from each item. Receives the item and its index.
171
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
172
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
232
+ * @param options - Batch processing options
173
233
  * @returns A promise that resolves to an object with grouped items
174
234
  * @throws {Error} When batchSize is not a positive integer
175
235
  *
176
236
  * @example
177
237
  * ```typescript
178
- * // Group by property
238
+ * // Group by property with default options
179
239
  * const byType = await groupByInBatches(
180
240
  * [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],
181
- * (item) => item.type,
182
- * 2
241
+ * (item) => item.type
183
242
  * );
184
243
  * // Result: {A: [{type: 'A', value: 1}, {type: 'A', value: 3}], B: [{type: 'B', value: 2}]}
185
244
  *
186
- * // Group by computed value
187
- * const byParity = await groupByInBatches([1, 2, 3, 4], (x) => x % 2 === 0 ? 'even' : 'odd');
188
- * // Result: {odd: [1, 3], even: [2, 4]}
189
- *
190
- * // Asynchronous key function
191
- * const byCategory = await groupByInBatches(items, async (item) => {
192
- * return await getCategoryForItem(item);
193
- * });
245
+ * // With custom batch size
246
+ * const byType = await groupByInBatches(
247
+ * [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],
248
+ * (item) => item.type,
249
+ * { batchSize: 2 }
250
+ * );
194
251
  * ```
195
252
  */
196
- function groupByInBatches(items, keyFn, batchSize = defaultBatchSize, yieldThreshold = defaultYieldThreshold) {
197
- if (!Number.isInteger(batchSize) || batchSize <= 0) {
198
- throw new Error('batchSize must be a positive integer');
199
- }
200
- if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {
201
- throw new Error('yieldThreshold must be a non-negative integer');
202
- }
203
- const process = async (i = 0, acc = {}, startTime = Date.now()) => {
204
- if (i >= items.length)
205
- return acc;
206
- const batch = items.slice(i, i + batchSize);
207
- const keys = await Promise.all(batch.map((item, j) => keyFn(item, i + j)));
208
- const updated = keys.reduce((res, key, j) => {
209
- const item = batch[j];
210
- if (!res[key])
211
- res[key] = [];
212
- res[key] = res[key].concat(item);
213
- return res;
214
- }, { ...acc });
215
- return defer(startTime, yieldThreshold).then((didYield) => process(i + batchSize, updated, didYield ? Date.now() : startTime));
216
- };
217
- return process();
253
+ function groupByInBatches(items, keyFn, options) {
254
+ const opts = getOptions(options);
255
+ return (async () => {
256
+ const result = {};
257
+ await processInBatches(items, opts, async (batch, batchStart) => {
258
+ if (opts.sequentialProcessing) {
259
+ // Process items sequentially
260
+ for (let j = 0; j < batch.length; j += 1) {
261
+ // eslint-disable-next-line no-await-in-loop -- sequential processing is required
262
+ const key = await keyFn(batch[j], batchStart + j);
263
+ if (!result[key])
264
+ result[key] = [];
265
+ result[key].push(batch[j]);
266
+ }
267
+ }
268
+ else {
269
+ // Process items concurrently
270
+ const keys = await Promise.all(batch.map((item, j) => keyFn(item, batchStart + j)));
271
+ for (let j = 0; j < batch.length; j += 1) {
272
+ const key = keys[j];
273
+ if (!result[key])
274
+ result[key] = [];
275
+ result[key].push(batch[j]);
276
+ }
277
+ }
278
+ });
279
+ return result;
280
+ })();
218
281
  }
219
282
  /**
220
283
  * Reduces an array in batches to avoid blocking the event loop.
221
284
  * Processes items in chunks and yields control back to the event loop between batches
222
- * when the processing time exceeds the threshold.
285
+ * when the processing time exceeds the threshold. Sequential processing is always used for the reducer function,
286
+ * irrespective of the `sequentialProcessing` option.
223
287
  *
224
288
  * @template T - The type of items in the array
225
289
  * @template R - The type of the accumulator/result
226
290
  * @param items - The array to reduce
227
291
  * @param reducer - The reducer function. Receives the accumulator, current item, and index.
228
292
  * @param initialValue - The initial value for the accumulator
229
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
230
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
293
+ * @param options - Batch processing options
231
294
  * @returns A promise that resolves to the reduced value
232
295
  * @throws {Error} When batchSize is not a positive integer
233
296
  *
234
297
  * @example
235
298
  * ```typescript
236
- * // Sum numbers
237
- * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, 2);
299
+ * // Sum numbers with default options
300
+ * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0);
238
301
  * // Result: 10
239
302
  *
240
- * // Concatenate strings
241
- * const joined = await reduceInBatches(['a', 'b', 'c'], (acc, x) => acc + x, '');
242
- * // Result: 'abc'
243
- *
244
- * // Build object with index
245
- * const indexed = await reduceInBatches(
246
- * ['x', 'y'],
247
- * (acc, item, index) => ({ ...acc, [index]: item }),
248
- * {}
249
- * );
250
- * // Result: {0: 'x', 1: 'y'}
251
- *
252
- * // Asynchronous reducer
253
- * const processed = await reduceInBatches(urls, async (acc, url) => {
254
- * const data = await fetch(url).then(r => r.json());
255
- * return [...acc, data];
256
- * }, []);
303
+ * // With custom batch size
304
+ * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, { batchSize: 2 });
257
305
  * ```
258
306
  */
259
- function reduceInBatches(items, reducer, initialValue, batchSize = defaultBatchSize, yieldThreshold = defaultYieldThreshold) {
260
- if (!Number.isInteger(batchSize) || batchSize <= 0) {
261
- throw new Error('batchSize must be a positive integer');
262
- }
263
- if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {
264
- throw new Error('yieldThreshold must be a non-negative integer');
265
- }
266
- const process = async (i = 0, acc = initialValue, startTime = Date.now()) => {
267
- if (i >= items.length)
268
- return acc;
269
- const batch = items.slice(i, i + batchSize);
270
- const reduced = await batch.reduce(async (prevAcc, item, j) => {
271
- const resolvedAcc = await prevAcc;
272
- return reducer(resolvedAcc, item, i + j);
273
- }, Promise.resolve(acc));
274
- return defer(startTime, yieldThreshold).then((didYield) => process(i + batchSize, reduced, didYield ? Date.now() : startTime));
275
- };
276
- return process();
307
+ function reduceInBatches(items, reducer, initialValue, options) {
308
+ const opts = getOptions(options);
309
+ return (async () => {
310
+ let acc = initialValue;
311
+ await processInBatches(items, opts, async (batch, batchStart) => {
312
+ // Always sequential, regardless of sequentialProcessing option
313
+ for (let j = 0; j < batch.length; j += 1) {
314
+ // eslint-disable-next-line no-await-in-loop -- sequential processing is required
315
+ acc = await reducer(acc, batch[j], batchStart + j);
316
+ }
317
+ });
318
+ return acc;
319
+ })();
277
320
  }
278
321
  /**
279
322
  * FlatMaps an array in batches to avoid blocking the event loop.
@@ -284,47 +327,83 @@ function reduceInBatches(items, reducer, initialValue, batchSize = defaultBatchS
284
327
  * @template R - The type of items in the flattened result array
285
328
  * @param items - The array to flatMap over
286
329
  * @param mapFn - The mapping function that returns an array for each item. Receives the item and its index.
287
- * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)
288
- * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)
330
+ * @param options - Batch processing options
289
331
  * @returns A promise that resolves to the flattened mapped array
290
332
  * @throws {Error} When batchSize is not a positive integer
291
333
  *
292
334
  * @example
293
335
  * ```typescript
294
- * // Duplicate each item
295
- * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x], 2);
336
+ * // Duplicate each item with default options
337
+ * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x]);
296
338
  * // Result: [1, 1, 2, 2, 3, 3]
297
339
  *
298
- * // Variable length results
299
- * const repeated = await flatMapInBatches([1, 2, 3], (x) => Array(x).fill(x));
300
- * // Result: [1, 2, 2, 3, 3, 3]
340
+ * // With custom batch size
341
+ * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x], { batchSize: 2 });
342
+ * ```
343
+ */
344
+ function flatMapInBatches(items, mapFn, options) {
345
+ const opts = getOptions(options);
346
+ return (async () => {
347
+ const result = [];
348
+ await processInBatches(items, opts, async (batch, batchStart) => {
349
+ if (opts.sequentialProcessing) {
350
+ // Process items sequentially
351
+ for (let j = 0; j < batch.length; j += 1) {
352
+ // eslint-disable-next-line no-await-in-loop -- sequential processing is required
353
+ const mapped = await mapFn(batch[j], batchStart + j);
354
+ result.push(...mapped);
355
+ }
356
+ }
357
+ else {
358
+ // Process items concurrently
359
+ const mapped = await Promise.all(batch.map((item, j) => mapFn(item, batchStart + j)));
360
+ mapped.forEach((arr) => {
361
+ result.push(...arr);
362
+ });
363
+ }
364
+ });
365
+ return result;
366
+ })();
367
+ }
368
+ /**
369
+ * forEach over an array in batches to avoid blocking the event loop.
370
+ * Processes items in chunks and yields control back to the event loop between batches
371
+ * when the processing time exceeds the threshold.
301
372
  *
302
- * // Split strings into characters
303
- * const chars = await flatMapInBatches(['ab', 'cd'], (str) => str.split(''));
304
- * // Result: ['a', 'b', 'c', 'd']
373
+ * @template T - The type of items in the input array
374
+ * @param items - The array to iterate over
375
+ * @param fn - The function to apply to each item. Receives the item and its index. Can be async.
376
+ * @param options - Batch processing options
377
+ * @returns A promise that resolves when all items have been processed
378
+ * @throws {Error} When batchSize is not a positive integer
305
379
  *
306
- * // Asynchronous mapping
307
- * const expanded = await flatMapInBatches(categories, async (category) => {
308
- * const items = await fetchItemsForCategory(category);
309
- * return items;
380
+ * @example
381
+ * ```typescript
382
+ * // Process items in batches with default options
383
+ * await forEachInBatches([1, 2, 3, 4], async (x) => {
384
+ * await doSomething(x);
310
385
  * });
386
+ *
387
+ * // With custom batch size
388
+ * await forEachInBatches([1, 2, 3, 4], async (x) => {
389
+ * await doSomething(x);
390
+ * }, { batchSize: 2 });
311
391
  * ```
312
392
  */
313
- function flatMapInBatches(items, mapFn, batchSize = defaultBatchSize, yieldThreshold = defaultYieldThreshold) {
314
- if (!Number.isInteger(batchSize) || batchSize <= 0) {
315
- throw new Error('batchSize must be a positive integer');
316
- }
317
- if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {
318
- throw new Error('yieldThreshold must be a non-negative integer');
319
- }
320
- const process = async (i = 0, acc = [], startTime = Date.now()) => {
321
- if (i >= items.length)
322
- return acc;
323
- const batch = items.slice(i, i + batchSize);
324
- const mapped = await Promise.all(batch.map((item, j) => mapFn(item, i + j)));
325
- const flattened = mapped.reduce((all, arr) => all.concat(arr), []);
326
- return defer(startTime, yieldThreshold).then((didYield) => process(i + batchSize, acc.concat(flattened), didYield ? Date.now() : startTime));
327
- };
328
- return process();
393
+ function forEachInBatches(items, fn, options) {
394
+ const opts = getOptions(options);
395
+ return processInBatches(items, opts, async (batch, batchStart) => {
396
+ if (opts.sequentialProcessing) {
397
+ // Process items sequentially
398
+ for (let j = 0; j < batch.length; j += 1) {
399
+ // eslint-disable-next-line no-await-in-loop -- sequential processing is required
400
+ await fn(batch[j], batchStart + j);
401
+ }
402
+ }
403
+ else {
404
+ // Process items concurrently
405
+ await Promise.all(batch.map((item, j) => fn(item, batchStart + j)));
406
+ }
407
+ });
329
408
  }
330
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"batch-processing.js","sourceRoot":"","sources":["../../src/utils/batch-processing.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Configuration options for batch processing operations\n */\nexport interface BatchProcessingDefaults {\n  /**\n   * Default number of items to process in each batch\n   */\n  batchSize?: number;\n\n  /**\n   * Default time threshold in milliseconds before yielding control back to the event loop\n   */\n  yieldThreshold?: number;\n}\n\n// Default configuration values (internal)\nlet defaultBatchSize = 10;\nlet defaultYieldThreshold = 10;\n\n/**\n * Configure global defaults for batch processing operations\n *\n * @param config - Configuration options for batch processing\n * @returns The current configuration after applying changes\n *\n * @example\n * ```typescript\n * // Set both defaults\n * configureBatchProcessingDefaults({ batchSize: 20, yieldThreshold: 5 });\n *\n * // Set only batch size\n * configureBatchProcessingDefaults({ batchSize: 50 });\n *\n * // Get current configuration\n * const currentConfig = configureBatchProcessingDefaults();\n * ```\n */\nexport function configureBatchProcessingDefaults(\n  config?: BatchProcessingDefaults,\n): BatchProcessingDefaults {\n  if (config) {\n    if (config.batchSize !== undefined) {\n      if (!Number.isInteger(config.batchSize) || config.batchSize <= 0) {\n        throw new Error('batchSize must be a positive integer');\n      }\n      defaultBatchSize = config.batchSize;\n    }\n\n    if (config.yieldThreshold !== undefined) {\n      if (!Number.isInteger(config.yieldThreshold) || config.yieldThreshold < 0) {\n        throw new Error('yieldThreshold must be a non-negative integer');\n      }\n      defaultYieldThreshold = config.yieldThreshold;\n    }\n  }\n\n  return {\n    batchSize: defaultBatchSize,\n    yieldThreshold: defaultYieldThreshold,\n  };\n}\n\n/**\n * Utility function to defer execution to the next tick of the event loop.\n * This prevents blocking the event loop during heavy batch operations by\n * yielding control back to the event loop using setImmediate.\n *\n * @param startTime - The timestamp when the current batch processing started\n * @param yieldThreshold - Time threshold in milliseconds before yielding control\n * @returns A promise that resolves to a boolean indicating whether a yield occurred\n * @internal\n */\nfunction defer(startTime: number, yieldThreshold: number): Promise<boolean> {\n  const elapsed = Date.now() - startTime;\n\n  // Only yield if we've exceeded the threshold\n  if (elapsed >= yieldThreshold) {\n    return new Promise((resolve) => {\n      setImmediate(() => resolve(true));\n    });\n  }\n\n  // Otherwise continue immediately\n  return Promise.resolve(false);\n}\n\n/**\n * Maps over an array in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the input array\n * @template R - The type of items in the result array\n * @param items - The array to map over\n * @param mapFn - The mapping function to apply to each item. Receives the item and its index.\n * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)\n * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)\n * @returns A promise that resolves to the mapped array\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Synchronous mapping\n * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2, 2);\n * // Result: [2, 4, 6, 8]\n *\n * // Asynchronous mapping\n * const fetched = await mapInBatches(urls, async (url) => fetch(url), 3);\n *\n * // With index\n * const indexed = await mapInBatches(['a', 'b'], (item, index) => `${item}-${index}`);\n * // Result: ['a-0', 'b-1']\n * ```\n */\nexport function mapInBatches<T, R>(\n  items: T[],\n  mapFn: (item: T, index: number) => R | Promise<R>,\n  batchSize = defaultBatchSize,\n  yieldThreshold = defaultYieldThreshold,\n): Promise<R[]> {\n  if (!Number.isInteger(batchSize) || batchSize <= 0) {\n    throw new Error('batchSize must be a positive integer');\n  }\n  if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {\n    throw new Error('yieldThreshold must be a non-negative integer');\n  }\n  const process = async (i = 0, acc: R[] = [], startTime = Date.now()): Promise<R[]> => {\n    if (i >= items.length) return acc;\n\n    const batch = items.slice(i, i + batchSize);\n    const mapped = await Promise.all(batch.map((item, j) => mapFn(item, i + j)));\n\n    return defer(startTime, yieldThreshold).then((didYield) =>\n      process(i + batchSize, acc.concat(mapped), didYield ? Date.now() : startTime),\n    );\n  };\n\n  return process();\n}\n\n/**\n * Filters an array in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the array\n * @param items - The array to filter\n * @param predicate - The predicate function to test each item. Receives the item and its index.\n * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)\n * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)\n * @returns A promise that resolves to the filtered array\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Synchronous filtering\n * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, 2);\n * // Result: [2, 4]\n *\n * // Asynchronous filtering\n * const valid = await filterInBatches(urls, async (url) => {\n *   const response = await fetch(url);\n *   return response.ok;\n * }, 3);\n *\n * // With index\n * const evenIndices = await filterInBatches(['a', 'b', 'c'], (_, index) => index % 2 === 0);\n * // Result: ['a', 'c']\n * ```\n */\nexport function filterInBatches<T>(\n  items: T[],\n  predicate: (item: T, index: number) => boolean | Promise<boolean>,\n  batchSize = defaultBatchSize,\n  yieldThreshold = defaultYieldThreshold,\n): Promise<T[]> {\n  if (!Number.isInteger(batchSize) || batchSize <= 0) {\n    throw new Error('batchSize must be a positive integer');\n  }\n  if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {\n    throw new Error('yieldThreshold must be a non-negative integer');\n  }\n  const process = async (i = 0, acc: T[] = [], startTime = Date.now()): Promise<T[]> => {\n    if (i >= items.length) return acc;\n\n    const batch = items.slice(i, i + batchSize);\n    const flags = await Promise.all(batch.map((item, j) => predicate(item, i + j)));\n\n    const filtered = batch.filter((_, j) => flags[j]);\n\n    return defer(startTime, yieldThreshold).then((didYield) =>\n      process(i + batchSize, acc.concat(filtered), didYield ? Date.now() : startTime),\n    );\n  };\n\n  return process();\n}\n\n/**\n * Groups an array by a key function in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the array\n * @template K - The type of the grouping key (must extend PropertyKey)\n * @param items - The array to group\n * @param keyFn - The function to extract the grouping key from each item. Receives the item and its index.\n * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)\n * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)\n * @returns A promise that resolves to an object with grouped items\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Group by property\n * const byType = await groupByInBatches(\n *   [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],\n *   (item) => item.type,\n *   2\n * );\n * // Result: {A: [{type: 'A', value: 1}, {type: 'A', value: 3}], B: [{type: 'B', value: 2}]}\n *\n * // Group by computed value\n * const byParity = await groupByInBatches([1, 2, 3, 4], (x) => x % 2 === 0 ? 'even' : 'odd');\n * // Result: {odd: [1, 3], even: [2, 4]}\n *\n * // Asynchronous key function\n * const byCategory = await groupByInBatches(items, async (item) => {\n *   return await getCategoryForItem(item);\n * });\n * ```\n */\nexport function groupByInBatches<T, K extends PropertyKey>(\n  items: T[],\n  keyFn: (item: T, index: number) => K | Promise<K>,\n  batchSize = defaultBatchSize,\n  yieldThreshold = defaultYieldThreshold,\n): Promise<Record<K, T[]>> {\n  if (!Number.isInteger(batchSize) || batchSize <= 0) {\n    throw new Error('batchSize must be a positive integer');\n  }\n  if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {\n    throw new Error('yieldThreshold must be a non-negative integer');\n  }\n  const process = async (\n    i = 0,\n    acc: Record<K, T[]> = {} as Record<K, T[]>,\n    startTime = Date.now(),\n  ): Promise<Record<K, T[]>> => {\n    if (i >= items.length) return acc;\n\n    const batch = items.slice(i, i + batchSize);\n    const keys = await Promise.all(batch.map((item, j) => keyFn(item, i + j)));\n\n    const updated = keys.reduce(\n      (res, key, j) => {\n        const item = batch[j];\n        if (!res[key]) res[key] = [];\n        res[key] = res[key].concat(item);\n        return res;\n      },\n      { ...acc },\n    );\n\n    return defer(startTime, yieldThreshold).then((didYield) =>\n      process(i + batchSize, updated, didYield ? Date.now() : startTime),\n    );\n  };\n\n  return process();\n}\n\n/**\n * Reduces an array in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the array\n * @template R - The type of the accumulator/result\n * @param items - The array to reduce\n * @param reducer - The reducer function. Receives the accumulator, current item, and index.\n * @param initialValue - The initial value for the accumulator\n * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)\n * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)\n * @returns A promise that resolves to the reduced value\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Sum numbers\n * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, 2);\n * // Result: 10\n *\n * // Concatenate strings\n * const joined = await reduceInBatches(['a', 'b', 'c'], (acc, x) => acc + x, '');\n * // Result: 'abc'\n *\n * // Build object with index\n * const indexed = await reduceInBatches(\n *   ['x', 'y'],\n *   (acc, item, index) => ({ ...acc, [index]: item }),\n *   {}\n * );\n * // Result: {0: 'x', 1: 'y'}\n *\n * // Asynchronous reducer\n * const processed = await reduceInBatches(urls, async (acc, url) => {\n *   const data = await fetch(url).then(r => r.json());\n *   return [...acc, data];\n * }, []);\n * ```\n */\nexport function reduceInBatches<T, R>(\n  items: T[],\n  reducer: (acc: R, item: T, index: number) => R | Promise<R>,\n  initialValue: R,\n  batchSize = defaultBatchSize,\n  yieldThreshold = defaultYieldThreshold,\n): Promise<R> {\n  if (!Number.isInteger(batchSize) || batchSize <= 0) {\n    throw new Error('batchSize must be a positive integer');\n  }\n  if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {\n    throw new Error('yieldThreshold must be a non-negative integer');\n  }\n  const process = async (i = 0, acc: R = initialValue, startTime = Date.now()): Promise<R> => {\n    if (i >= items.length) return acc;\n\n    const batch = items.slice(i, i + batchSize);\n    const reduced = await batch.reduce<Promise<R>>(async (prevAcc, item, j) => {\n      const resolvedAcc = await prevAcc;\n      return reducer(resolvedAcc, item, i + j);\n    }, Promise.resolve(acc));\n\n    return defer(startTime, yieldThreshold).then((didYield) =>\n      process(i + batchSize, reduced, didYield ? Date.now() : startTime),\n    );\n  };\n\n  return process();\n}\n\n/**\n * FlatMaps an array in batches to avoid blocking the event loop.\n * Processes items in chunks, flattens the results, and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the input array\n * @template R - The type of items in the flattened result array\n * @param items - The array to flatMap over\n * @param mapFn - The mapping function that returns an array for each item. Receives the item and its index.\n * @param batchSize - The number of items to process in each batch (default: @see defaultBatchSize)\n * @param yieldThreshold - Time threshold in milliseconds before yielding control (default: @see defaultYieldThreshold)\n * @returns A promise that resolves to the flattened mapped array\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Duplicate each item\n * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x], 2);\n * // Result: [1, 1, 2, 2, 3, 3]\n *\n * // Variable length results\n * const repeated = await flatMapInBatches([1, 2, 3], (x) => Array(x).fill(x));\n * // Result: [1, 2, 2, 3, 3, 3]\n *\n * // Split strings into characters\n * const chars = await flatMapInBatches(['ab', 'cd'], (str) => str.split(''));\n * // Result: ['a', 'b', 'c', 'd']\n *\n * // Asynchronous mapping\n * const expanded = await flatMapInBatches(categories, async (category) => {\n *   const items = await fetchItemsForCategory(category);\n *   return items;\n * });\n * ```\n */\nexport function flatMapInBatches<T, R>(\n  items: T[],\n  mapFn: (item: T, index: number) => R[] | Promise<R[]>,\n  batchSize = defaultBatchSize,\n  yieldThreshold = defaultYieldThreshold,\n): Promise<R[]> {\n  if (!Number.isInteger(batchSize) || batchSize <= 0) {\n    throw new Error('batchSize must be a positive integer');\n  }\n  if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {\n    throw new Error('yieldThreshold must be a non-negative integer');\n  }\n  const process = async (i = 0, acc: R[] = [], startTime = Date.now()): Promise<R[]> => {\n    if (i >= items.length) return acc;\n\n    const batch = items.slice(i, i + batchSize);\n    const mapped = await Promise.all(batch.map((item, j) => mapFn(item, i + j)));\n    const flattened = mapped.reduce((all, arr) => all.concat(arr), []);\n\n    return defer(startTime, yieldThreshold).then((didYield) =>\n      process(i + batchSize, acc.concat(flattened), didYield ? Date.now() : startTime),\n    );\n  };\n\n  return process();\n}\n"]}
409
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"batch-processing.js","sourceRoot":"","sources":["../../src/utils/batch-processing.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Options for batch processing operations\n */\nexport interface BatchProcessingOptions {\n  /**\n   * Number of items to process in each batch (default: 10)\n   * Must be a positive integer.\n   */\n  batchSize?: number;\n\n  /**\n   * Time threshold in milliseconds (default: 10) before yielding control back to the event loop.\n   * Set to 0 to yield after every batch. Must be a non-negative integer.\n   */\n  yieldThreshold?: number;\n\n  /**\n   * Whether to process items sequentially within each batch. When true (default), each item in a batch\n   * will be processed one at a time. When false, all items in a batch will be processed concurrently.\n   * Consider the implications of concurrency on your processing logic before setting this to false, e.g. race conditions, rate limits, memory pressure, etc.\n   */\n  sequentialProcessing?: boolean;\n}\n\n// Default configuration values (internal)\nlet defaultBatchSize = 10;\nlet defaultYieldThreshold = 10;\nlet defaultSequentialProcessing = true;\n\n/**\n * Configure global defaults for batch processing operations\n *\n * @param config - Configuration options for batch processing\n * @returns The current configuration after applying changes\n *\n * @example\n * ```typescript\n * // Set both defaults\n * configureBatchProcessingDefaults({ batchSize: 20, yieldThreshold: 5 });\n *\n * // Set only batch size\n * configureBatchProcessingDefaults({ batchSize: 50 });\n *\n * // Enable concurrent processing within batches\n * configureBatchProcessingDefaults({ sequentialProcessing: false });\n *\n * // Get current configuration\n * const currentConfig = configureBatchProcessingDefaults();\n * ```\n */\nexport function configureBatchProcessingDefaults(\n  config?: BatchProcessingOptions,\n): BatchProcessingOptions {\n  if (config) {\n    if (config.batchSize !== undefined) {\n      if (!Number.isInteger(config.batchSize) || config.batchSize <= 0) {\n        throw new Error('batchSize must be a positive integer');\n      }\n      defaultBatchSize = config.batchSize;\n    }\n\n    if (config.yieldThreshold !== undefined) {\n      if (!Number.isInteger(config.yieldThreshold) || config.yieldThreshold < 0) {\n        throw new Error('yieldThreshold must be a non-negative integer');\n      }\n      defaultYieldThreshold = config.yieldThreshold;\n    }\n\n    if (config.sequentialProcessing !== undefined) {\n      defaultSequentialProcessing = Boolean(config.sequentialProcessing);\n    }\n  }\n\n  return {\n    batchSize: defaultBatchSize,\n    yieldThreshold: defaultYieldThreshold,\n    sequentialProcessing: defaultSequentialProcessing,\n  };\n}\n\n/**\n * Utility function to defer execution to the next tick of the event loop.\n * This prevents blocking the event loop during heavy batch operations by\n * yielding control back to the event loop using setImmediate.\n *\n * @param startTime - The timestamp when the current batch processing started\n * @param yieldThreshold - Time threshold in milliseconds before yielding control\n * @returns A promise that resolves to a boolean indicating whether a yield occurred\n * @internal\n */\nfunction defer(startTime: number, yieldThreshold: number): Promise<boolean> {\n  const elapsed = Date.now() - startTime;\n\n  // Only yield if we've exceeded the threshold\n  if (elapsed >= yieldThreshold) {\n    return new Promise((resolve) => {\n      setImmediate(() => resolve(true));\n    });\n  }\n\n  // Otherwise continue immediately\n  return Promise.resolve(false);\n}\n\n/**\n * Helper to get batch processing options with defaults applied\n * @param options - User provided options\n * @returns Options with defaults applied\n * @internal\n */\nfunction getOptions(options?: BatchProcessingOptions): Required<BatchProcessingOptions> {\n  const batchSize = options?.batchSize ?? defaultBatchSize;\n  const yieldThreshold = options?.yieldThreshold ?? defaultYieldThreshold;\n  const sequentialProcessing = options?.sequentialProcessing ?? defaultSequentialProcessing;\n\n  if (!Number.isInteger(batchSize) || batchSize <= 0) {\n    throw new Error('batchSize must be a positive integer');\n  }\n  if (!Number.isInteger(yieldThreshold) || yieldThreshold < 0) {\n    throw new Error('yieldThreshold must be a non-negative integer');\n  }\n\n  return {\n    batchSize,\n    yieldThreshold,\n    sequentialProcessing,\n  };\n}\n\n/**\n * Helper to process an array in batches, yielding as needed.\n * @param items - The array to process\n * @param options - Batch processing options\n * @param batchHandler - Function to handle each batch: (batch, batchStartIndex) => Promise<any>\n * @returns Promise<void>\n */\nasync function processInBatches<T>(\n  items: T[],\n  options: Required<BatchProcessingOptions>,\n  batchHandler: (batch: T[], batchStart: number) => Promise<void>,\n): Promise<void> {\n  let i = 0;\n  let startTime = Date.now();\n  const { batchSize } = options;\n  const n = items.length;\n  // Allocate a single batch array and reuses it for all batches\n  const batch: T[] = Array(batchSize);\n  while (i < n) {\n    const len = Math.min(batchSize, n - i);\n    for (let j = 0; j < len; j += 1) {\n      batch[j] = items[i + j];\n    }\n    batch.length = len;\n    // eslint-disable-next-line no-await-in-loop\n    await batchHandler(batch, i);\n    i += batchSize;\n    // eslint-disable-next-line no-await-in-loop\n    const didYield = await defer(startTime, options.yieldThreshold);\n    if (didYield) startTime = Date.now();\n  }\n}\n\n/**\n * Maps over an array in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the input array\n * @template R - The type of items in the result array\n * @param items - The array to map over\n * @param mapFn - The mapping function to apply to each item. Receives the item and its index.\n * @param options - Batch processing options\n * @returns A promise that resolves to the mapped array\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Synchronous mapping with default options (sequential processing)\n * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2);\n *\n * // With concurrent processing within batches\n * const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2, { sequentialProcessing: false });\n * ```\n */\nexport function mapInBatches<T, R>(\n  items: T[],\n  mapFn: (item: T, index: number) => R | Promise<R>,\n  options?: BatchProcessingOptions,\n): Promise<R[]> {\n  const opts = getOptions(options);\n\n  return (async () => {\n    const result: R[] = [];\n    await processInBatches(items, opts, async (batch, batchStart) => {\n      if (opts.sequentialProcessing) {\n        // Process items sequentially\n        for (let j = 0; j < batch.length; j += 1) {\n          // eslint-disable-next-line no-await-in-loop -- sequential processing is required\n          const mapped = await mapFn(batch[j], batchStart + j);\n          result.push(mapped);\n        }\n      } else {\n        // Process items concurrently\n        const mapped = await Promise.all(batch.map((item, j) => mapFn(item, batchStart + j)));\n        result.push(...mapped);\n      }\n    });\n    return result;\n  })();\n}\n\n/**\n * Filters an array in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the array\n * @param items - The array to filter\n * @param predicate - The predicate function to test each item. Receives the item and its index.\n * @param options - Batch processing options\n * @returns A promise that resolves to the filtered array\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Synchronous filtering with default options\n * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0);\n * // Result: [2, 4]\n *\n * // With custom batch size\n * const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, { batchSize: 2 });\n * ```\n */\nexport function filterInBatches<T>(\n  items: T[],\n  predicate: (item: T, index: number) => boolean | Promise<boolean>,\n  options?: BatchProcessingOptions,\n): Promise<T[]> {\n  const opts = getOptions(options);\n\n  return (async () => {\n    const result: T[] = [];\n    await processInBatches(items, opts, async (batch, batchStart) => {\n      if (opts.sequentialProcessing) {\n        // Process items sequentially\n        for (let j = 0; j < batch.length; j += 1) {\n          // eslint-disable-next-line no-await-in-loop -- sequential processing is required\n          const passes = await predicate(batch[j], batchStart + j);\n          if (passes) result.push(batch[j]);\n        }\n      } else {\n        // Process items concurrently\n        const flags = await Promise.all(batch.map((item, j) => predicate(item, batchStart + j)));\n        for (let j = 0; j < batch.length; j += 1) {\n          if (flags[j]) result.push(batch[j]);\n        }\n      }\n    });\n    return result;\n  })();\n}\n\n/**\n * Groups an array by a key function in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the array\n * @template K - The type of the grouping key (must extend PropertyKey)\n * @param items - The array to group\n * @param keyFn - The function to extract the grouping key from each item. Receives the item and its index.\n * @param options - Batch processing options\n * @returns A promise that resolves to an object with grouped items\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Group by property with default options\n * const byType = await groupByInBatches(\n *   [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],\n *   (item) => item.type\n * );\n * // Result: {A: [{type: 'A', value: 1}, {type: 'A', value: 3}], B: [{type: 'B', value: 2}]}\n *\n * // With custom batch size\n * const byType = await groupByInBatches(\n *   [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],\n *   (item) => item.type,\n *   { batchSize: 2 }\n * );\n * ```\n */\nexport function groupByInBatches<T, K extends PropertyKey>(\n  items: T[],\n  keyFn: (item: T, index: number) => K | Promise<K>,\n  options?: BatchProcessingOptions,\n): Promise<Record<K, T[]>> {\n  const opts = getOptions(options);\n\n  return (async () => {\n    const result: Record<K, T[]> = {} as Record<K, T[]>;\n    await processInBatches(items, opts, async (batch, batchStart) => {\n      if (opts.sequentialProcessing) {\n        // Process items sequentially\n        for (let j = 0; j < batch.length; j += 1) {\n          // eslint-disable-next-line no-await-in-loop -- sequential processing is required\n          const key = await keyFn(batch[j], batchStart + j);\n          if (!result[key]) result[key] = [];\n          result[key].push(batch[j]);\n        }\n      } else {\n        // Process items concurrently\n        const keys = await Promise.all(batch.map((item, j) => keyFn(item, batchStart + j)));\n        for (let j = 0; j < batch.length; j += 1) {\n          const key = keys[j];\n          if (!result[key]) result[key] = [];\n          result[key].push(batch[j]);\n        }\n      }\n    });\n    return result;\n  })();\n}\n\n/**\n * Reduces an array in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold. Sequential processing is always used for the reducer function,\n * irrespective of the `sequentialProcessing` option.\n *\n * @template T - The type of items in the array\n * @template R - The type of the accumulator/result\n * @param items - The array to reduce\n * @param reducer - The reducer function. Receives the accumulator, current item, and index.\n * @param initialValue - The initial value for the accumulator\n * @param options - Batch processing options\n * @returns A promise that resolves to the reduced value\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Sum numbers with default options\n * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0);\n * // Result: 10\n *\n * // With custom batch size\n * const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, { batchSize: 2 });\n * ```\n */\nexport function reduceInBatches<T, R>(\n  items: T[],\n  reducer: (acc: R, item: T, index: number) => R | Promise<R>,\n  initialValue: R,\n  options?: BatchProcessingOptions,\n): Promise<R> {\n  const opts = getOptions(options);\n\n  return (async () => {\n    let acc = initialValue;\n    await processInBatches(items, opts, async (batch, batchStart) => {\n      // Always sequential, regardless of sequentialProcessing option\n      for (let j = 0; j < batch.length; j += 1) {\n        // eslint-disable-next-line no-await-in-loop -- sequential processing is required\n        acc = await reducer(acc, batch[j], batchStart + j);\n      }\n    });\n    return acc;\n  })();\n}\n\n/**\n * FlatMaps an array in batches to avoid blocking the event loop.\n * Processes items in chunks, flattens the results, and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the input array\n * @template R - The type of items in the flattened result array\n * @param items - The array to flatMap over\n * @param mapFn - The mapping function that returns an array for each item. Receives the item and its index.\n * @param options - Batch processing options\n * @returns A promise that resolves to the flattened mapped array\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Duplicate each item with default options\n * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x]);\n * // Result: [1, 1, 2, 2, 3, 3]\n *\n * // With custom batch size\n * const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x], { batchSize: 2 });\n * ```\n */\nexport function flatMapInBatches<T, R>(\n  items: T[],\n  mapFn: (item: T, index: number) => R[] | Promise<R[]>,\n  options?: BatchProcessingOptions,\n): Promise<R[]> {\n  const opts = getOptions(options);\n\n  return (async () => {\n    const result: R[] = [];\n    await processInBatches(items, opts, async (batch, batchStart) => {\n      if (opts.sequentialProcessing) {\n        // Process items sequentially\n        for (let j = 0; j < batch.length; j += 1) {\n          // eslint-disable-next-line no-await-in-loop -- sequential processing is required\n          const mapped = await mapFn(batch[j], batchStart + j);\n          result.push(...mapped);\n        }\n      } else {\n        // Process items concurrently\n        const mapped = await Promise.all(batch.map((item, j) => mapFn(item, batchStart + j)));\n        mapped.forEach((arr) => {\n          result.push(...arr);\n        });\n      }\n    });\n    return result;\n  })();\n}\n\n/**\n * forEach over an array in batches to avoid blocking the event loop.\n * Processes items in chunks and yields control back to the event loop between batches\n * when the processing time exceeds the threshold.\n *\n * @template T - The type of items in the input array\n * @param items - The array to iterate over\n * @param fn - The function to apply to each item. Receives the item and its index. Can be async.\n * @param options - Batch processing options\n * @returns A promise that resolves when all items have been processed\n * @throws {Error} When batchSize is not a positive integer\n *\n * @example\n * ```typescript\n * // Process items in batches with default options\n * await forEachInBatches([1, 2, 3, 4], async (x) => {\n *   await doSomething(x);\n * });\n *\n * // With custom batch size\n * await forEachInBatches([1, 2, 3, 4], async (x) => {\n *   await doSomething(x);\n * }, { batchSize: 2 });\n * ```\n */\nexport function forEachInBatches<T>(\n  items: T[],\n  fn: (item: T, index: number) => void | Promise<void>,\n  options?: BatchProcessingOptions,\n): Promise<void> {\n  const opts = getOptions(options);\n\n  return processInBatches(items, opts, async (batch, batchStart) => {\n    if (opts.sequentialProcessing) {\n      // Process items sequentially\n      for (let j = 0; j < batch.length; j += 1) {\n        // eslint-disable-next-line no-await-in-loop -- sequential processing is required\n        await fn(batch[j], batchStart + j);\n      }\n    } else {\n      // Process items concurrently\n      await Promise.all(batch.map((item, j) => fn(item, batchStart + j)));\n    }\n  });\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@rudderstack/integrations-lib",
3
- "version": "0.2.33",
3
+ "version": "0.2.34",
4
4
  "description": "",
5
5
  "main": "build/index.js",
6
6
  "module": "build/index.js",