| #!/usr/bin/perl |
| |
| # (C) Sergey Kandaurov |
| # (C) Nginx, Inc. |
| |
| # Tests for upstream hash balancer module distribution consistency |
| # with Cache::Memcached and Cache::Memcached::Fast. |
| |
| ############################################################################### |
| |
| 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 $@; |
| eval { require Cache::Memcached::Fast; }; |
| plan(skip_all => 'Cache::Memcached::Fast not installed') if $@; |
| |
| my $t = Test::Nginx->new()->has(qw/http rewrite memcached upstream_hash/) |
| ->has_daemon('memcached')->plan(4); |
| |
| $t->write_file_expand('nginx.conf', <<'EOF'); |
| |
| %%TEST_GLOBALS%% |
| |
| daemon off; |
| |
| events { |
| } |
| |
| http { |
| %%TEST_GLOBALS_HTTP%% |
| |
| upstream memd { |
| hash $arg_a; |
| server 127.0.0.1:8081; |
| server 127.0.0.1:8082; |
| server 127.0.0.1:8083; |
| } |
| |
| upstream memd_c { |
| hash $arg_a consistent; |
| server 127.0.0.1:8081; |
| server 127.0.0.1:8082; |
| server 127.0.0.1:8083; |
| } |
| |
| upstream memd_w { |
| hash $arg_a; |
| server 127.0.0.1:8081 weight=2; |
| server 127.0.0.1:8082 weight=3; |
| server 127.0.0.1:8083; |
| } |
| |
| upstream memd_cw { |
| hash $arg_a consistent; |
| server 127.0.0.1:8081 weight=2; |
| server 127.0.0.1:8082 weight=3; |
| server 127.0.0.1:8083; |
| } |
| |
| server { |
| listen 127.0.0.1:8080; |
| server_name localhost; |
| |
| set $memcached_key $arg_a; |
| |
| location / { |
| memcached_pass memd; |
| } |
| location /c { |
| memcached_pass memd_c; |
| } |
| location /w { |
| memcached_pass memd_w; |
| } |
| location /cw { |
| memcached_pass memd_cw; |
| } |
| } |
| } |
| |
| EOF |
| |
| my $memhelp = `memcached -h`; |
| my @memopts = (); |
| |
| if ($memhelp =~ /repcached/) { |
| # repcached patch adds additional listen socket |
| push @memopts, '-X', '0'; |
| } |
| if ($memhelp =~ /-U/) { |
| # UDP port is on by default in memcached 1.2.7+ |
| push @memopts, '-U', '0'; |
| } |
| |
| $t->run_daemon('memcached', '-l', '127.0.0.1', '-p', port(8081), @memopts); |
| $t->run_daemon('memcached', '-l', '127.0.0.1', '-p', port(8082), @memopts); |
| $t->run_daemon('memcached', '-l', '127.0.0.1', '-p', port(8083), @memopts); |
| $t->run(); |
| |
| $t->waitforsocket('127.0.0.1:' . port(8081)) or die "Can't start memcached"; |
| $t->waitforsocket('127.0.0.1:' . port(8082)) or die "Can't start memcached"; |
| $t->waitforsocket('127.0.0.1:' . port(8083)) or die "Can't start 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); |
| my $memd3 = Cache::Memcached->new(servers => [ '127.0.0.1:' . port(8083) ], |
| connect_timeout => 1.0); |
| |
| for my $i (1 .. 20) { |
| $memd1->set($i, port(8081)) or die "can't put value into memcached: $!"; |
| $memd2->set($i, port(8082)) or die "can't put value into memcached: $!"; |
| $memd3->set($i, port(8083)) or die "can't put value into memcached: $!"; |
| } |
| |
| my $memd = new Cache::Memcached(servers => [ |
| '127.0.0.1:' . port(8081), |
| '127.0.0.1:' . port(8082), |
| '127.0.0.1:' . port(8083) ]); |
| |
| is_deeply(ngx('/'), mem($memd), 'cache::memcached'); |
| |
| $memd = new Cache::Memcached::Fast({ ketama_points => 160, servers => [ |
| '127.0.0.1:' . port(8081), |
| '127.0.0.1:' . port(8082), |
| '127.0.0.1:' . port(8083)] }); |
| |
| # Cache::Memcached::Fast may be incompatible with recent Perl, |
| # see https://github.com/JRaspass/Cache-Memcached-Fast/issues/12 |
| |
| my $cmf_bug = ! keys %{$memd->server_versions}; |
| |
| SKIP: { |
| skip 'Cache::Memcached::Fast bug', 1 if $cmf_bug; |
| |
| is_deeply(ngx('/c'), mem($memd), 'cache::memcached::fast'); |
| |
| } |
| |
| $memd = new Cache::Memcached(servers => [ |
| [ '127.0.0.1:' . port(8081), 2 ], |
| [ '127.0.0.1:' . port(8082), 3 ], |
| [ '127.0.0.1:' . port(8083), 1 ]]); |
| |
| is_deeply(ngx('/w'), mem($memd), 'cache::memcached weight'); |
| |
| $memd = new Cache::Memcached::Fast({ ketama_points => 160, servers => [ |
| { address => '127.0.0.1:' . port(8081), weight => 2 }, |
| { address => '127.0.0.1:' . port(8082), weight => 3 }, |
| { address => '127.0.0.1:' . port(8083), weight => 1 }] }); |
| |
| SKIP: { |
| skip 'Cache::Memcached::Fast bug', 1 if $cmf_bug; |
| |
| is_deeply(ngx('/cw'), mem($memd), 'cache::memcached::fast weight'); |
| |
| } |
| |
| ############################################################################### |
| |
| sub ngx { |
| my ($uri) = @_; |
| [ map { http_get("/$uri?a=$_") =~ /^(\d+)/ms && $1; } (1 .. 20) ]; |
| } |
| |
| sub mem { |
| my ($memd) = @_; |
| [ map { $memd->get($_); } (1 .. 20) ]; |
| } |
| |
| ############################################################################### |