@@ -10,7 +10,10 @@ const {
10
10
} ,
11
11
AbortError,
12
12
} = require ( 'internal/errors' ) ;
13
- const { validateInteger } = require ( 'internal/validators' ) ;
13
+ const {
14
+ validateAbortSignal,
15
+ validateInteger,
16
+ } = require ( 'internal/validators' ) ;
14
17
const { kWeakHandler } = require ( 'internal/event_target' ) ;
15
18
const { finished } = require ( 'internal/streams/end-of-stream' ) ;
16
19
@@ -38,6 +41,10 @@ function map(fn, options) {
38
41
throw new ERR_INVALID_ARG_TYPE ( 'options' , [ 'Object' ] ) ;
39
42
}
40
43
44
+ if ( options ?. signal != null ) {
45
+ validateAbortSignal ( options . signal , 'options.signal' ) ;
46
+ }
47
+
41
48
let concurrency = 1 ;
42
49
if ( options ?. concurrency != null ) {
43
50
concurrency = MathFloor ( options . concurrency ) ;
@@ -161,17 +168,35 @@ function map(fn, options) {
161
168
} . call ( this ) ;
162
169
}
163
170
164
- async function * asIndexedPairs ( options ) {
165
- let index = 0 ;
166
- for await ( const val of this ) {
167
- if ( options ?. signal ?. aborted ) {
168
- throw new AbortError ( { cause : options . signal . reason } ) ;
169
- }
170
- yield [ index ++ , val ] ;
171
+ function asIndexedPairs ( options ) {
172
+ if ( options != null && typeof options !== 'object' ) {
173
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , [ 'Object' ] ) ;
171
174
}
175
+
176
+ if ( options ?. signal != null ) {
177
+ validateAbortSignal ( options . signal , 'options.signal' ) ;
178
+ }
179
+
180
+ return async function * asIndexedPairs ( ) {
181
+ let index = 0 ;
182
+ for await ( const val of this ) {
183
+ if ( options ?. signal ?. aborted ) {
184
+ throw new AbortError ( { cause : options . signal . reason } ) ;
185
+ }
186
+ yield [ index ++ , val ] ;
187
+ }
188
+ } . call ( this ) ;
172
189
}
173
190
174
- async function some ( fn , options ) {
191
+ function some ( fn , options ) {
192
+ if ( options != null && typeof options !== 'object' ) {
193
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , [ 'Object' ] ) ;
194
+ }
195
+
196
+ if ( options ?. signal != null ) {
197
+ validateAbortSignal ( options . signal , 'options.signal' ) ;
198
+ }
199
+
175
200
// https://tc39.es/proposal-iterator-helpers/#sec-iteratorprototype.some
176
201
// Note that some does short circuit but also closes the iterator if it does
177
202
const ac = new AbortController ( ) ;
@@ -185,27 +210,32 @@ async function some(fn, options) {
185
210
} ) ;
186
211
}
187
212
const mapped = this . map ( fn , { ...options , signal : ac . signal } ) ;
188
- for await ( const result of mapped ) {
189
- if ( result ) {
190
- ac . abort ( ) ;
191
- return true ;
213
+ return async function some ( ) {
214
+ for await ( const result of mapped ) {
215
+ if ( result ) {
216
+ ac . abort ( ) ;
217
+ return true ;
218
+ }
192
219
}
193
- }
194
- return false ;
220
+ return false ;
221
+ } . call ( this ) ;
195
222
}
196
223
197
- async function every ( fn , options ) {
224
+ function every ( fn , options ) {
198
225
if ( typeof fn !== 'function' ) {
199
226
throw new ERR_INVALID_ARG_TYPE (
200
227
'fn' , [ 'Function' , 'AsyncFunction' ] , fn ) ;
201
228
}
202
229
// https://en.wikipedia.org/wiki/De_Morgan%27s_laws
203
- return ! ( await some . call ( this , async ( ...args ) => {
230
+ const somePromise = some . call ( this , async ( ...args ) => {
204
231
return ! ( await fn ( ...args ) ) ;
205
- } , options ) ) ;
232
+ } , options ) ;
233
+ return async function every ( ) {
234
+ return ! ( await somePromise ) ;
235
+ } . call ( this ) ;
206
236
}
207
237
208
- async function forEach ( fn , options ) {
238
+ function forEach ( fn , options ) {
209
239
if ( typeof fn !== 'function' ) {
210
240
throw new ERR_INVALID_ARG_TYPE (
211
241
'fn' , [ 'Function' , 'AsyncFunction' ] , fn ) ;
@@ -214,8 +244,11 @@ async function forEach(fn, options) {
214
244
await fn ( value , options ) ;
215
245
return kEmpty ;
216
246
}
217
- // eslint-disable-next-line no-unused-vars
218
- for await ( const unused of this . map ( forEachFn , options ) ) ;
247
+ const values = this . map ( forEachFn , options ) ;
248
+ return async function forEach ( ) {
249
+ // eslint-disable-next-line no-unused-vars
250
+ for await ( const unused of values ) ;
251
+ } . call ( this ) ;
219
252
}
220
253
221
254
function filter ( fn , options ) {
@@ -241,56 +274,78 @@ class ReduceAwareErrMissingArgs extends ERR_MISSING_ARGS {
241
274
}
242
275
}
243
276
244
- async function reduce ( reducer , initialValue , options ) {
277
+ function reduce ( reducer , initialValue , options ) {
245
278
if ( typeof reducer !== 'function' ) {
246
279
throw new ERR_INVALID_ARG_TYPE (
247
280
'reducer' , [ 'Function' , 'AsyncFunction' ] , reducer ) ;
248
281
}
249
- let hasInitialValue = arguments . length > 1 ;
250
- if ( options ?. signal ?. aborted ) {
251
- const err = new AbortError ( undefined , { cause : options . signal . reason } ) ;
252
- this . once ( 'error' , ( ) => { } ) ; // The error is already propagated
253
- await finished ( this . destroy ( err ) ) ;
254
- throw err ;
282
+
283
+ if ( options != null && typeof options !== 'object' ) {
284
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , [ 'Object' ] ) ;
255
285
}
256
- const ac = new AbortController ( ) ;
257
- const signal = ac . signal ;
258
- if ( options ?. signal ) {
259
- const opts = { once : true , [ kWeakHandler ] : this } ;
260
- options . signal . addEventListener ( 'abort' , ( ) => ac . abort ( ) , opts ) ;
286
+
287
+ if ( options ?. signal != null ) {
288
+ validateAbortSignal ( options . signal , 'options.signal' ) ;
261
289
}
262
- let gotAnyItemFromStream = false ;
263
- try {
264
- for await ( const value of this ) {
265
- gotAnyItemFromStream = true ;
266
- if ( options ?. signal ?. aborted ) {
267
- throw new AbortError ( ) ;
290
+
291
+ let hasInitialValue = arguments . length > 1 ;
292
+
293
+ return async function reduce ( ) {
294
+ if ( options ?. signal ?. aborted ) {
295
+ const err = new AbortError ( undefined , { cause : options . signal . reason } ) ;
296
+ this . once ( 'error' , ( ) => { } ) ; // The error is already propagated
297
+ await finished ( this . destroy ( err ) ) ;
298
+ throw err ;
299
+ }
300
+ const ac = new AbortController ( ) ;
301
+ const signal = ac . signal ;
302
+ if ( options ?. signal ) {
303
+ const opts = { once : true , [ kWeakHandler ] : this } ;
304
+ options . signal . addEventListener ( 'abort' , ( ) => ac . abort ( ) , opts ) ;
305
+ }
306
+ let gotAnyItemFromStream = false ;
307
+ try {
308
+ for await ( const value of this ) {
309
+ gotAnyItemFromStream = true ;
310
+ if ( options ?. signal ?. aborted ) {
311
+ throw new AbortError ( ) ;
312
+ }
313
+ if ( ! hasInitialValue ) {
314
+ initialValue = value ;
315
+ hasInitialValue = true ;
316
+ } else {
317
+ initialValue = await reducer ( initialValue , value , { signal } ) ;
318
+ }
268
319
}
269
- if ( ! hasInitialValue ) {
270
- initialValue = value ;
271
- hasInitialValue = true ;
272
- } else {
273
- initialValue = await reducer ( initialValue , value , { signal } ) ;
320
+ if ( ! gotAnyItemFromStream && ! hasInitialValue ) {
321
+ throw new ReduceAwareErrMissingArgs ( ) ;
274
322
}
323
+ } finally {
324
+ ac . abort ( ) ;
275
325
}
276
- if ( ! gotAnyItemFromStream && ! hasInitialValue ) {
277
- throw new ReduceAwareErrMissingArgs ( ) ;
278
- }
279
- } finally {
280
- ac . abort ( ) ;
281
- }
282
- return initialValue ;
326
+ return initialValue ;
327
+ } . call ( this ) ;
283
328
}
284
329
285
- async function toArray ( options ) {
286
- const result = [ ] ;
287
- for await ( const val of this ) {
288
- if ( options ?. signal ?. aborted ) {
289
- throw new AbortError ( undefined , { cause : options . signal . reason } ) ;
290
- }
291
- ArrayPrototypePush ( result , val ) ;
330
+ function toArray ( options ) {
331
+ if ( options != null && typeof options !== 'object' ) {
332
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , [ 'Object' ] ) ;
333
+ }
334
+
335
+ if ( options ?. signal != null ) {
336
+ validateAbortSignal ( options . signal , 'options.signal' ) ;
292
337
}
293
- return result ;
338
+
339
+ return async function toArray ( ) {
340
+ const result = [ ] ;
341
+ for await ( const val of this ) {
342
+ if ( options ?. signal ?. aborted ) {
343
+ throw new AbortError ( undefined , { cause : options . signal . reason } ) ;
344
+ }
345
+ ArrayPrototypePush ( result , val ) ;
346
+ }
347
+ return result ;
348
+ } . call ( this ) ;
294
349
}
295
350
296
351
function flatMap ( fn , options ) {
@@ -316,6 +371,14 @@ function toIntegerOrInfinity(number) {
316
371
}
317
372
318
373
function drop ( number , options ) {
374
+ if ( options != null && typeof options !== 'object' ) {
375
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , [ 'Object' ] ) ;
376
+ }
377
+
378
+ if ( options ?. signal != null ) {
379
+ validateAbortSignal ( options . signal , 'options.signal' ) ;
380
+ }
381
+
319
382
number = toIntegerOrInfinity ( number ) ;
320
383
return async function * drop ( ) {
321
384
if ( options ?. signal ?. aborted ) {
@@ -334,6 +397,14 @@ function drop(number, options) {
334
397
335
398
336
399
function take ( number , options ) {
400
+ if ( options != null && typeof options !== 'object' ) {
401
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , [ 'Object' ] ) ;
402
+ }
403
+
404
+ if ( options ?. signal != null ) {
405
+ validateAbortSignal ( options . signal , 'options.signal' ) ;
406
+ }
407
+
337
408
number = toIntegerOrInfinity ( number ) ;
338
409
return async function * take ( ) {
339
410
if ( options ?. signal ?. aborted ) {
0 commit comments