@rudderstack/integrations-lib 0.2.32 → 0.2.33
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,205 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Configuration options for batch processing operations
|
|
3
|
+
*/
|
|
4
|
+
export interface BatchProcessingDefaults {
|
|
5
|
+
/**
|
|
6
|
+
* Default number of items to process in each batch
|
|
7
|
+
*/
|
|
8
|
+
batchSize?: number;
|
|
9
|
+
/**
|
|
10
|
+
* Default time threshold in milliseconds before yielding control back to the event loop
|
|
11
|
+
*/
|
|
12
|
+
yieldThreshold?: number;
|
|
13
|
+
}
|
|
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
|
+
* // Get current configuration
|
|
29
|
+
* const currentConfig = configureBatchProcessingDefaults();
|
|
30
|
+
* ```
|
|
31
|
+
*/
|
|
32
|
+
export declare function configureBatchProcessingDefaults(config?: BatchProcessingDefaults): BatchProcessingDefaults;
|
|
33
|
+
/**
|
|
34
|
+
* Maps over an array in batches to avoid blocking the event loop.
|
|
35
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
36
|
+
* when the processing time exceeds the threshold.
|
|
37
|
+
*
|
|
38
|
+
* @template T - The type of items in the input array
|
|
39
|
+
* @template R - The type of items in the result array
|
|
40
|
+
* @param items - The array to map over
|
|
41
|
+
* @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)
|
|
44
|
+
* @returns A promise that resolves to the mapped array
|
|
45
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
46
|
+
*
|
|
47
|
+
* @example
|
|
48
|
+
* ```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);
|
|
55
|
+
*
|
|
56
|
+
* // With index
|
|
57
|
+
* const indexed = await mapInBatches(['a', 'b'], (item, index) => `${item}-${index}`);
|
|
58
|
+
* // Result: ['a-0', 'b-1']
|
|
59
|
+
* ```
|
|
60
|
+
*/
|
|
61
|
+
export declare function mapInBatches<T, R>(items: T[], mapFn: (item: T, index: number) => R | Promise<R>, batchSize?: number, yieldThreshold?: number): Promise<R[]>;
|
|
62
|
+
/**
|
|
63
|
+
* Filters an array in batches to avoid blocking the event loop.
|
|
64
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
65
|
+
* when the processing time exceeds the threshold.
|
|
66
|
+
*
|
|
67
|
+
* @template T - The type of items in the array
|
|
68
|
+
* @param items - The array to filter
|
|
69
|
+
* @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)
|
|
72
|
+
* @returns A promise that resolves to the filtered array
|
|
73
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
74
|
+
*
|
|
75
|
+
* @example
|
|
76
|
+
* ```typescript
|
|
77
|
+
* // Synchronous filtering
|
|
78
|
+
* const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, 2);
|
|
79
|
+
* // Result: [2, 4]
|
|
80
|
+
*
|
|
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']
|
|
90
|
+
* ```
|
|
91
|
+
*/
|
|
92
|
+
export declare function filterInBatches<T>(items: T[], predicate: (item: T, index: number) => boolean | Promise<boolean>, batchSize?: number, yieldThreshold?: number): Promise<T[]>;
|
|
93
|
+
/**
|
|
94
|
+
* Groups an array by a key function in batches to avoid blocking the event loop.
|
|
95
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
96
|
+
* when the processing time exceeds the threshold.
|
|
97
|
+
*
|
|
98
|
+
* @template T - The type of items in the array
|
|
99
|
+
* @template K - The type of the grouping key (must extend PropertyKey)
|
|
100
|
+
* @param items - The array to group
|
|
101
|
+
* @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)
|
|
104
|
+
* @returns A promise that resolves to an object with grouped items
|
|
105
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
106
|
+
*
|
|
107
|
+
* @example
|
|
108
|
+
* ```typescript
|
|
109
|
+
* // Group by property
|
|
110
|
+
* const byType = await groupByInBatches(
|
|
111
|
+
* [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],
|
|
112
|
+
* (item) => item.type,
|
|
113
|
+
* 2
|
|
114
|
+
* );
|
|
115
|
+
* // Result: {A: [{type: 'A', value: 1}, {type: 'A', value: 3}], B: [{type: 'B', value: 2}]}
|
|
116
|
+
*
|
|
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
|
+
* });
|
|
125
|
+
* ```
|
|
126
|
+
*/
|
|
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[]>>;
|
|
128
|
+
/**
|
|
129
|
+
* Reduces an array in batches to avoid blocking the event loop.
|
|
130
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
131
|
+
* when the processing time exceeds the threshold.
|
|
132
|
+
*
|
|
133
|
+
* @template T - The type of items in the array
|
|
134
|
+
* @template R - The type of the accumulator/result
|
|
135
|
+
* @param items - The array to reduce
|
|
136
|
+
* @param reducer - The reducer function. Receives the accumulator, current item, and index.
|
|
137
|
+
* @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)
|
|
140
|
+
* @returns A promise that resolves to the reduced value
|
|
141
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
142
|
+
*
|
|
143
|
+
* @example
|
|
144
|
+
* ```typescript
|
|
145
|
+
* // Sum numbers
|
|
146
|
+
* const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, 2);
|
|
147
|
+
* // Result: 10
|
|
148
|
+
*
|
|
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
|
+
* }, []);
|
|
166
|
+
* ```
|
|
167
|
+
*/
|
|
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>;
|
|
169
|
+
/**
|
|
170
|
+
* FlatMaps an array in batches to avoid blocking the event loop.
|
|
171
|
+
* Processes items in chunks, flattens the results, and yields control back to the event loop between batches
|
|
172
|
+
* when the processing time exceeds the threshold.
|
|
173
|
+
*
|
|
174
|
+
* @template T - The type of items in the input array
|
|
175
|
+
* @template R - The type of items in the flattened result array
|
|
176
|
+
* @param items - The array to flatMap over
|
|
177
|
+
* @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)
|
|
180
|
+
* @returns A promise that resolves to the flattened mapped array
|
|
181
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
182
|
+
*
|
|
183
|
+
* @example
|
|
184
|
+
* ```typescript
|
|
185
|
+
* // Duplicate each item
|
|
186
|
+
* const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x], 2);
|
|
187
|
+
* // Result: [1, 1, 2, 2, 3, 3]
|
|
188
|
+
*
|
|
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]
|
|
192
|
+
*
|
|
193
|
+
* // Split strings into characters
|
|
194
|
+
* const chars = await flatMapInBatches(['ab', 'cd'], (str) => str.split(''));
|
|
195
|
+
* // Result: ['a', 'b', 'c', 'd']
|
|
196
|
+
*
|
|
197
|
+
* // Asynchronous mapping
|
|
198
|
+
* const expanded = await flatMapInBatches(categories, async (category) => {
|
|
199
|
+
* const items = await fetchItemsForCategory(category);
|
|
200
|
+
* return items;
|
|
201
|
+
* });
|
|
202
|
+
* ```
|
|
203
|
+
*/
|
|
204
|
+
export declare function flatMapInBatches<T, R>(items: T[], mapFn: (item: T, index: number) => R[] | Promise<R[]>, batchSize?: number, yieldThreshold?: number): Promise<R[]>;
|
|
205
|
+
//# 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,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"}
|
|
@@ -0,0 +1,330 @@
|
|
|
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
|
+
// Default configuration values (internal)
|
|
10
|
+
let defaultBatchSize = 10;
|
|
11
|
+
let defaultYieldThreshold = 10;
|
|
12
|
+
/**
|
|
13
|
+
* Configure global defaults for batch processing operations
|
|
14
|
+
*
|
|
15
|
+
* @param config - Configuration options for batch processing
|
|
16
|
+
* @returns The current configuration after applying changes
|
|
17
|
+
*
|
|
18
|
+
* @example
|
|
19
|
+
* ```typescript
|
|
20
|
+
* // Set both defaults
|
|
21
|
+
* configureBatchProcessingDefaults({ batchSize: 20, yieldThreshold: 5 });
|
|
22
|
+
*
|
|
23
|
+
* // Set only batch size
|
|
24
|
+
* configureBatchProcessingDefaults({ batchSize: 50 });
|
|
25
|
+
*
|
|
26
|
+
* // Get current configuration
|
|
27
|
+
* const currentConfig = configureBatchProcessingDefaults();
|
|
28
|
+
* ```
|
|
29
|
+
*/
|
|
30
|
+
function configureBatchProcessingDefaults(config) {
|
|
31
|
+
if (config) {
|
|
32
|
+
if (config.batchSize !== undefined) {
|
|
33
|
+
if (!Number.isInteger(config.batchSize) || config.batchSize <= 0) {
|
|
34
|
+
throw new Error('batchSize must be a positive integer');
|
|
35
|
+
}
|
|
36
|
+
defaultBatchSize = config.batchSize;
|
|
37
|
+
}
|
|
38
|
+
if (config.yieldThreshold !== undefined) {
|
|
39
|
+
if (!Number.isInteger(config.yieldThreshold) || config.yieldThreshold < 0) {
|
|
40
|
+
throw new Error('yieldThreshold must be a non-negative integer');
|
|
41
|
+
}
|
|
42
|
+
defaultYieldThreshold = config.yieldThreshold;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return {
|
|
46
|
+
batchSize: defaultBatchSize,
|
|
47
|
+
yieldThreshold: defaultYieldThreshold,
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Utility function to defer execution to the next tick of the event loop.
|
|
52
|
+
* This prevents blocking the event loop during heavy batch operations by
|
|
53
|
+
* yielding control back to the event loop using setImmediate.
|
|
54
|
+
*
|
|
55
|
+
* @param startTime - The timestamp when the current batch processing started
|
|
56
|
+
* @param yieldThreshold - Time threshold in milliseconds before yielding control
|
|
57
|
+
* @returns A promise that resolves to a boolean indicating whether a yield occurred
|
|
58
|
+
* @internal
|
|
59
|
+
*/
|
|
60
|
+
function defer(startTime, yieldThreshold) {
|
|
61
|
+
const elapsed = Date.now() - startTime;
|
|
62
|
+
// Only yield if we've exceeded the threshold
|
|
63
|
+
if (elapsed >= yieldThreshold) {
|
|
64
|
+
return new Promise((resolve) => {
|
|
65
|
+
setImmediate(() => resolve(true));
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
// Otherwise continue immediately
|
|
69
|
+
return Promise.resolve(false);
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Maps over an array in batches to avoid blocking the event loop.
|
|
73
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
74
|
+
* when the processing time exceeds the threshold.
|
|
75
|
+
*
|
|
76
|
+
* @template T - The type of items in the input array
|
|
77
|
+
* @template R - The type of items in the result array
|
|
78
|
+
* @param items - The array to map over
|
|
79
|
+
* @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)
|
|
82
|
+
* @returns A promise that resolves to the mapped array
|
|
83
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
84
|
+
*
|
|
85
|
+
* @example
|
|
86
|
+
* ```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);
|
|
93
|
+
*
|
|
94
|
+
* // With index
|
|
95
|
+
* const indexed = await mapInBatches(['a', 'b'], (item, index) => `${item}-${index}`);
|
|
96
|
+
* // Result: ['a-0', 'b-1']
|
|
97
|
+
* ```
|
|
98
|
+
*/
|
|
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();
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Filters an array in batches to avoid blocking the event loop.
|
|
117
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
118
|
+
* when the processing time exceeds the threshold.
|
|
119
|
+
*
|
|
120
|
+
* @template T - The type of items in the array
|
|
121
|
+
* @param items - The array to filter
|
|
122
|
+
* @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)
|
|
125
|
+
* @returns A promise that resolves to the filtered array
|
|
126
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
127
|
+
*
|
|
128
|
+
* @example
|
|
129
|
+
* ```typescript
|
|
130
|
+
* // Synchronous filtering
|
|
131
|
+
* const evens = await filterInBatches([1, 2, 3, 4, 5], (x) => x % 2 === 0, 2);
|
|
132
|
+
* // Result: [2, 4]
|
|
133
|
+
*
|
|
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']
|
|
143
|
+
* ```
|
|
144
|
+
*/
|
|
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();
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Groups an array by a key function in batches to avoid blocking the event loop.
|
|
164
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
165
|
+
* when the processing time exceeds the threshold.
|
|
166
|
+
*
|
|
167
|
+
* @template T - The type of items in the array
|
|
168
|
+
* @template K - The type of the grouping key (must extend PropertyKey)
|
|
169
|
+
* @param items - The array to group
|
|
170
|
+
* @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)
|
|
173
|
+
* @returns A promise that resolves to an object with grouped items
|
|
174
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
175
|
+
*
|
|
176
|
+
* @example
|
|
177
|
+
* ```typescript
|
|
178
|
+
* // Group by property
|
|
179
|
+
* const byType = await groupByInBatches(
|
|
180
|
+
* [{type: 'A', value: 1}, {type: 'B', value: 2}, {type: 'A', value: 3}],
|
|
181
|
+
* (item) => item.type,
|
|
182
|
+
* 2
|
|
183
|
+
* );
|
|
184
|
+
* // Result: {A: [{type: 'A', value: 1}, {type: 'A', value: 3}], B: [{type: 'B', value: 2}]}
|
|
185
|
+
*
|
|
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
|
+
* });
|
|
194
|
+
* ```
|
|
195
|
+
*/
|
|
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();
|
|
218
|
+
}
|
|
219
|
+
/**
|
|
220
|
+
* Reduces an array in batches to avoid blocking the event loop.
|
|
221
|
+
* Processes items in chunks and yields control back to the event loop between batches
|
|
222
|
+
* when the processing time exceeds the threshold.
|
|
223
|
+
*
|
|
224
|
+
* @template T - The type of items in the array
|
|
225
|
+
* @template R - The type of the accumulator/result
|
|
226
|
+
* @param items - The array to reduce
|
|
227
|
+
* @param reducer - The reducer function. Receives the accumulator, current item, and index.
|
|
228
|
+
* @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)
|
|
231
|
+
* @returns A promise that resolves to the reduced value
|
|
232
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
233
|
+
*
|
|
234
|
+
* @example
|
|
235
|
+
* ```typescript
|
|
236
|
+
* // Sum numbers
|
|
237
|
+
* const sum = await reduceInBatches([1, 2, 3, 4], (acc, x) => acc + x, 0, 2);
|
|
238
|
+
* // Result: 10
|
|
239
|
+
*
|
|
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
|
+
* }, []);
|
|
257
|
+
* ```
|
|
258
|
+
*/
|
|
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();
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* FlatMaps an array in batches to avoid blocking the event loop.
|
|
280
|
+
* Processes items in chunks, flattens the results, and yields control back to the event loop between batches
|
|
281
|
+
* when the processing time exceeds the threshold.
|
|
282
|
+
*
|
|
283
|
+
* @template T - The type of items in the input array
|
|
284
|
+
* @template R - The type of items in the flattened result array
|
|
285
|
+
* @param items - The array to flatMap over
|
|
286
|
+
* @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)
|
|
289
|
+
* @returns A promise that resolves to the flattened mapped array
|
|
290
|
+
* @throws {Error} When batchSize is not a positive integer
|
|
291
|
+
*
|
|
292
|
+
* @example
|
|
293
|
+
* ```typescript
|
|
294
|
+
* // Duplicate each item
|
|
295
|
+
* const duplicated = await flatMapInBatches([1, 2, 3], (x) => [x, x], 2);
|
|
296
|
+
* // Result: [1, 1, 2, 2, 3, 3]
|
|
297
|
+
*
|
|
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]
|
|
301
|
+
*
|
|
302
|
+
* // Split strings into characters
|
|
303
|
+
* const chars = await flatMapInBatches(['ab', 'cd'], (str) => str.split(''));
|
|
304
|
+
* // Result: ['a', 'b', 'c', 'd']
|
|
305
|
+
*
|
|
306
|
+
* // Asynchronous mapping
|
|
307
|
+
* const expanded = await flatMapInBatches(categories, async (category) => {
|
|
308
|
+
* const items = await fetchItemsForCategory(category);
|
|
309
|
+
* return items;
|
|
310
|
+
* });
|
|
311
|
+
* ```
|
|
312
|
+
*/
|
|
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();
|
|
329
|
+
}
|
|
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"]}
|
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=
|