2 Star 0 Fork 0

mirrors_nginx/nginx-tests

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
upstream_service_reload.t 6.48 KB
一键复制 编辑 原始数据 按行查看 历史
#!/usr/bin/perl
# (C) Sergey Kandaurov
# (C) Nginx, Inc.
# Tests for dynamic upstream configuration with service (SRV) feature.
# Ensure that upstream configuration is inherited on reload.
###############################################################################
use warnings;
use strict;
use Test::More;
use IO::Select;
BEGIN { use FindBin; chdir($FindBin::Bin); }
use lib 'lib';
use Test::Nginx;
###############################################################################
select STDERR; $| = 1;
select STDOUT; $| = 1;
plan(skip_all => 'win32') if $^O eq 'MSWin32';
my $t = Test::Nginx->new()->has(qw/http upstream_zone/);
$t->write_file_expand('nginx.conf', <<'EOF');
%%TEST_GLOBALS%%
daemon off;
events {
}
http {
%%TEST_GLOBALS_HTTP%%
upstream u {
zone z 1m;
server example.net resolve service=http;
}
# lower the retry timeout after empty reply
resolver 127.0.0.1:%%PORT_8980_UDP%% valid=1s;
# retry query shortly after DNS is started
resolver_timeout 1s;
server {
listen 127.0.0.1:8080;
server_name localhost;
location / {
proxy_pass http://u;
proxy_connect_timeout 50ms;
add_header X-IP $upstream_addr always;
}
}
}
EOF
my $p = port(8081);
$t->run_daemon(\&dns_daemon, $t)->waitforfile($t->testdir . '/' . port(8980));
$t->try_run('no resolve in upstream server')->plan(6);
###############################################################################
update_name({A => '127.0.0.201', SRV => "1 5 $p example.net"});
like(http_get('/'), qr/X-IP: 127.0.0.201:$p/, 'reload - before - request');
like(http_get('/'), qr/X-IP: 127.0.0.201:$p/, 'reload - before - request 2');
update_name({ERROR => 'SERVFAIL'}, 0);
$t->reload();
waitforworker($t);
like(http_get('/'), qr/X-IP: 127.0.0.201:$p/, 'reload - preresolve - request');
like(http_get('/'), qr/X-IP: 127.0.0.201:$p/, 'reload - preresolve - request 2');
update_name({A => '127.0.0.202', SRV => "1 5 $p example.net"});
like(http_get('/'), qr/X-IP: 127.0.0.202:$p/, 'reload - update - request');
like(http_get('/'), qr/X-IP: 127.0.0.202:$p/, 'reload - update - request 2');
###############################################################################
sub waitforworker {
my ($t) = @_;
for (1 .. 30) {
last if $t->read_file('error.log') =~ /exited with code/;
select undef, undef, undef, 0.2;
}
}
sub update_name {
my ($name, $plan) = @_;
$plan = 3 if !defined $plan;
sub sock {
IO::Socket::INET->new(
Proto => 'tcp',
PeerAddr => '127.0.0.1:' . port(8081)
)
or die "Can't connect to nginx: $!\n";
}
$name->{A} = '' unless $name->{A};
$name->{ERROR} = '' unless $name->{ERROR};
$name->{SRV} = '' unless $name->{SRV};
my $req =<<EOF;
GET / HTTP/1.0
Host: localhost
X-A: $name->{A}
X-ERROR: $name->{ERROR}
X-SRV: $name->{SRV}
EOF
my ($gen) = http($req, socket => sock()) =~ /X-Gen: (\d+)/;
for (1 .. 10) {
my ($gen2) = http($req, socket => sock()) =~ /X-Gen: (\d+)/;
# let resolver cache expire to finish upstream reconfiguration
select undef, undef, undef, 0.5;
last unless ($gen + $plan > $gen2);
}
}
###############################################################################
sub reply_handler {
my ($recv_data, $h) = @_;
my (@name, @rdata);
use constant NOERROR => 0;
use constant SERVFAIL => 2;
use constant NXDOMAIN => 3;
use constant A => 1;
use constant SRV => 33;
use constant IN => 1;
# default values
my ($hdr, $rcode, $ttl, $port) = (0x8180, NOERROR, 3600, port(8080));
$h = {A => [ "127.0.0.201" ], SRV => [ "1 5 $port example.net" ]}
unless defined $h;
# decode name
my ($len, $offset) = (undef, 12);
while (1) {
$len = unpack("\@$offset C", $recv_data);
last if $len == 0;
$offset++;
push @name, unpack("\@$offset A$len", $recv_data);
$offset += $len;
}
$offset -= 1;
my ($id, $type, $class) = unpack("n x$offset n2", $recv_data);
my $name = join('.', @name);
if ($h->{ERROR}) {
$rcode = SERVFAIL;
goto bad;
}
if ($name eq '_http._tcp.example.net' && $type == SRV && $h->{SRV}) {
map { push @rdata, rd_srv($ttl, (split ' ', $_)) }
@{$h->{SRV}};
} elsif ($name eq 'example.net' && $type == A && $h->{A}) {
map { push @rdata, rd_addr($ttl, $_) } @{$h->{A}};
}
bad:
Test::Nginx::log_core('||', "DNS: $name $type $rcode");
$len = @name;
pack("n6 (C/a*)$len x n2", $id, $hdr | $rcode, 1, scalar @rdata,
0, 0, @name, $type, $class) . join('', @rdata);
}
sub rd_srv {
my ($ttl, $pri, $w, $port, $name) = @_;
my @rdname = split /\./, $name;
my $rdlen = length(join '', @rdname) + @rdname + 7; # pri w port x
pack 'n3N n n3 (C/a*)* x',
0xc00c, SRV, IN, $ttl, $rdlen, $pri, $w, $port, @rdname;
}
sub rd_addr {
my ($ttl, $addr) = @_;
my $code = 'split(/\./, $addr)';
pack 'n3N nC4', 0xc00c, A, IN, $ttl, eval "scalar $code", eval($code);
}
sub dns_daemon {
my ($t) = @_;
my ($data, $recv_data, $h);
my $socket = IO::Socket::INET->new(
LocalAddr => '127.0.0.1',
LocalPort => port(8980),
Proto => 'udp',
)
or die "Can't create listening socket: $!\n";
my $control = IO::Socket::INET->new(
Proto => 'tcp',
LocalHost => "127.0.0.1:" . port(8081),
Listen => 5,
Reuse => 1
)
or die "Can't create listening socket: $!\n";
my $sel = IO::Select->new($socket, $control);
local $SIG{PIPE} = 'IGNORE';
# signal we are ready
open my $fh, '>', $t->testdir() . '/' . port(8980);
close $fh;
my $cnt = 0;
while (my @ready = $sel->can_read) {
foreach my $fh (@ready) {
if ($control == $fh) {
my $new = $fh->accept;
$new->autoflush(1);
$sel->add($new);
} elsif ($socket == $fh) {
$fh->recv($recv_data, 65536);
$data = reply_handler($recv_data, $h);
$fh->send($data);
$cnt++;
} else {
$h = process_name($fh, $cnt);
$sel->remove($fh);
$fh->close;
}
}
}
}
# parse dns update
sub process_name {
my ($client, $cnt) = @_;
my $port = $client->sockport();
my $headers = '';
my $uri = '';
my %h;
while (<$client>) {
$headers .= $_;
last if (/^\x0d?\x0a?$/);
}
return 1 if $headers eq '';
$uri = $1 if $headers =~ /^\S+\s+([^ ]+)\s+HTTP/i;
return 1 if $uri eq '';
$headers =~ /X-A: (.*)$/m;
map { push @{$h{A}}, $_ } split(/ /, $1);
$headers =~ /X-SRV: (.*)$/m;
map { push @{$h{SRV}}, $_ } split(/;/, $1);
$headers =~ /X-ERROR: (.*)$/m;
$h{ERROR} = $1;
Test::Nginx::log_core('||', "$port: response, 200");
print $client <<EOF;
HTTP/1.1 200 OK
Connection: close
X-Gen: $cnt
OK
EOF
return \%h;
}
###############################################################################
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/mirrors_nginx/nginx-tests.git
git@gitee.com:mirrors_nginx/nginx-tests.git
mirrors_nginx
nginx-tests
nginx-tests
master

搜索帮助