… | |
… | |
2 | |
2 | |
3 | use Coro; |
3 | use Coro; |
4 | use Coro::Handle; |
4 | use Coro::Handle; |
5 | use Coro::Socket; |
5 | use Coro::Socket; |
6 | use Event; |
6 | use Event; |
|
|
7 | use Time::HiRes 'time'; |
|
|
8 | |
|
|
9 | use Text::Abbrev; |
|
|
10 | |
|
|
11 | my $last_ts = time; |
|
|
12 | |
|
|
13 | my %complete; |
|
|
14 | my @commands = qw(quit squit refresh country restart block info); |
|
|
15 | |
|
|
16 | abbrev \%complete, @commands; |
7 | |
17 | |
8 | sub shell { |
18 | sub shell { |
9 | my $fh = shift; |
19 | my $fh = shift; |
10 | |
20 | |
11 | while (defined (print $fh "cmd> "), $_ = <$fh>) { |
21 | while (defined (print $fh "cmd> "), $_ = <$fh>) { |
12 | chomp; |
22 | s/\015?\012$//; |
13 | if (/quit/) { |
23 | if (s/^(\S+)\s*// && (my $cmd = $complete{$1})) { |
|
|
24 | if ($cmd eq "quit") { |
|
|
25 | print "bye bye.\n";#d# |
|
|
26 | last; |
|
|
27 | } elsif ($cmd eq "squit") { |
14 | Event::unloop; |
28 | Event::unloop; |
15 | } elsif (/info/) { |
29 | } elsif ($cmd eq "block") { |
16 | my @conn; |
30 | print "blocked '$_'\n";#d# |
17 | push @conn, values %$_ for values %conn::conn; |
31 | $conn::blocked{$_} = time + $::BLOCKTIME; |
|
|
32 | } elsif ($cmd eq "info") { |
|
|
33 | $::NOW = time+1e-6; |
|
|
34 | my @data; |
18 | for (values %conn::conn) { |
35 | for (values %conn::conn) { |
19 | for (values %$_) { |
36 | for (values %$_) { |
20 | next unless $_; |
37 | next unless $_; |
21 | print "$_: $_->{remote_addr} $_->{uri}\n"; |
38 | my $rate = sprintf "%.1f", $_->{written} / ($::NOW - $_->{time}); |
|
|
39 | push @data, "$_->{country}/$_->{remote_addr} $_->{written} $rate $_->{method} $_->{uri}\n"; |
|
|
40 | } |
22 | } |
41 | } |
|
|
42 | print $fh sort @data; |
|
|
43 | print $fh scalar@data, " ($::conns) connections\n";#d# |
|
|
44 | print $fh "$::written bytes written in the last ",$::NOW - $last_ts, " seconds\n"; |
|
|
45 | printf $fh "(%.1f bytes/s)\n", $::written / ($::NOW - $last_ts); |
|
|
46 | ($last_ts, $::written) = ($::NOW, 0); |
|
|
47 | } elsif ($cmd eq "refresh") { |
|
|
48 | read_blocklist; |
|
|
49 | } elsif ($cmd eq "restart") { |
|
|
50 | $::RESTART = 1; |
|
|
51 | unloop; |
|
|
52 | print $fh "restarting, cu!\n"; |
|
|
53 | last; |
|
|
54 | } elsif ($cmd eq "country") { |
|
|
55 | print $fh ip_request($_), "\n"; |
23 | } |
56 | } |
24 | } else { |
57 | } else { |
25 | print "try quit\n"; |
58 | print $fh "try one of @commands\n"; |
26 | } |
59 | } |
27 | } |
60 | } |
28 | } |
61 | } |
29 | |
62 | |
30 | # bind to tcp port |
63 | # bind to tcp port |
… | |
… | |
34 | LocalPort => $CMDSHELL_PORT, |
67 | LocalPort => $CMDSHELL_PORT, |
35 | ReuseAddr => 1, |
68 | ReuseAddr => 1, |
36 | Listen => 1, |
69 | Listen => 1, |
37 | or die "unable to bind cmdshell port: $!"; |
70 | or die "unable to bind cmdshell port: $!"; |
38 | |
71 | |
|
|
72 | push @listen_sockets, $port; |
|
|
73 | |
39 | async { |
74 | async { |
|
|
75 | while () { |
40 | async \&shell, $port->accept; |
76 | async \&shell, scalar $port->accept; |
|
|
77 | } |
41 | }; |
78 | }; |
42 | } |
79 | } |
43 | |
80 | |
44 | # bind to stdin (debug) |
81 | # bind to stdin (debug) |
45 | if (1) { |
82 | if (1) { |