2
2
3
3
const {
4
4
DateNow,
5
- SafeMap,
6
- Symbol,
5
+ SafeSet,
7
6
globalThis,
8
7
} = primordials ;
9
8
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
+ }
10
25
class Timers {
26
+ #currentTimer = 1 ;
11
27
constructor ( ) {
12
- this . timers = new SafeMap ( ) ;
28
+ this . timers = new PriorityQueue ( compareTimersLists , setPosition ) ;
13
29
14
30
this . setTimeout = this . #createTimer. bind ( this , false ) ;
15
31
this . clearTimeout = this . #clearTimer. bind ( this ) ;
@@ -18,20 +34,20 @@ class Timers {
18
34
}
19
35
20
36
#createTimer( isInterval , callback , delay , ...args ) {
21
- const timerId = Symbol ( 'kTimerId' ) ;
22
- const timer = {
37
+ const timerId = this . #currentTimer ++ ;
38
+ this . timers . insert ( {
23
39
id : timerId ,
24
40
callback,
25
41
runAt : DateNow ( ) + delay ,
26
42
interval : isInterval ,
27
43
args,
28
- } ;
29
- this . timers . set ( timerId , timer ) ;
44
+ } ) ;
45
+
30
46
return timerId ;
31
47
}
32
48
33
- #clearTimer( timerId ) {
34
- this . timers . delete ( timerId ) ;
49
+ #clearTimer( position ) {
50
+ this . timers . removeAt ( position ) ;
35
51
}
36
52
37
53
}
@@ -56,18 +72,27 @@ class FakeTimers {
56
72
57
73
this . now += time ;
58
74
const timers = this . fakeTimers . timers ;
75
+ const alreadyProcessed = new SafeSet ( ) ;
76
+ while ( true ) {
77
+ const timer = timers . peek ( ) ;
59
78
60
- for ( const timer of timers . values ( ) ) {
79
+ if ( ! timer ) {
80
+ alreadyProcessed . clear ( ) ;
81
+ break ;
82
+ }
61
83
62
- if ( ! ( this . now >= timer . runAt ) ) continue ;
84
+ if ( alreadyProcessed . has ( timer ) ) break ;
85
+ alreadyProcessed . add ( timer ) ;
63
86
87
+ if ( ! ( this . now >= timer . runAt ) ) continue ;
64
88
timer . callback ( ...timer . args ) ;
65
- if ( timer . interval ) {
66
- timer . runAt = this . now + ( timer . runAt - this . now ) % timer . args [ 0 ] ;
67
- continue ;
68
- }
69
89
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 ) ;
71
96
}
72
97
}
73
98
@@ -89,6 +114,7 @@ class FakeTimers {
89
114
globalThis . setInterval = this . fakeTimers . setInterval ;
90
115
globalThis . clearInterval = this . fakeTimers . clearInterval ;
91
116
117
+ // this.#dispatchPendingTimers()
92
118
}
93
119
94
120
reset ( ) {
0 commit comments