import {_setImmediate} from './_setImmediate'
/**
* Run `items` on async `task` function in series. Stops at the first error encountered.
*
* @name eachSeries
* @memberOf module:serial
* @static
* @method
* @param {Array<any>} items - Array of items
* @param {Function} task - iterator function of type `function (item: any, cb: Function, index: Number)`
* @param {Function} [callback] - optional callback `function (errors: <Error>, result: Array<any>)`
* @example
* eachSeries([1, 2, 3],
* (item, cb, index) => {
* setImmediate(() => {
* cb(index % 2 ? null : 'error', item + index)
* })
* }, (err, res) => {
* //> err = 'error'
* //> res = [1, 4]
* }
* )
*/
export default function eachSeries (items, task, callback) {
let length = items.length
let results = []
let i = 0
function cb (err, res) {
results.push(res)
/* istanbul ignore else */
if (err || length === i) {
callback && callback(err, results)
} else if (i < length) {
_setImmediate(() => { // prevent RangeError: Maximum call stack size exceeded for sync tasks
run()
})
}
}
function run () {
task(items[i], cb, i++)
}
run()
}