-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy path6-promise.js
103 lines (89 loc) · 2.41 KB
/
6-promise.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
'use strict';
class Queue {
constructor(concurrency) {
this.concurrency = concurrency;
this.count = 0;
this.waiting = [];
this.onProcess = null;
this.onDone = null;
this.onSuccess = null;
this.onFailure = null;
this.onDrain = null;
}
static channels(concurrency) {
return new Queue(concurrency);
}
add(task) {
this.waiting.push(task);
const hasChannel = this.count < this.concurrency;
if (hasChannel) this.next();
}
next() {
const emptyChannels = this.concurrency - this.count;
let launchCount = Math.min(emptyChannels, this.waiting.length);
while (launchCount-- > 0) {
this.count++;
const task = this.waiting.shift();
this.onProcess(task)
.then(
(res) => void this.finish(null, res),
(err) => void this.finish(err),
)
.finally(() => {
this.count--;
if (this.waiting.length > 0) this.next();
});
}
}
finish(error, res) {
const { onFailure, onSuccess, onDone, onDrain } = this;
if (error && onFailure) onFailure(error, res);
else if (onSuccess) onSuccess(res);
if (onDone) onDone(error, res);
if (this.count === 0 && this.waiting.length === 0 && onDrain) onDrain();
}
process(listener) {
this.onProcess = listener;
return this;
}
done(listener) {
this.onDone = listener;
return this;
}
success(listener) {
this.onSuccess = listener;
return this;
}
failure(listener) {
this.onFailure = listener;
return this;
}
drain(listener) {
this.onDrain = listener;
return this;
}
}
// Usage
const job = ({ name, interval }) =>
new Promise((resolve, reject) => {
if (interval === 1200) {
setTimeout(reject, interval, new Error('Big error!'));
} else {
setTimeout(resolve, interval, name);
}
});
const queue = Queue.channels(3)
.process(job)
.done((error, res) => {
const { count } = queue;
const waiting = queue.waiting.length;
console.log(`Done | res: ${res}, err: ${error}`);
console.log(` | count: ${count}, waiting: ${waiting}`);
})
// .success((res) => void console.log(`Success: ${res}`))
// .failure((err) => void console.log(`Failure: ${err}`))
.drain(() => void console.log('Queue drain'));
for (let i = 0; i < 20; i++) {
if (i < 10) queue.add({ name: `Task${i}`, interval: 1000 });
else queue.add({ name: `Task${i}`, interval: i * 100 });
}