Large number of patches from Brian Grossman to fix a number of bugs

Implement connection timeout


git-svn-id: https://svn.perl.org/qpsmtpd/branches/high_perf@413 958fd67b-6ff1-0310-b445-bb7760255be9
This commit is contained in:
Matt Sergeant 2005-05-09 13:43:40 +00:00
parent 12d9fa8311
commit 8dad7435e5
9 changed files with 101 additions and 32 deletions

View File

@ -6,7 +6,8 @@ use fields qw(line closing disable_read can_read_mode);
use Time::HiRes ();
# 30 seconds max timeout!
sub max_idle_time { 30 }
sub max_idle_time { 30 }
sub max_connect_time { 1200 }
sub new {
my Danga::Client $self = shift;
@ -45,7 +46,7 @@ sub can_read {
my Danga::Client $self = shift;
my ($timeout) = @_;
my $end = Time::HiRes::time() + $timeout;
warn("Calling can-read\n");
# warn("Calling can-read\n");
$self->{can_read_mode} = 1;
if (!length($self->{line})) {
my $old = $self->watch_read();
@ -61,7 +62,7 @@ sub can_read {
$self->SetPostLoopCallback(sub { $self->have_line ? 0 : 1 });
return if $self->{closing};
$self->{alive_time} = time;
warn("can_read returning for '$self->{line}'\n");
# warn("can_read returning for '$self->{line}'\n");
return 1 if length($self->{line});
return;
}

View File

@ -286,6 +286,9 @@ sub event_read {
#$self->{timeout}{$id} = time();
}
elsif ($err eq "NOERROR") {
$asker->run_callback($err, $query);
}
elsif($err) {
print("error: $err\n");
$asker->run_callback($err, $query);

View File

@ -24,7 +24,7 @@ use vars qw{$VERSION};
$VERSION = do { my @r = (q$Revision: 1.4 $ =~ /\d+/g); sprintf "%d."."%02d" x $#r, @r };
use fields qw(sock fd write_buf write_buf_offset write_buf_size
read_push_back
read_push_back post_loop_callback
closed event_watch debug_level);
use Errno qw(EINPROGRESS EWOULDBLOCK EISCONN
@ -307,9 +307,21 @@ sub PostEventLoop {
# now we can close sockets that wanted to close during our event processing.
# (we didn't want to close them during the loop, as we didn't want fd numbers
# being reused and confused during the event loop)
$_->close while ($_ = shift @ToClose);
while(my $j = shift @ToClose) {
$j->[1]->close();
$j->[0]->{closing} = 0;
}
# now we're at the very end, call callback if defined
# now we're at the very end, call per-connection callbacks if defined
for my $fd (%DescriptorMap) {
my $pob = $DescriptorMap{$fd};
if( defined $pob->{post_loop_callback} ) {
return unless $pob->{post_loop_callback}->(\%DescriptorMap, \%OtherFds);
}
}
# now we're at the very end, call global callback if defined
if (defined $PostLoopCallback) {
return $PostLoopCallback->(\%DescriptorMap, \%OtherFds);
}
@ -401,6 +413,7 @@ sub new {
$self->{write_buf_size} = 0;
$self->{closed} = 0;
$self->{read_push_back} = [];
$self->{post_loop_callback} = undef;
$self->{event_watch} = POLLERR|POLLHUP|POLLNVAL;
@ -472,7 +485,7 @@ sub close {
# defer closing the actual socket until the event loop is done
# processing this round of events. (otherwise we might reuse fds)
push @ToClose, $sock;
push @ToClose, [$self,$sock];
return 0;
}
@ -785,7 +798,18 @@ sub as_string {
### be passed two parameters: \%DescriptorMap, \%OtherFds.
sub SetPostLoopCallback {
my ($class, $ref) = @_;
$PostLoopCallback = (defined $ref && ref $ref eq 'CODE') ? $ref : undef;
if(ref $class) {
my Danga::Socket $self = $class;
if( defined $ref && ref $ref eq 'CODE' ) {
$self->{PostLoopCallback} = $ref;
}
else {
delete $self->{PostLoopCallback};
}
}
else {
$PostLoopCallback = (defined $ref && ref $ref eq 'CODE') ? $ref : undef;
}
}
#####################################################################

View File

@ -24,22 +24,46 @@ sub new {
return $self;
}
sub ticker {
my Danga::TimeoutSocket $self = shift;
my $now = time;
if ($now - 15 > $last_cleanup) {
$last_cleanup = $now;
_do_cleanup($now);
}
}
# overload these in a subclass
sub max_idle_time { 0 }
sub max_connect_time { 0 }
sub _do_cleanup {
my $now = shift;
my $sf = __PACKAGE__->get_sock_ref;
my %max_age; # classname -> max age (0 means forever)
my %max_connect; # classname -> max connect time
my @to_close;
while (my $k = each %$sf) {
my Danga::TimeoutSocket $v = $sf->{$k};
my $ref = ref $v;
next unless $v->isa('Danga::TimeoutSocket');
unless (defined $max_age{$ref}) {
$max_age{$ref} = $ref->max_idle_time || 0;
$max_age{$ref} = $ref->max_idle_time || 0;
$max_connect{$ref} = $ref->max_connect_time || 0;
}
next unless $max_age{$ref};
if ($v->{alive_time} < $now - $max_age{$ref}) {
push @to_close, $v;
if (my $t = $max_connect{$ref}) {
if ($v->{create_time} < $now - $t) {
push @to_close, $v;
next;
}
}
if (my $t = $max_age{$ref}) {
if ($v->{alive_time} < $now - $t) {
push @to_close, $v;
}
}
}

View File

@ -15,6 +15,7 @@ use fields qw(
_transaction
_test_mode
_extras
other_fds
);
my $PROMPT = "Enter command: ";
@ -130,6 +131,16 @@ sub cmd_pause {
return "PAUSED";
}
sub cmd_continue {
my $self = shift;
my $other_fds = $self->{other_fds};
$self->OtherFds( %$other_fds );
%$other_fds = ();
return "UNPAUSED";
}
sub cmd_status {
my $self = shift;
@ -173,7 +184,7 @@ sub cmd_status {
}
}
$output .= "Curr Connections: $current_connections\n".
$output .= "Curr Connections: $current_connections / $::MAXconn\n".
"Curr DNS Queries: $current_dns";
return $output;
@ -206,7 +217,7 @@ sub cmd_list {
}
}
foreach my $item (@all) {
$list .= sprintf("%x : %s [%s] Connected %0.2fs\n", @$item);
$list .= sprintf("%x : %s [%s] Connected %0.2fs\n", map { defined()?$_:'' } @$item);
}
return $list;

View File

@ -108,7 +108,7 @@ sub compile {
}
close F;
my $line = "\n#line 1 $file\n";
my $line = "\n#line 0 $file\n";
if ($test_mode) {
if (open(F, "t/plugin_tests/$plugin")) {

View File

@ -32,6 +32,9 @@ use Socket qw(inet_aton AF_INET CRLF);
use Time::HiRes qw(time);
use strict;
sub max_idle_time { 60 }
sub max_connect_time { 1200 }
sub input_sock {
my $self = shift;
@_ and $self->{input_sock} = shift;
@ -91,7 +94,7 @@ sub process_line {
if ($::DEBUG > 1) { print "$$:".($self+0)."C($self->{mode}): $line"; }
local $SIG{ALRM} = sub {
my ($pkg, $file, $line) = caller();
die "ALARM: $pkg, $file, $line";
die "ALARM: ($self->{mode}) $pkg, $file, $line";
};
my $prev = alarm(2); # must process a command in < 2 seconds
eval { $self->_process_line($line) };
@ -169,6 +172,7 @@ sub start_conversation {
my ($ip, $port) = split(':', $self->peer_addr_string);
$conn->remote_ip($ip);
$conn->remote_port($port);
$conn->remote_info("[$ip]");
Danga::DNS->new(
client => $self,
# NB: Setting remote_info to the same as remote_host

View File

@ -44,8 +44,6 @@ and terminating the SMTP connection.
=cut
use Time::HiRes ();
use warnings;
use strict;
@ -70,25 +68,19 @@ sub register {
sub connect_handler {
my ($self, $transaction) = @_;
my $qp = $self->qp;
my $end = Time::HiRes::time + $self->{_args}->{'wait'} ;
my $time;
for( $time = Time::HiRes::time; $time < $end && !length($qp->{line}) ; $time = Time::HiRes::time ) {
$qp->can_read($end-$time);
}
my $earlytalker = 0;
$earlytalker = 1 if $time < $end ;
if ($earlytalker) {
if ($self->qp->can_read($self->{_args}->{'wait'})) {
$self->log(LOGNOTICE, 'remote host started talking before we said hello');
if ($self->{_args}->{'defer-reject'}) {
$self->connection->notes('earlytalker', 1);
} else {
$self->connection->notes('earlytalker', 1);
}
else {
my $msg = 'Connecting host started transmitting before SMTP greeting';
return (DENY,$msg) if $self->{_args}->{'action'} eq 'deny';
return (DENYSOFT,$msg) if $self->{_args}->{'action'} eq 'denysoft';
}
} else {
}
else {
$self->log(LOGINFO, 'remote host said nothing spontaneous, proceeding');
}
return DECLINED;

12
qpsmtpd
View File

@ -199,6 +199,7 @@ sub run_as_inetd {
}
sub run_as_server {
local $::MAXconn = $MAXCONN;
# establish SERVER socket, bind and listen.
$SERVER = IO::Socket::INET->new(LocalPort => $PORT,
LocalAddr => $LOCALADDR,
@ -290,11 +291,19 @@ sub config_handler {
# Accept a new connection
sub accept_handler {
my $running = scalar keys %childstatus;
my $running;
if( $LineMode ) {
$running = scalar keys %childstatus;
}
else {
my $descriptors = Danga::Client->DescriptorMap;
$running = scalar keys %$descriptors;
}
while ($running >= $MAXCONN) {
::log(LOGINFO,"Too many connections: $running >= $MAXCONN.");
return;
}
++$running if $LineMode; # count self
my $csock = $SERVER->accept();
if (!$csock) {
@ -341,6 +350,7 @@ sub accept_handler {
$client->close;
return;
}
::log(LOGINFO, "accepted connection $running/$MAXCONN ($num_conn/$MAXCONNIP) from $rem_ip");
}
my $rc = $client->start_conversation;