Commit 803466b6 authored by Tom Lane's avatar Tom Lane

Avoid picking already-bound TCP ports in kerberos and ldap test suites.

src/test/kerberos and src/test/ldap need to run a private authentication
server of the relevant type, for which they need a free TCP port.
They were just picking a random port number in 48K-64K, which works
except when something's already using the particular port.  Notably,
the probability of failure rises dramatically if one simply runs those
tests in a tight loop, because each test cycle leaves behind a bunch of
high ports that are transiently in TIME_WAIT state.

To fix, split out the code that PostgresNode.pm already had for
identifying a free TCP port number, so that it can be invoked to choose
a port for the KDC or LDAP server.  This isn't 100% bulletproof, since
conceivably something else on the machine could grab the port between
the time we check and the time we actually start the server.  But that's
a pretty short window, so in practice this should be good enough.

Back-patch to v11 where these test suites were added.

Patch by me, reviewed by Andrew Dunstan.

Discussion: https://postgr.es/m/3397.1564872168@sss.pgh.pa.us
parent 489247b0
...@@ -69,7 +69,7 @@ my $krb5_conf = "${TestLib::tmp_check}/krb5.conf"; ...@@ -69,7 +69,7 @@ my $krb5_conf = "${TestLib::tmp_check}/krb5.conf";
my $kdc_conf = "${TestLib::tmp_check}/kdc.conf"; my $kdc_conf = "${TestLib::tmp_check}/kdc.conf";
my $krb5_log = "${TestLib::tmp_check}/krb5libs.log"; my $krb5_log = "${TestLib::tmp_check}/krb5libs.log";
my $kdc_log = "${TestLib::tmp_check}/krb5kdc.log"; my $kdc_log = "${TestLib::tmp_check}/krb5kdc.log";
my $kdc_port = int(rand() * 16384) + 49152; my $kdc_port = get_free_port();
my $kdc_datadir = "${TestLib::tmp_check}/krb5kdc"; my $kdc_datadir = "${TestLib::tmp_check}/krb5kdc";
my $kdc_pidfile = "${TestLib::tmp_check}/krb5kdc.pid"; my $kdc_pidfile = "${TestLib::tmp_check}/krb5kdc.pid";
my $keytab = "${TestLib::tmp_check}/krb5.keytab"; my $keytab = "${TestLib::tmp_check}/krb5.keytab";
......
...@@ -55,7 +55,7 @@ my $slapd_pidfile = "${TestLib::tmp_check}/slapd.pid"; ...@@ -55,7 +55,7 @@ my $slapd_pidfile = "${TestLib::tmp_check}/slapd.pid";
my $slapd_logfile = "${TestLib::tmp_check}/slapd.log"; my $slapd_logfile = "${TestLib::tmp_check}/slapd.log";
my $ldap_conf = "${TestLib::tmp_check}/ldap.conf"; my $ldap_conf = "${TestLib::tmp_check}/ldap.conf";
my $ldap_server = 'localhost'; my $ldap_server = 'localhost';
my $ldap_port = int(rand() * 16384) + 49152; my $ldap_port = get_free_port();
my $ldaps_port = $ldap_port + 1; my $ldaps_port = $ldap_port + 1;
my $ldap_url = "ldap://$ldap_server:$ldap_port"; my $ldap_url = "ldap://$ldap_server:$ldap_port";
my $ldaps_url = "ldaps://$ldap_server:$ldaps_port"; my $ldaps_url = "ldaps://$ldap_server:$ldaps_port";
......
...@@ -63,6 +63,9 @@ PostgresNode - class representing PostgreSQL server instance ...@@ -63,6 +63,9 @@ PostgresNode - class representing PostgreSQL server instance
# Stop the server # Stop the server
$node->stop('fast'); $node->stop('fast');
# Find a free, unprivileged TCP port to bind some other service to
my $port = get_free_port();
=head1 DESCRIPTION =head1 DESCRIPTION
PostgresNode contains a set of routines able to work on a PostgreSQL node, PostgresNode contains a set of routines able to work on a PostgreSQL node,
...@@ -102,6 +105,7 @@ use Scalar::Util qw(blessed); ...@@ -102,6 +105,7 @@ use Scalar::Util qw(blessed);
our @EXPORT = qw( our @EXPORT = qw(
get_new_node get_new_node
get_free_port
); );
our ($use_tcp, $test_localhost, $test_pghost, $last_host_assigned, our ($use_tcp, $test_localhost, $test_pghost, $last_host_assigned,
...@@ -1071,9 +1075,68 @@ sub get_new_node ...@@ -1071,9 +1075,68 @@ sub get_new_node
my $class = 'PostgresNode'; my $class = 'PostgresNode';
$class = shift if scalar(@_) % 2 != 1; $class = shift if scalar(@_) % 2 != 1;
my ($name, %params) = @_; my ($name, %params) = @_;
my $port_is_forced = defined $params{port};
my $found = $port_is_forced; # Select a port.
my $port = $port_is_forced ? $params{port} : $last_port_assigned; my $port;
if (defined $params{port})
{
$port = $params{port};
}
else
{
# When selecting a port, we look for an unassigned TCP port number,
# even if we intend to use only Unix-domain sockets. This is clearly
# necessary on $use_tcp (Windows) configurations, and it seems like a
# good idea on Unixen as well.
$port = get_free_port();
}
# Select a host.
my $host = $test_pghost;
if ($params{own_host})
{
if ($use_tcp)
{
$last_host_assigned++;
$last_host_assigned > 254 and BAIL_OUT("too many own_host nodes");
$host = '127.0.0.' . $last_host_assigned;
}
else
{
$host = "$test_pghost/$name"; # Assume $name =~ /^[-_a-zA-Z0-9]+$/
mkdir $host;
}
}
# Lock port number found by creating a new node
my $node = $class->new($name, $host, $port);
# Add node to list of nodes
push(@all_nodes, $node);
return $node;
}
=pod
=item get_free_port()
Locate an unprivileged (high) TCP port that's not currently bound to
anything. This is used by get_new_node, and is also exported for use
by test cases that need to start other, non-Postgres servers.
Ports assigned to existing PostgresNode objects are automatically
excluded, even if those servers are not currently running.
XXX A port available now may become unavailable by the time we start
the desired service.
=cut
sub get_free_port
{
my $found = 0;
my $port = $last_port_assigned;
while ($found == 0) while ($found == 0)
{ {
...@@ -1090,63 +1153,38 @@ sub get_new_node ...@@ -1090,63 +1153,38 @@ sub get_new_node
$found = 0 if ($node->port == $port); $found = 0 if ($node->port == $port);
} }
# Check to see if anything else is listening on this TCP port. This # Check to see if anything else is listening on this TCP port.
# is *necessary* on $use_tcp (Windows) configurations. Seek a port # Seek a port available for all possible listen_addresses values,
# available for all possible listen_addresses values, for own_host # so callers can harness this port for the widest range of purposes.
# nodes and so the caller can harness this port for the widest range # The 0.0.0.0 test achieves that for post-2006 Cygwin, which
# of purposes. The 0.0.0.0 test achieves that for post-2006 Cygwin, # automatically sets SO_EXCLUSIVEADDRUSE. The same holds for MSYS (a
# which automatically sets SO_EXCLUSIVEADDRUSE. The same holds for # Cygwin fork). Testing 0.0.0.0 is insufficient for Windows native
# MSYS (a Cygwin fork). Testing 0.0.0.0 is insufficient for Windows # Perl (https://stackoverflow.com/a/14388707), so we also test
# native Perl (https://stackoverflow.com/a/14388707), so we also test
# individual addresses. # individual addresses.
# #
# This seems like a good idea on Unixen as well, even though we don't # On non-Linux, non-Windows kernels, binding to 127.0.0/24 addresses
# ask the postmaster to open a TCP port on Unix. On Non-Linux, # other than 127.0.0.1 might fail with EADDRNOTAVAIL. Binding to
# non-Windows kernels, binding to 127.0.0.1/24 addresses other than # 0.0.0.0 is unnecessary on non-Windows systems.
# 127.0.0.1 might fail with EADDRNOTAVAIL. Binding to 0.0.0.0 is
# unnecessary on non-Windows systems.
#
# XXX A port available now may become unavailable by the time we start
# the postmaster.
if ($found == 1) if ($found == 1)
{ {
foreach my $addr (qw(127.0.0.1), foreach my $addr (qw(127.0.0.1),
$use_tcp ? qw(127.0.0.2 127.0.0.3 0.0.0.0) : ()) $use_tcp ? qw(127.0.0.2 127.0.0.3 0.0.0.0) : ())
{ {
can_bind($addr, $port) or $found = 0; if (!can_bind($addr, $port))
{
$found = 0;
last;
}
} }
} }
} }
print "# Found port $port\n"; print "# Found port $port\n";
# Select a host. # Update port for next time
my $host = $test_pghost; $last_port_assigned = $port;
if ($params{own_host})
{
if ($use_tcp)
{
$last_host_assigned++;
$last_host_assigned > 254 and BAIL_OUT("too many own_host nodes");
$host = '127.0.0.' . $last_host_assigned;
}
else
{
$host = "$test_pghost/$name"; # Assume $name =~ /^[-_a-zA-Z0-9]+$/
mkdir $host;
}
}
# Lock port number found by creating a new node
my $node = $class->new($name, $host, $port);
# Add node to list of nodes
push(@all_nodes, $node);
# And update port for next time return $port;
$port_is_forced or $last_port_assigned = $port;
return $node;
} }
# Internal routine to check whether a host:port is available to bind # Internal routine to check whether a host:port is available to bind
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment