| #!/usr/bin/perl |
| |
| # (C) Maxim Dounin |
| |
| # Test for memcached with keepalive. |
| |
| ############################################################################### |
| |
| use warnings; |
| use strict; |
| |
| use Test::More; |
| |
| BEGIN { use FindBin; chdir($FindBin::Bin); } |
| |
| use lib 'lib'; |
| use Test::Nginx; |
| |
| ############################################################################### |
| |
| select STDERR; $| = 1; |
| select STDOUT; $| = 1; |
| |
| eval { require Cache::Memcached; }; |
| plan(skip_all => 'Cache::Memcached not installed') if $@; |
| |
| my $t = Test::Nginx->new()->has(qw/http memcached upstream_keepalive rewrite/) |
| ->has_daemon('memcached')->plan(15) |
| ->write_file_expand('nginx.conf', <<'EOF'); |
| |
| %%TEST_GLOBALS%% |
| |
| daemon off; |
| |
| events { |
| } |
| |
| http { |
| %%TEST_GLOBALS_HTTP%% |
| |
| upstream memd { |
| server 127.0.0.1:8081; |
| keepalive 1; |
| } |
| |
| upstream memd3 { |
| server 127.0.0.1:8081; |
| server 127.0.0.1:8082; |
| keepalive 1; |
| } |
| |
| upstream memd4 { |
| server 127.0.0.1:8081; |
| server 127.0.0.1:8082; |
| keepalive 10; |
| } |
| |
| server { |
| listen 127.0.0.1:8080; |
| server_name localhost; |
| |
| location / { |
| set $memcached_key $uri; |
| memcached_pass memd; |
| } |
| |
| location /next { |
| set $memcached_key $uri; |
| memcached_next_upstream not_found; |
| memcached_pass memd; |
| } |
| |
| location /memd3 { |
| set $memcached_key "/"; |
| memcached_pass memd3; |
| } |
| |
| location /memd4 { |
| set $memcached_key "/"; |
| memcached_pass memd4; |
| } |
| } |
| } |
| |
| EOF |
| |
| my $memhelp = `memcached -h`; |
| my @memopts1 = (); |
| my @memopts2 = (); |
| |
| if ($memhelp =~ /repcached/) { |
| # repcached patches adds additional listen socket memcached |
| # that should be different too |
| |
| push @memopts1, '-X', port(8083); |
| push @memopts2, '-X', port(8084); |
| } |
| if ($memhelp =~ /-U/) { |
| # UDP ports no longer off by default in memcached 1.2.7+ |
| |
| push @memopts1, '-U', '0'; |
| push @memopts2, '-U', '0'; |
| } |
| if ($memhelp =~ /-t/) { |
| # for connection stats consistency in threaded memcached 1.3+ |
| |
| push @memopts1, '-t', '1'; |
| push @memopts2, '-t', '1'; |
| } |
| |
| $t->run_daemon('memcached', '-l', '127.0.0.1', '-p', port(8081), @memopts1); |
| $t->run_daemon('memcached', '-l', '127.0.0.1', '-p', port(8082), @memopts2); |
| |
| $t->run(); |
| |
| $t->waitforsocket('127.0.0.1:' . port(8081)) |
| or die "Unable to start memcached"; |
| $t->waitforsocket('127.0.0.1:' . port(8082)) |
| or die "Unable to start second memcached"; |
| |
| ############################################################################### |
| |
| my $memd1 = Cache::Memcached->new(servers => [ '127.0.0.1:' . port(8081) ], |
| connect_timeout => 1.0); |
| my $memd2 = Cache::Memcached->new(servers => [ '127.0.0.1:' . port(8082) ], |
| connect_timeout => 1.0); |
| |
| $memd1->set('/', 'SEE-THIS'); |
| $memd2->set('/', 'SEE-THIS'); |
| $memd1->set('/big', 'X' x 1000000); |
| |
| my $total = $memd1->stats()->{total}->{total_connections}; |
| |
| like(http_get('/'), qr/SEE-THIS/, 'keepalive memcached request'); |
| like(http_get('/notfound'), qr/ 404 /, 'keepalive memcached not found'); |
| like(http_get('/next'), qr/ 404 /, |
| 'keepalive not found with memcached_next_upstream'); |
| like(http_get('/'), qr/SEE-THIS/, 'keepalive memcached request again'); |
| like(http_get('/'), qr/SEE-THIS/, 'keepalive memcached request again'); |
| like(http_get('/'), qr/SEE-THIS/, 'keepalive memcached request again'); |
| |
| is($memd1->stats()->{total}->{total_connections}, $total + 1, |
| 'only one connection used'); |
| |
| # Since nginx doesn't read all data from connection in some situations (head |
| # requests, post_action, errors writing to client) we have to close such |
| # connections. Check if we really do close them. |
| |
| $total = $memd1->stats()->{total}->{total_connections}; |
| |
| unlike(http_head('/'), qr/SEE-THIS/, 'head request'); |
| like(http_get('/'), qr/SEE-THIS/, 'get after head'); |
| |
| is($memd1->stats()->{total}->{total_connections}, $total + 1, |
| 'head request closes connection'); |
| |
| $total = $memd1->stats()->{total}->{total_connections}; |
| |
| unlike(http_head('/big'), qr/XXX/, 'big head'); |
| like(http_get('/'), qr/SEE-THIS/, 'get after big head'); |
| |
| is($memd1->stats()->{total}->{total_connections}, $total + 1, |
| 'big head request closes connection'); |
| |
| # two backends with maximum number of cached connections set to 1, |
| # should establish new connection on each request |
| |
| $total = $memd1->stats()->{total}->{total_connections} + |
| $memd2->stats()->{total}->{total_connections}; |
| |
| http_get('/memd3'); |
| http_get('/memd3'); |
| http_get('/memd3'); |
| |
| is($memd1->stats()->{total}->{total_connections} + |
| $memd2->stats()->{total}->{total_connections}, $total + 3, |
| '3 connections should be established'); |
| |
| # two backends with maximum number of cached connections set to 10, |
| # should establish only two connections (1 per backend) |
| |
| $total = $memd1->stats()->{total}->{total_connections} + |
| $memd2->stats()->{total}->{total_connections}; |
| |
| http_get('/memd4'); |
| http_get('/memd4'); |
| http_get('/memd4'); |
| |
| is($memd1->stats()->{total}->{total_connections} + |
| $memd2->stats()->{total}->{total_connections}, $total + 2, |
| 'connection per backend'); |
| |
| ############################################################################### |