Support for HUPing the server to clear the cache

Wake-one child support


git-svn-id: https://svn.perl.org/qpsmtpd/trunk@822 958fd67b-6ff1-0310-b445-bb7760255be9
This commit is contained in:
Matt Sergeant 2007-12-03 17:45:31 +00:00
parent 8700e5c6fe
commit 40c485e769

View File

@ -21,10 +21,11 @@ use IO::Socket;
use Carp; use Carp;
use POSIX qw(WNOHANG); use POSIX qw(WNOHANG);
use Getopt::Long; use Getopt::Long;
use List::Util qw(shuffle);
$|++; $|++;
use Socket qw(SOMAXCONN IPPROTO_TCP SO_KEEPALIVE TCP_NODELAY SOL_SOCKET); use Socket qw(SOMAXCONN IPPROTO_TCP SO_KEEPALIVE TCP_NODELAY SOL_SOCKET AF_UNIX SOCK_STREAM PF_UNSPEC);
$SIG{'PIPE'} = "IGNORE"; # handled manually $SIG{'PIPE'} = "IGNORE"; # handled manually
@ -79,6 +80,7 @@ GetOptions(
'pid-file=s' => \$PID_FILE, 'pid-file=s' => \$PID_FILE,
'd|detach' => \$DETACH, 'd|detach' => \$DETACH,
'h|help' => \&help, 'h|help' => \&help,
'config-port' => \$CONFIG_PORT,
) || help(); ) || help();
# detaint the commandline # detaint the commandline
@ -98,7 +100,11 @@ my $POLL = "with " . ($Danga::Socket::HaveEpoll ? "epoll()" :
my $SERVER; my $SERVER;
my $CONFIG_SERVER; my $CONFIG_SERVER;
use constant ACCEPTING => 1;
use constant RESTARTING => 2;
my %childstatus = (); my %childstatus = ();
my %childhandle = ();
if ($PID_FILE && -r $PID_FILE) { if ($PID_FILE && -r $PID_FILE) {
open PID, "<$PID_FILE" open PID, "<$PID_FILE"
or die "open_pidfile $PID_FILE: $!\n"; or die "open_pidfile $PID_FILE: $!\n";
@ -142,14 +148,23 @@ sub _fork {
sub spawn_child { sub spawn_child {
my $plugin_loader = shift || Qpsmtpd::SMTP->new; my $plugin_loader = shift || Qpsmtpd::SMTP->new;
socketpair(my $reader, my $writer, AF_UNIX, SOCK_STREAM, PF_UNSPEC) || die "Unable to create a pipe";
$reader->autoflush(1);
$writer->autoflush(1);
if (my $pid = _fork) { if (my $pid = _fork) {
$childstatus{$pid} = ACCEPTING;
$childhandle{$pid} = $writer;
return $pid; return $pid;
} }
$SIG{HUP} = $SIG{CHLD} = $SIG{INT} = $SIG{TERM} = 'DEFAULT'; $SIG{CHLD} = $SIG{INT} = $SIG{TERM} = 'DEFAULT';
$SIG{PIPE} = 'IGNORE'; $SIG{PIPE} = 'IGNORE';
$SIG{HUP} = sub { print "Child got SIGHUP\n" };
# sub { cmd_hup(); Qpsmtpd::PollServer->EventLoop(); }; # so we can HUP just one child
Qpsmtpd::PollServer->OtherFds(fileno($SERVER) => \&accept_handler); Qpsmtpd::PollServer->OtherFds(fileno($reader) => sub { command_handler($reader) });
$plugin_loader->run_hooks('post-fork'); $plugin_loader->run_hooks('post-fork');
@ -157,6 +172,15 @@ sub spawn_child {
exit; exit;
} }
sub sig_hup {
for my $writer (values %childhandle) {
print $writer "hup\n";
my $result = <$writer>;
}
$SIG{HUP} = \&sig_hup;
Qpsmtpd::PollServer->EventLoop();
}
sub sig_chld { sub sig_chld {
my $spawn_count = 0; my $spawn_count = 0;
while ( (my $child = waitpid(-1,WNOHANG)) > 0) { while ( (my $child = waitpid(-1,WNOHANG)) > 0) {
@ -167,13 +191,13 @@ sub sig_chld {
last unless $child > 0; last unless $child > 0;
print "SIGCHLD: child $child died\n"; print "SIGCHLD: child $child died\n";
delete $childstatus{$child}; delete $childstatus{$child};
delete $childhandle{$child};
$spawn_count++; $spawn_count++;
} }
if ($spawn_count) { if ($spawn_count) {
for (1..$spawn_count) { for (1..$spawn_count) {
# restart a new child if in poll server mode # restart a new child if in poll server mode
my $pid = spawn_child(); my $pid = spawn_child();
$childstatus{$pid} = 1;
} }
} }
$SIG{CHLD} = \&sig_chld; $SIG{CHLD} = \&sig_chld;
@ -250,26 +274,20 @@ sub run_as_server {
$SIG{INT} = $SIG{TERM} = \&HUNTSMAN; $SIG{INT} = $SIG{TERM} = \&HUNTSMAN;
if ($PROCS > 1) {
for (1..$PROCS) { for (1..$PROCS) {
my $pid = spawn_child($plugin_loader); my $pid = spawn_child($plugin_loader);
$childstatus{$pid} = 1;
} }
$plugin_loader->log(LOGDEBUG, "Listening on $PORT with $PROCS children $POLL"); $plugin_loader->log(LOGDEBUG, "Listening on $PORT with $PROCS children $POLL");
$SIG{'CHLD'} = \&sig_chld; $SIG{CHLD} = \&sig_chld;
sleep while (1); $SIG{HUP} = \&sig_hup;
}
else { Qpsmtpd::PollServer->OtherFds(
$plugin_loader->log(LOGDEBUG, "Listening on $PORT with single process $POLL"); fileno($SERVER) => \&accept_handler,
Qpsmtpd::PollServer->OtherFds(fileno($SERVER) => \&accept_handler, fileno($CONFIG_SERVER) => \&config_handler
fileno($CONFIG_SERVER) => \&config_handler,
); );
$plugin_loader->run_hooks('post-fork'); Qpsmtpd::PollServer->EventLoop;
while (1) {
Qpsmtpd::PollServer->EventLoop();
}
exit; exit;
}
} }
@ -291,8 +309,44 @@ sub config_handler {
return; return;
} }
# Accept all new connections # server is ready to accept - tell a child to accept().
sub accept_handler { sub accept_handler {
# pick a random child to tell to accept()
my $child = (shuffle keys %childstatus)[0];
if ($childstatus{$child} != ACCEPTING) {
# recurse...
return accept_handler() if %childstatus;
die "No children available";
}
my $writer = $childhandle{$child};
print $writer "accept\n";
my $result = <$writer>;
}
sub command_handler {
my $reader = shift;
chomp(my $command = <$reader>);
#print "Got command: $command\n";
my $real_command = "cmd_$command";
no strict 'refs';
my $result = $real_command->();
print $reader "$result\n";
}
sub cmd_hup {
# clear cache
#print "Clearing cache\n";
Qpsmtpd::clear_config_cache();
# should also reload modules... but can't do that yet.
return "ok";
}
# Accept all new connections
sub cmd_accept {
for (1 .. $NUMACCEPT) { for (1 .. $NUMACCEPT) {
return unless _accept_handler(); return unless _accept_handler();
} }
@ -303,6 +357,7 @@ sub accept_handler {
$NUMACCEPT = ACCEPT_MAX if $NUMACCEPT > ACCEPT_MAX; $NUMACCEPT = ACCEPT_MAX if $NUMACCEPT > ACCEPT_MAX;
$ACCEPT_RSET->cancel; $ACCEPT_RSET->cancel;
$ACCEPT_RSET = Danga::Socket->AddTimer(30, \&reset_num_accept); $ACCEPT_RSET = Danga::Socket->AddTimer(30, \&reset_num_accept);
return "ok";
} }
use Errno qw(EAGAIN EWOULDBLOCK); use Errno qw(EAGAIN EWOULDBLOCK);
@ -321,6 +376,7 @@ sub _accept_handler {
IO::Handle::blocking($csock, 0); IO::Handle::blocking($csock, 0);
#setsockopt($csock, IPPROTO_TCP, TCP_NODELAY, pack("l", 1)) or die; #setsockopt($csock, IPPROTO_TCP, TCP_NODELAY, pack("l", 1)) or die;
#print "Got connection\n";
my $client = Qpsmtpd::PollServer->new($csock); my $client = Qpsmtpd::PollServer->new($csock);
if ($PAUSED) { if ($PAUSED) {