2001-02-13 23:41:00 +00:00
|
|
|
#!/usr/bin/perl -w
|
|
|
|
#
|
2017-09-08 13:39:09 -07:00
|
|
|
# Copyright (C) 2001, 2004-2007, 2012, 2016, 2017 Internet Systems Consortium, Inc. ("ISC")
|
2012-06-29 11:39:47 +10:00
|
|
|
#
|
2016-06-27 14:56:38 +10:00
|
|
|
# This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
# License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2007-06-19 23:47:24 +00:00
|
|
|
# $Id: stop.pl,v 1.12 2007/06/19 23:47:00 tbox Exp $
|
2001-02-13 23:41:00 +00:00
|
|
|
|
|
|
|
# Framework for stopping test servers
|
|
|
|
# Based on the type of server specified, signal the server to stop, wait
|
|
|
|
# briefly for it to die, and then kill it if it is still alive.
|
|
|
|
# If a server is specified, stop it. Otherwise, stop all servers for test.
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use Cwd 'abs_path';
|
2017-11-24 12:50:19 +00:00
|
|
|
use Getopt::Long;
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2017-11-24 12:50:19 +00:00
|
|
|
# Usage:
|
2018-01-25 12:29:31 +01:00
|
|
|
# perl stop.pl [--use-rndc [--port port]] test [server]
|
2001-02-13 23:41:00 +00:00
|
|
|
#
|
2017-11-24 12:50:19 +00:00
|
|
|
# --use-rndc Attempt to stop the server via the "rndc stop" command.
|
|
|
|
#
|
|
|
|
# --port port Only relevant if --use-rndc is specified, this sets the
|
|
|
|
# command port over which the attempt should be made. If
|
|
|
|
# not specified, port 9953 is used.
|
|
|
|
#
|
2017-12-15 18:42:52 +00:00
|
|
|
# test Name of the test directory.
|
2017-11-24 12:50:19 +00:00
|
|
|
#
|
2017-12-15 18:42:52 +00:00
|
|
|
# server Name of the server directory.
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2017-12-15 18:42:52 +00:00
|
|
|
my $usage = "usage: $0 [--use-rndc [--port port]] test-directory [server-directory]";
|
2001-03-08 02:34:01 +00:00
|
|
|
|
2017-11-24 12:50:19 +00:00
|
|
|
my $use_rndc = 0;
|
|
|
|
my $port = 9953;
|
|
|
|
GetOptions('use-rndc' => \$use_rndc, 'port=i' => \$port) or die "$usage\n";
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2001-03-01 02:35:13 +00:00
|
|
|
my $errors = 0;
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2017-11-24 12:50:19 +00:00
|
|
|
my $test = $ARGV[0];
|
|
|
|
my $server = $ARGV[1];
|
2001-03-08 02:34:01 +00:00
|
|
|
die "$usage\n" unless defined($test);
|
|
|
|
die "No test directory: \"$test\"\n" unless (-d $test);
|
2006-03-03 00:43:35 +00:00
|
|
|
die "No server directory: \"$server\"\n" if (defined($server) && !-d "$test/$server");
|
2017-12-15 18:42:52 +00:00
|
|
|
|
2001-02-13 23:41:00 +00:00
|
|
|
# Global variables
|
2001-03-08 02:34:01 +00:00
|
|
|
my $testdir = abs_path($test);
|
|
|
|
my @servers;
|
2001-02-13 23:41:00 +00:00
|
|
|
|
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
# Determine which servers need to be stopped.
|
|
|
|
if (defined $server) {
|
|
|
|
@servers = ($server);
|
2001-02-13 23:41:00 +00:00
|
|
|
} else {
|
2001-03-08 02:34:01 +00:00
|
|
|
local *DIR;
|
|
|
|
opendir DIR, $testdir or die "$testdir: $!\n";
|
2001-02-14 23:57:33 +00:00
|
|
|
my @files = sort readdir DIR;
|
2001-02-13 23:41:00 +00:00
|
|
|
closedir DIR;
|
|
|
|
|
|
|
|
my @ns = grep /^ns[0-9]*$/, @files;
|
|
|
|
my @ans = grep /^ans[0-9]*$/, @files;
|
2001-03-08 02:34:01 +00:00
|
|
|
|
2017-09-08 13:39:09 -07:00
|
|
|
push @servers, @ns, @ans;
|
2001-03-08 02:34:01 +00:00
|
|
|
}
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
|
|
|
|
# Stop the server(s), pass 1: rndc.
|
|
|
|
if ($use_rndc) {
|
|
|
|
foreach my $server (grep /^ns/, @servers) {
|
|
|
|
stop_rndc($server);
|
2001-02-13 23:41:00 +00:00
|
|
|
}
|
2001-03-08 02:34:01 +00:00
|
|
|
|
2005-09-28 04:36:06 +00:00
|
|
|
wait_for_servers(30, grep /^ns/, @servers);
|
2001-03-08 02:34:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# Pass 2: SIGTERM
|
|
|
|
foreach my $server (@servers) {
|
|
|
|
stop_signal($server, "TERM");
|
|
|
|
}
|
|
|
|
|
2005-09-28 04:36:06 +00:00
|
|
|
wait_for_servers(60, @servers);
|
2001-03-08 02:34:01 +00:00
|
|
|
|
2001-09-20 18:33:51 +00:00
|
|
|
# Pass 3: SIGABRT
|
2001-03-08 02:34:01 +00:00
|
|
|
foreach my $server (@servers) {
|
2001-09-20 18:33:51 +00:00
|
|
|
stop_signal($server, "ABRT");
|
2001-02-13 23:41:00 +00:00
|
|
|
}
|
|
|
|
|
2001-03-01 02:35:13 +00:00
|
|
|
exit($errors ? 1 : 0);
|
|
|
|
|
2001-02-13 23:41:00 +00:00
|
|
|
# Subroutines
|
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
# Return the full path to a given server's PID file.
|
|
|
|
sub server_pid_file {
|
|
|
|
my($server) = @_;
|
2001-02-13 23:41:00 +00:00
|
|
|
|
|
|
|
my $pid_file;
|
|
|
|
if ($server =~ /^ns/) {
|
|
|
|
$pid_file = "named.pid";
|
|
|
|
} elsif ($server =~ /^ans/) {
|
|
|
|
$pid_file = "ans.pid";
|
|
|
|
} else {
|
|
|
|
print "I:Unknown server type $server\n";
|
|
|
|
exit 1;
|
|
|
|
}
|
2001-03-08 02:34:01 +00:00
|
|
|
$pid_file = "$testdir/$server/$pid_file";
|
|
|
|
}
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
# Read a PID.
|
|
|
|
sub read_pid {
|
|
|
|
my($pid_file) = @_;
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
local *FH;
|
|
|
|
my $result = open FH, "< $pid_file";
|
|
|
|
if (!$result) {
|
|
|
|
print "I:$pid_file: $!\n";
|
|
|
|
unlink $pid_file;
|
|
|
|
return;
|
2001-02-13 23:41:00 +00:00
|
|
|
}
|
2001-03-08 02:34:01 +00:00
|
|
|
|
|
|
|
my $pid = <FH>;
|
|
|
|
chomp($pid);
|
|
|
|
return $pid;
|
2001-02-13 23:41:00 +00:00
|
|
|
}
|
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
# Stop a named process with rndc.
|
|
|
|
sub stop_rndc {
|
|
|
|
my($server) = @_;
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
return unless ($server =~ /^ns(\d+)$/);
|
|
|
|
my $ip = "10.53.0.$1";
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
# Ugly, but should work.
|
2017-11-24 12:50:19 +00:00
|
|
|
system("$ENV{RNDC} -c ../common/rndc.conf -s $ip -p $port stop | sed 's/^/I:$server /'");
|
2001-03-08 02:34:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Stop a server by sending a signal to it.
|
|
|
|
sub stop_signal {
|
|
|
|
my($server, $sig) = @_;
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2001-03-08 02:34:01 +00:00
|
|
|
my $pid_file = server_pid_file($server);
|
|
|
|
return unless -f $pid_file;
|
|
|
|
|
|
|
|
my $pid = read_pid($pid_file);
|
|
|
|
return unless defined($pid);
|
2001-02-13 23:41:00 +00:00
|
|
|
|
2001-09-20 18:33:51 +00:00
|
|
|
if ($sig eq 'ABRT') {
|
|
|
|
print "I:$server didn't die when sent a SIGTERM\n";
|
|
|
|
$errors++;
|
|
|
|
}
|
2001-03-08 02:34:01 +00:00
|
|
|
|
2016-10-19 17:18:42 +02:00
|
|
|
my $result;
|
|
|
|
if ($^O eq 'cygwin') {
|
|
|
|
$result = system("/bin/kill -f -$sig $pid");
|
2001-02-13 23:41:00 +00:00
|
|
|
unlink $pid_file;
|
2016-10-19 17:18:42 +02:00
|
|
|
if ($result != 0) {
|
|
|
|
print "I:$server died before a SIG$sig was sent\n";
|
|
|
|
$errors++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$result = kill $sig, $pid;
|
|
|
|
if (!$result) {
|
|
|
|
print "I:$server died before a SIG$sig was sent\n";
|
|
|
|
unlink $pid_file;
|
|
|
|
$errors++;
|
|
|
|
}
|
2001-02-13 23:41:00 +00:00
|
|
|
}
|
2001-03-08 02:34:01 +00:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub wait_for_servers {
|
|
|
|
my($timeout, @servers) = @_;
|
|
|
|
|
|
|
|
my @pid_files = grep { defined($_) }
|
|
|
|
map { server_pid_file($_) } @servers;
|
|
|
|
|
|
|
|
while ($timeout > 0 && @pid_files > 0) {
|
|
|
|
@pid_files = grep { -f $_ } @pid_files;
|
|
|
|
sleep 1 if (@pid_files > 0);
|
|
|
|
$timeout--;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
2001-02-13 23:41:00 +00:00
|
|
|
}
|