]> Sergey Matveev's repositories - public-inbox.git/blob - lib/PublicInbox/EvCleanup.pm
nntp: allow and ignore empty commands
[public-inbox.git] / lib / PublicInbox / EvCleanup.pm
1 # Copyright (C) 2016-2018 all contributors <meta@public-inbox.org>
2 # License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
3
4 # event cleanups (currently for Danga::Socket)
5 package PublicInbox::EvCleanup;
6 use strict;
7 use warnings;
8 use base qw(Danga::Socket);
9 use fields qw(rd);
10 my $singleton;
11 my $asapq = [ [], undef ];
12 my $nextq = [ [], undef ];
13 my $laterq = [ [], undef ];
14
15 sub once_init () {
16         my $self = fields::new('PublicInbox::EvCleanup');
17         my ($r, $w);
18         pipe($r, $w) or die "pipe: $!";
19         $self->SUPER::new($w);
20         $self->{rd} = $r; # never read, since we never write..
21         $self;
22 }
23
24 sub _run_all ($) {
25         my ($q) = @_;
26
27         my $run = $q->[0];
28         $q->[0] = [];
29         $q->[1] = undef;
30         $_->() foreach @$run;
31 }
32
33 # ensure Danga::Socket::ToClose fires after timers fire
34 sub _asap_close () { $asapq->[1] ||= _asap_timer() }
35
36 sub _run_asap () { _run_all($asapq) }
37 sub _run_next () {
38         _run_all($nextq);
39         _asap_close();
40 }
41
42 sub _run_later () {
43         _run_all($laterq);
44         _asap_close();
45 }
46
47 # Called by Danga::Socket
48 sub event_write {
49         my ($self) = @_;
50         $self->watch_write(0);
51         _run_asap();
52 }
53
54 sub _asap_timer () {
55         $singleton ||= once_init();
56         $singleton->watch_write(1);
57         1;
58 }
59
60 sub asap ($) {
61         my ($cb) = @_;
62         push @{$asapq->[0]}, $cb;
63         $asapq->[1] ||= _asap_timer();
64 }
65
66 sub next_tick ($) {
67         my ($cb) = @_;
68         push @{$nextq->[0]}, $cb;
69         $nextq->[1] ||= Danga::Socket->AddTimer(0, *_run_next);
70 }
71
72 sub later ($) {
73         my ($cb) = @_;
74         push @{$laterq->[0]}, $cb;
75         $laterq->[1] ||= Danga::Socket->AddTimer(60, *_run_later);
76 }
77
78 END {
79         _run_asap();
80         _run_next();
81         _run_later();
82 }
83
84 1;