Skip to content

Commit 71ffd10

Browse files
committed
test_runner: implement PriorityQueue for timers
Signed-off-by: Erick Wendel <[email protected]>
1 parent e54a162 commit 71ffd10

File tree

2 files changed

+46
-20
lines changed

2 files changed

+46
-20
lines changed

lib/internal/test_runner/mock/fake_timers.js

+42-16
Original file line numberDiff line numberDiff line change
@@ -2,14 +2,30 @@
22

33
const {
44
DateNow,
5-
SafeMap,
6-
Symbol,
5+
SafeSet,
76
globalThis,
87
} = primordials;
98

9+
const PriorityQueue = require('internal/priority_queue');
10+
11+
function compareTimersLists(a, b) {
12+
const expiryDiff = a.runAt - b.runAt;
13+
if (expiryDiff === 0) {
14+
if (a.id < b.id)
15+
return -1;
16+
if (a.id > b.id)
17+
return 1;
18+
}
19+
return expiryDiff;
20+
}
21+
22+
function setPosition(node, pos) {
23+
node.priorityQueuePosition = pos;
24+
}
1025
class Timers {
26+
#currentTimer = 1;
1127
constructor() {
12-
this.timers = new SafeMap();
28+
this.timers = new PriorityQueue(compareTimersLists, setPosition);
1329

1430
this.setTimeout = this.#createTimer.bind(this, false);
1531
this.clearTimeout = this.#clearTimer.bind(this);
@@ -18,20 +34,20 @@ class Timers {
1834
}
1935

2036
#createTimer(isInterval, callback, delay, ...args) {
21-
const timerId = Symbol('kTimerId');
22-
const timer = {
37+
const timerId = this.#currentTimer++;
38+
this.timers.insert({
2339
id: timerId,
2440
callback,
2541
runAt: DateNow() + delay,
2642
interval: isInterval,
2743
args,
28-
};
29-
this.timers.set(timerId, timer);
44+
});
45+
3046
return timerId;
3147
}
3248

33-
#clearTimer(timerId) {
34-
this.timers.delete(timerId);
49+
#clearTimer(position) {
50+
this.timers.removeAt(position);
3551
}
3652

3753
}
@@ -56,18 +72,27 @@ class FakeTimers {
5672

5773
this.now += time;
5874
const timers = this.fakeTimers.timers;
75+
const alreadyProcessed = new SafeSet();
76+
while (true) {
77+
const timer = timers.peek();
5978

60-
for (const timer of timers.values()) {
79+
if (!timer) {
80+
alreadyProcessed.clear();
81+
break;
82+
}
6183

62-
if (!(this.now >= timer.runAt)) continue;
84+
if (alreadyProcessed.has(timer)) break;
85+
alreadyProcessed.add(timer);
6386

87+
if (!(this.now >= timer.runAt)) continue;
6488
timer.callback(...timer.args);
65-
if (timer.interval) {
66-
timer.runAt = this.now + (timer.runAt - this.now) % timer.args[0];
67-
continue;
68-
}
6989

70-
timers.delete(timer.id);
90+
// if (timer.interval) {
91+
// timer.runAt = this.now + (timer.runAt - this.now) % timer.args[0];
92+
// continue;
93+
// }
94+
95+
timers.removeAt(alreadyProcessed.size - 1);
7196
}
7297
}
7398

@@ -89,6 +114,7 @@ class FakeTimers {
89114
globalThis.setInterval = this.fakeTimers.setInterval;
90115
globalThis.clearInterval = this.fakeTimers.clearInterval;
91116

117+
// this.#dispatchPendingTimers()
92118
}
93119

94120
reset() {

test/parallel/test-runner-mocking-fake-timers.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -23,13 +23,13 @@ describe('Faketimers Test Suite', () => {
2323
});
2424

2525
it('should advance in time and trigger timers when calling the .tick function multiple times', (t) => {
26-
fakeTimers.enable();
27-
const fn = mock.fn();
26+
t.mock.fakeTimers.enable();
27+
const fn = t.mock.fn();
2828

2929
global.setTimeout(fn, 2000);
3030

31-
fakeTimers.tick(1000);
32-
fakeTimers.tick(1000);
31+
t.mock.fakeTimers.tick(1000);
32+
t.mock.fakeTimers.tick(1000);
3333

3434
assert.strictEqual(fn.mock.callCount(), 1);
3535
});

0 commit comments

Comments
 (0)