package transferqueue; sub new { my $class = shift; bless { conns => $_[0], }, $class; } sub start_transfer { my $self = shift; my $trans = bless [ $self ], transfer::; push @{$self->{wait}}, $trans; Scalar::Util::weaken($self->{wait}[-1]); print "ALLOC $Coro::current\n";#d# --$self->{conns}; $self->wake_next; $trans; } sub wake_next { my $self = shift; while ($self->{conns} >= 0 && @{$self->{wait}}) { print "WAKING some\n";#d# while(@{$self->{wait}}) { my $transfer = shift @{$self->{wait}}; if ($transfer) { print "WAKING $transfer\n";#d# $transfer->wake; last; } } } } sub waiters { map $_->[1], @{$_[0]{wait}}; } package transfer; use Coro::Timer (); sub wake { my $self = shift; $self->[2] = 1; ref $self->[1] and $self->[1]->ready; } sub try { my $self = shift; unless ($self->[2]) { my $timeout = Coro::Timer::timeout $_[0]; local $self->[1] = $Coro::current; Coro::schedule; print "WOKE $Coro::current\n" if $self->[2]; } return $self->[2]; } sub DESTROY { my $self = shift; eval { $self->[0]{conns}++; $self->[0]->wake_next; }; print "DESTROY $Coro::current $@\n";#d# } package conn; our %blockuri; our $blockref; sub read_blockuri { local *B; my %group; %blockuri = (); if (open B, ") { chomp; if (/^group\s+(\S+)\s+(.*)/i) { $group{$1} = [split /\s+/, $2]; } elsif (/^!([^\t]*)\t\s*(.*)/) { my $g = $1; my @r; for (split /\s+/, $2) { push @r, $group{$_} ? @{$group{$_}} : $_; } print "not($g) => (@r)\n"; push @{$blockuri{$_}}, $g for @r; push @blockuri, [qr/$g/i, \@r]; } elsif (/\S/) { print "blockuri: unparsable line: $_\n"; } } for (keys %blockuri) { my $qr = join ")|(?:", @{$blockuri{$_}}; $blockuri{$_} = qr{(?:$qr)}i; } } else { print "no blockuri\n"; } } sub read_blockref { local *B; my @blockref; if (open B, ") { chomp; if (/^([^\t]*)\t\s*(.*)/) { push @blockref, $1; } elsif (/\S/) { print "blockref: unparsable line: $_\n"; } } $blockref = join ")|(?:", @blockref; $blockref = qr{^(?:$blockref)}i; } else { print "no blockref\n"; $blockref = qr{^x^}; } } read_blockuri; read_blockref; use Tie::Cache; tie %whois_cache, Tie::Cache::, 32; sub access_check { my $self = shift; my $ref = $self->{h}{referer}; my $uri = $self->{path}; my %disallow; $self->err_block_referer if $self->{h}{referer} =~ $blockref; my $whois = $whois_cache{$self->{remote_addr}} ||= netgeo::ip_request($self->{remote_addr}); my $country = "XX"; if ($whois =~ /^\*cy: (\S+)/m) { $country = uc $1; } else { $self->slog(9, "no country($whois)"); } $self->{country} = $country; $self->err_block_country($whois) if $self->{path} =~ $blockuri{$country}; } 1;