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
| | # Copyright (C) 2016 all contributors <meta@public-inbox.org>
# License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
# event cleanups (currently for Danga::Socket)
package PublicInbox::EvCleanup;
use strict;
use warnings;
use base qw(Danga::Socket);
use fields qw(rd);
my $singleton;
my $asapq = [ [], undef ];
my $nextq = [ [], undef ];
my $laterq = [ [], undef ];
sub once_init () {
my $self = fields::new('PublicInbox::EvCleanup');
my ($r, $w);
pipe($r, $w) or die "pipe: $!";
$self->SUPER::new($w);
$self->{rd} = $r; # never read, since we never write..
$self;
}
sub _run_all ($) {
my ($q) = @_;
my $run = $q->[0];
$q->[0] = [];
$q->[1] = undef;
$_->() foreach @$run;
}
# ensure Danga::Socket::ToClose fires after timers fire
sub _asap_close () { $asapq->[1] ||= _asap_timer() }
sub _run_asap () { _run_all($asapq) }
sub _run_next () {
_run_all($nextq);
_asap_close();
}
sub _run_later () {
_run_all($laterq);
_asap_close();
}
# Called by Danga::Socket
sub event_write {
my ($self) = @_;
$self->watch_write(0);
_run_asap();
}
sub _asap_timer () {
$singleton ||= once_init();
$singleton->watch_write(1);
1;
}
sub asap ($) {
my ($cb) = @_;
push @{$asapq->[0]}, $cb;
$asapq->[1] ||= _asap_timer();
}
sub next_tick ($) {
my ($cb) = @_;
push @{$nextq->[0]}, $cb;
$nextq->[1] ||= Danga::Socket->AddTimer(0, *_run_next);
}
sub later ($) {
my ($cb) = @_;
push @{$laterq->[0]}, $cb;
$laterq->[1] ||= Danga::Socket->AddTimer(60, *_run_later);
}
END {
_run_asap();
_run_next();
_run_later();
}
1;
|