@rudderstack/integrations-lib 0.2.32 → 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.
- package/build/utils/batch-processing.bench.d.ts +2 -0
- package/build/utils/batch-processing.bench.d.ts.map +1 -0
- package/build/utils/batch-processing.bench.js +96 -0
- package/build/utils/batch-processing.d.ts +197 -0
- package/build/utils/batch-processing.d.ts.map +1 -0
- package/build/utils/batch-processing.js +409 -0
- package/build/utils/index.d.ts +1 -0
- package/build/utils/index.d.ts.map +1 -1
- package/build/utils/index.js +2 -1
- package/package.json +1 -1
|
@@ -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":"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","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"]}
|
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Options for batch processing operations
|
|
3
|
+
*/
|
|
4
|
+
export interface BatchProcessingOptions {
|
|
5
|
+
/**
|
|
6
|
+
* Number of items to process in each batch (default: 10)
|
|
7
|
+
* Must be a positive integer.
|
|
8
|
+
*/
|
|
9
|
+
batchSize?: number;
|
|
10
|
+
/**
|
|
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.
|
|
13
|
+
*/
|
|
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;
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Configure global defaults for batch processing operations
|
|
24
|
+
*
|
|
25
|
+
* @param config - Configuration options for batch processing
|
|
26
|
+
* @returns The current configuration after applying changes
|
|
27
|
+
*
|
|
28
|
+
* @example
|
|
29
|
+
* ```typescript
|
|
30
|
+
* // Set both defaults
|
|
31
|
+
* configureBatchProcessingDefaults({ batchSize: 20, yieldThreshold: 5 });
|
|
32
|
+
*
|
|
33
|
+
* // Set only batch size
|
|
34
|
+
* configureBatchProcessingDefaults({ batchSize: 50 });
|
|
35
|
+
*
|
|
36
|
+
* // Enable concurrent processing within batches
|
|
37
|
+
* configureBatchProcessingDefaults({ sequentialProcessing: false });
|
|
38
|
+
*
|
|
39
|
+
* // Get current configuration
|
|
40
|
+
* const currentConfig = configureBatchProcessingDefaults();
|
|
41
|
+
* ```
|
|
42
|
+
*/
|
|
43
|
+
export declare function configureBatchProcessingDefaults(config?: BatchProcessingOptions): BatchProcessingOptions;
|
|
44
|
+
/**
|
|
45
|
+
* Maps over an array in batches to avoid blocking the event loop.
|
|
46
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
47
|
+
* when the processing time exceeds the threshold.
|
|
48
|
+
*
|
|
49
|
+
* @template T - The type of items in the input array
|
|
50
|
+
* @template R - The type of items in the result array
|
|
51
|
+
* @param items - The array to map over
|
|
52
|
+
* @param mapFn - The mapping function to apply to each item. Receives the item and its index.
|
|
53
|
+
* @param options - Batch processing options
|
|
54
|
+
* @returns A promise that resolves to the mapped array
|
|
55
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
56
|
+
*
|
|
57
|
+
* @example
|
|
58
|
+
* ```typescript
|
|
59
|
+
* // Synchronous mapping with default options (sequential processing)
|
|
60
|
+
* const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2);
|
|
61
|
+
*
|
|
62
|
+
* // With concurrent processing within batches
|
|
63
|
+
* const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2, { sequentialProcessing: false });
|
|
64
|
+
* ```
|
|
65
|
+
*/
|
|
66
|
+
export declare function mapInBatches<T, R>(items: T[], mapFn: (item: T, index: number) => R | Promise<R>, options?: BatchProcessingOptions): Promise<R[]>;
|
|
67
|
+
/**
|
|
68
|
+
* Filters an array in batches to avoid blocking the event loop.
|
|
69
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
70
|
+
* when the processing time exceeds the threshold.
|
|
71
|
+
*
|
|
72
|
+
* @template T - The type of items in the array
|
|
73
|
+
* @param items - The array to filter
|
|
74
|
+
* @param predicate - The predicate function to test each item. Receives the item and its index.
|
|
75
|
+
* @param options - Batch processing options
|
|
76
|
+
* @returns A promise that resolves to the filtered array
|
|
77
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
78
|
+
*
|
|
79
|
+
* @example
|
|
80
|
+
* ```typescript
|
|
81
|
+
* // Synchronous filtering with default options
|
|
82
|
+
* const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0);
|
|
83
|
+
* // Result: [2, 4]
|
|
84
|
+
*
|
|
85
|
+
* // With custom batch size
|
|
86
|
+
* const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, { batchSize: 2 });
|
|
87
|
+
* ```
|
|
88
|
+
*/
|
|
89
|
+
export declare function filterInBatches<T>(items: T[], predicate: (item: T, index: number) => boolean | Promise<boolean>, options?: BatchProcessingOptions): Promise<T[]>;
|
|
90
|
+
/**
|
|
91
|
+
* Groups an array by a key function in batches to avoid blocking the event loop.
|
|
92
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
93
|
+
* when the processing time exceeds the threshold.
|
|
94
|
+
*
|
|
95
|
+
* @template T - The type of items in the array
|
|
96
|
+
* @template K - The type of the grouping key (must extend PropertyKey)
|
|
97
|
+
* @param items - The array to group
|
|
98
|
+
* @param keyFn - The function to extract the grouping key from each item. Receives the item and its index.
|
|
99
|
+
* @param options - Batch processing options
|
|
100
|
+
* @returns A promise that resolves to an object with grouped items
|
|
101
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
102
|
+
*
|
|
103
|
+
* @example
|
|
104
|
+
* ```typescript
|
|
105
|
+
* // Group by property with default options
|
|
106
|
+
* const byType = await groupByInBatches(
|
|
107
|
+
* [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],
|
|
108
|
+
* (item) => item.type
|
|
109
|
+
* );
|
|
110
|
+
* // Result: {A: [{type: 'A', value: 1}, {type: 'A', value: 3}], B: [{type: 'B', value: 2}]}
|
|
111
|
+
*
|
|
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
|
+
* );
|
|
118
|
+
* ```
|
|
119
|
+
*/
|
|
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[]>>;
|
|
121
|
+
/**
|
|
122
|
+
* Reduces an array in batches to avoid blocking the event loop.
|
|
123
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
124
|
+
* when the processing time exceeds the threshold. Sequential processing is always used for the reducer function,
|
|
125
|
+
* irrespective of the `sequentialProcessing` option.
|
|
126
|
+
*
|
|
127
|
+
* @template T - The type of items in the array
|
|
128
|
+
* @template R - The type of the accumulator/result
|
|
129
|
+
* @param items - The array to reduce
|
|
130
|
+
* @param reducer - The reducer function. Receives the accumulator, current item, and index.
|
|
131
|
+
* @param initialValue - The initial value for the accumulator
|
|
132
|
+
* @param options - Batch processing options
|
|
133
|
+
* @returns A promise that resolves to the reduced value
|
|
134
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
135
|
+
*
|
|
136
|
+
* @example
|
|
137
|
+
* ```typescript
|
|
138
|
+
* // Sum numbers with default options
|
|
139
|
+
* const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0);
|
|
140
|
+
* // Result: 10
|
|
141
|
+
*
|
|
142
|
+
* // With custom batch size
|
|
143
|
+
* const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, { batchSize: 2 });
|
|
144
|
+
* ```
|
|
145
|
+
*/
|
|
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>;
|
|
147
|
+
/**
|
|
148
|
+
* FlatMaps an array in batches to avoid blocking the event loop.
|
|
149
|
+
* Processes items in chunks, flattens the results, and yields control back to the event loop between batches
|
|
150
|
+
* when the processing time exceeds the threshold.
|
|
151
|
+
*
|
|
152
|
+
* @template T - The type of items in the input array
|
|
153
|
+
* @template R - The type of items in the flattened result array
|
|
154
|
+
* @param items - The array to flatMap over
|
|
155
|
+
* @param mapFn - The mapping function that returns an array for each item. Receives the item and its index.
|
|
156
|
+
* @param options - Batch processing options
|
|
157
|
+
* @returns A promise that resolves to the flattened mapped array
|
|
158
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
159
|
+
*
|
|
160
|
+
* @example
|
|
161
|
+
* ```typescript
|
|
162
|
+
* // Duplicate each item with default options
|
|
163
|
+
* const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x]);
|
|
164
|
+
* // Result: [1, 1, 2, 2, 3, 3]
|
|
165
|
+
*
|
|
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.
|
|
175
|
+
*
|
|
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
|
|
182
|
+
*
|
|
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);
|
|
188
|
+
* });
|
|
189
|
+
*
|
|
190
|
+
* // With custom batch size
|
|
191
|
+
* await forEachInBatches([1, 2, 3, 4], async (x) => {
|
|
192
|
+
* await doSomething(x);
|
|
193
|
+
* }, { batchSize: 2 });
|
|
194
|
+
* ```
|
|
195
|
+
*/
|
|
196
|
+
export declare function forEachInBatches<T>(items: T[], fn: (item: T, index: number) => void | Promise<void>, options?: BatchProcessingOptions): Promise<void>;
|
|
197
|
+
//# sourceMappingURL=batch-processing.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1,409 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.configureBatchProcessingDefaults = configureBatchProcessingDefaults;
|
|
4
|
+
exports.mapInBatches = mapInBatches;
|
|
5
|
+
exports.filterInBatches = filterInBatches;
|
|
6
|
+
exports.groupByInBatches = groupByInBatches;
|
|
7
|
+
exports.reduceInBatches = reduceInBatches;
|
|
8
|
+
exports.flatMapInBatches = flatMapInBatches;
|
|
9
|
+
exports.forEachInBatches = forEachInBatches;
|
|
10
|
+
// Default configuration values (internal)
|
|
11
|
+
let defaultBatchSize = 10;
|
|
12
|
+
let defaultYieldThreshold = 10;
|
|
13
|
+
let defaultSequentialProcessing = true;
|
|
14
|
+
/**
|
|
15
|
+
* Configure global defaults for batch processing operations
|
|
16
|
+
*
|
|
17
|
+
* @param config - Configuration options for batch processing
|
|
18
|
+
* @returns The current configuration after applying changes
|
|
19
|
+
*
|
|
20
|
+
* @example
|
|
21
|
+
* ```typescript
|
|
22
|
+
* // Set both defaults
|
|
23
|
+
* configureBatchProcessingDefaults({ batchSize: 20, yieldThreshold: 5 });
|
|
24
|
+
*
|
|
25
|
+
* // Set only batch size
|
|
26
|
+
* configureBatchProcessingDefaults({ batchSize: 50 });
|
|
27
|
+
*
|
|
28
|
+
* // Enable concurrent processing within batches
|
|
29
|
+
* configureBatchProcessingDefaults({ sequentialProcessing: false });
|
|
30
|
+
*
|
|
31
|
+
* // Get current configuration
|
|
32
|
+
* const currentConfig = configureBatchProcessingDefaults();
|
|
33
|
+
* ```
|
|
34
|
+
*/
|
|
35
|
+
function configureBatchProcessingDefaults(config) {
|
|
36
|
+
if (config) {
|
|
37
|
+
if (config.batchSize !== undefined) {
|
|
38
|
+
if (!Number.isInteger(config.batchSize) || config.batchSize <= 0) {
|
|
39
|
+
throw new Error('batchSize must be a positive integer');
|
|
40
|
+
}
|
|
41
|
+
defaultBatchSize = config.batchSize;
|
|
42
|
+
}
|
|
43
|
+
if (config.yieldThreshold !== undefined) {
|
|
44
|
+
if (!Number.isInteger(config.yieldThreshold) || config.yieldThreshold < 0) {
|
|
45
|
+
throw new Error('yieldThreshold must be a non-negative integer');
|
|
46
|
+
}
|
|
47
|
+
defaultYieldThreshold = config.yieldThreshold;
|
|
48
|
+
}
|
|
49
|
+
if (config.sequentialProcessing !== undefined) {
|
|
50
|
+
defaultSequentialProcessing = Boolean(config.sequentialProcessing);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
return {
|
|
54
|
+
batchSize: defaultBatchSize,
|
|
55
|
+
yieldThreshold: defaultYieldThreshold,
|
|
56
|
+
sequentialProcessing: defaultSequentialProcessing,
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Utility function to defer execution to the next tick of the event loop.
|
|
61
|
+
* This prevents blocking the event loop during heavy batch operations by
|
|
62
|
+
* yielding control back to the event loop using setImmediate.
|
|
63
|
+
*
|
|
64
|
+
* @param startTime - The timestamp when the current batch processing started
|
|
65
|
+
* @param yieldThreshold - Time threshold in milliseconds before yielding control
|
|
66
|
+
* @returns A promise that resolves to a boolean indicating whether a yield occurred
|
|
67
|
+
* @internal
|
|
68
|
+
*/
|
|
69
|
+
function defer(startTime, yieldThreshold) {
|
|
70
|
+
const elapsed = Date.now() - startTime;
|
|
71
|
+
// Only yield if we've exceeded the threshold
|
|
72
|
+
if (elapsed >= yieldThreshold) {
|
|
73
|
+
return new Promise((resolve) => {
|
|
74
|
+
setImmediate(() => resolve(true));
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
// Otherwise continue immediately
|
|
78
|
+
return Promise.resolve(false);
|
|
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
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Maps over an array in batches to avoid blocking the event loop.
|
|
133
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
134
|
+
* when the processing time exceeds the threshold.
|
|
135
|
+
*
|
|
136
|
+
* @template T - The type of items in the input array
|
|
137
|
+
* @template R - The type of items in the result array
|
|
138
|
+
* @param items - The array to map over
|
|
139
|
+
* @param mapFn - The mapping function to apply to each item. Receives the item and its index.
|
|
140
|
+
* @param options - Batch processing options
|
|
141
|
+
* @returns A promise that resolves to the mapped array
|
|
142
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
143
|
+
*
|
|
144
|
+
* @example
|
|
145
|
+
* ```typescript
|
|
146
|
+
* // Synchronous mapping with default options (sequential processing)
|
|
147
|
+
* const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2);
|
|
148
|
+
*
|
|
149
|
+
* // With concurrent processing within batches
|
|
150
|
+
* const doubled = await mapInBatches([1, 2, 3, 4], (x) => x * 2, { sequentialProcessing: false });
|
|
151
|
+
* ```
|
|
152
|
+
*/
|
|
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
|
+
})();
|
|
174
|
+
}
|
|
175
|
+
/**
|
|
176
|
+
* Filters an array in batches to avoid blocking the event loop.
|
|
177
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
178
|
+
* when the processing time exceeds the threshold.
|
|
179
|
+
*
|
|
180
|
+
* @template T - The type of items in the array
|
|
181
|
+
* @param items - The array to filter
|
|
182
|
+
* @param predicate - The predicate function to test each item. Receives the item and its index.
|
|
183
|
+
* @param options - Batch processing options
|
|
184
|
+
* @returns A promise that resolves to the filtered array
|
|
185
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
186
|
+
*
|
|
187
|
+
* @example
|
|
188
|
+
* ```typescript
|
|
189
|
+
* // Synchronous filtering with default options
|
|
190
|
+
* const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0);
|
|
191
|
+
* // Result: [2, 4]
|
|
192
|
+
*
|
|
193
|
+
* // With custom batch size
|
|
194
|
+
* const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, { batchSize: 2 });
|
|
195
|
+
* ```
|
|
196
|
+
*/
|
|
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
|
+
})();
|
|
222
|
+
}
|
|
223
|
+
/**
|
|
224
|
+
* Groups an array by a key function in batches to avoid blocking the event loop.
|
|
225
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
226
|
+
* when the processing time exceeds the threshold.
|
|
227
|
+
*
|
|
228
|
+
* @template T - The type of items in the array
|
|
229
|
+
* @template K - The type of the grouping key (must extend PropertyKey)
|
|
230
|
+
* @param items - The array to group
|
|
231
|
+
* @param keyFn - The function to extract the grouping key from each item. Receives the item and its index.
|
|
232
|
+
* @param options - Batch processing options
|
|
233
|
+
* @returns A promise that resolves to an object with grouped items
|
|
234
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
235
|
+
*
|
|
236
|
+
* @example
|
|
237
|
+
* ```typescript
|
|
238
|
+
* // Group by property with default options
|
|
239
|
+
* const byType = await groupByInBatches(
|
|
240
|
+
* [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],
|
|
241
|
+
* (item) => item.type
|
|
242
|
+
* );
|
|
243
|
+
* // Result: {A: [{type: 'A', value: 1}, {type: 'A', value: 3}], B: [{type: 'B', value: 2}]}
|
|
244
|
+
*
|
|
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
|
+
* );
|
|
251
|
+
* ```
|
|
252
|
+
*/
|
|
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
|
+
})();
|
|
281
|
+
}
|
|
282
|
+
/**
|
|
283
|
+
* Reduces an array in batches to avoid blocking the event loop.
|
|
284
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
285
|
+
* when the processing time exceeds the threshold. Sequential processing is always used for the reducer function,
|
|
286
|
+
* irrespective of the `sequentialProcessing` option.
|
|
287
|
+
*
|
|
288
|
+
* @template T - The type of items in the array
|
|
289
|
+
* @template R - The type of the accumulator/result
|
|
290
|
+
* @param items - The array to reduce
|
|
291
|
+
* @param reducer - The reducer function. Receives the accumulator, current item, and index.
|
|
292
|
+
* @param initialValue - The initial value for the accumulator
|
|
293
|
+
* @param options - Batch processing options
|
|
294
|
+
* @returns A promise that resolves to the reduced value
|
|
295
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
296
|
+
*
|
|
297
|
+
* @example
|
|
298
|
+
* ```typescript
|
|
299
|
+
* // Sum numbers with default options
|
|
300
|
+
* const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0);
|
|
301
|
+
* // Result: 10
|
|
302
|
+
*
|
|
303
|
+
* // With custom batch size
|
|
304
|
+
* const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, { batchSize: 2 });
|
|
305
|
+
* ```
|
|
306
|
+
*/
|
|
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
|
+
})();
|
|
320
|
+
}
|
|
321
|
+
/**
|
|
322
|
+
* FlatMaps an array in batches to avoid blocking the event loop.
|
|
323
|
+
* Processes items in chunks, flattens the results, and yields control back to the event loop between batches
|
|
324
|
+
* when the processing time exceeds the threshold.
|
|
325
|
+
*
|
|
326
|
+
* @template T - The type of items in the input array
|
|
327
|
+
* @template R - The type of items in the flattened result array
|
|
328
|
+
* @param items - The array to flatMap over
|
|
329
|
+
* @param mapFn - The mapping function that returns an array for each item. Receives the item and its index.
|
|
330
|
+
* @param options - Batch processing options
|
|
331
|
+
* @returns A promise that resolves to the flattened mapped array
|
|
332
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
333
|
+
*
|
|
334
|
+
* @example
|
|
335
|
+
* ```typescript
|
|
336
|
+
* // Duplicate each item with default options
|
|
337
|
+
* const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x]);
|
|
338
|
+
* // Result: [1, 1, 2, 2, 3, 3]
|
|
339
|
+
*
|
|
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.
|
|
372
|
+
*
|
|
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
|
|
379
|
+
*
|
|
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);
|
|
385
|
+
* });
|
|
386
|
+
*
|
|
387
|
+
* // With custom batch size
|
|
388
|
+
* await forEachInBatches([1, 2, 3, 4], async (x) => {
|
|
389
|
+
* await doSomething(x);
|
|
390
|
+
* }, { batchSize: 2 });
|
|
391
|
+
* ```
|
|
392
|
+
*/
|
|
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
|
+
});
|
|
408
|
+
}
|
|
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/build/utils/index.d.ts
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/utils/index.ts"],"names":[],"mappings":"AAAA,cAAc,yBAAyB,CAAC;AACxC,cAAc,QAAQ,CAAC;AACvB,cAAc,WAAW,CAAC;AAC1B,cAAc,SAAS,CAAC;AACxB,cAAc,OAAO,CAAC"}
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/utils/index.ts"],"names":[],"mappings":"AAAA,cAAc,oBAAoB,CAAC;AACnC,cAAc,yBAAyB,CAAC;AACxC,cAAc,QAAQ,CAAC;AACvB,cAAc,WAAW,CAAC;AAC1B,cAAc,SAAS,CAAC;AACxB,cAAc,OAAO,CAAC"}
|
package/build/utils/index.js
CHANGED
|
@@ -14,9 +14,10 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
14
14
|
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
15
|
};
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./batch-processing"), exports);
|
|
17
18
|
__exportStar(require("./json-schema-generator"), exports);
|
|
18
19
|
__exportStar(require("./misc"), exports);
|
|
19
20
|
__exportStar(require("./request"), exports);
|
|
20
21
|
__exportStar(require("./tests"), exports);
|
|
21
22
|
__exportStar(require("./zod"), exports);
|
|
22
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdXRpbHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHFEQUFtQztBQUNuQywwREFBd0M7QUFDeEMseUNBQXVCO0FBQ3ZCLDRDQUEwQjtBQUMxQiwwQ0FBd0I7QUFDeEIsd0NBQXNCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9iYXRjaC1wcm9jZXNzaW5nJztcbmV4cG9ydCAqIGZyb20gJy4vanNvbi1zY2hlbWEtZ2VuZXJhdG9yJztcbmV4cG9ydCAqIGZyb20gJy4vbWlzYyc7XG5leHBvcnQgKiBmcm9tICcuL3JlcXVlc3QnO1xuZXhwb3J0ICogZnJvbSAnLi90ZXN0cyc7XG5leHBvcnQgKiBmcm9tICcuL3pvZCc7XG4iXX0=
|