blob: ff828a26ee63da46a9371f4ed3fa64b38f2901b4 [file] [log] [blame]
#!/usr/bin/perl
# (C) Maxim Dounin
# Tests for http proxy cache revalidation with conditional requests.
###############################################################################
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;
my $t = Test::Nginx->new()->has(qw/http proxy cache rewrite/)->plan(23)
->write_file_expand('nginx.conf', <<'EOF');
%%TEST_GLOBALS%%
daemon off;
events {
}
http {
%%TEST_GLOBALS_HTTP%%
proxy_cache_path %%TESTDIR%%/cache levels=1:2
keys_zone=one:1m;
proxy_cache_revalidate on;
server {
listen 127.0.0.1:8080;
server_name localhost;
location / {
proxy_pass http://127.0.0.1:8081;
proxy_cache one;
proxy_cache_valid 200 404 2s;
add_header X-Cache-Status $upstream_cache_status;
}
}
server {
listen 127.0.0.1:8081;
server_name localhost;
location / { }
location /etag/ {
proxy_pass http://127.0.0.1:8081/;
proxy_hide_header Last-Modified;
}
location /201 {
add_header Last-Modified "Mon, 02 Mar 2015 17:20:58 GMT";
add_header Cache-Control "max-age=1";
add_header X-If-Modified-Since $http_if_modified_since;
return 201;
}
}
}
EOF
my $d = $t->testdir();
$t->write_file('t', 'SEE-THIS');
$t->write_file('t2', 'SEE-THIS');
$t->write_file('t3', 'SEE-THIS');
$t->run();
###############################################################################
# request documents and make sure they are cached
like(http_get('/t'), qr/X-Cache-Status: MISS.*SEE/ms, 'request');
like(http_get('/t'), qr/X-Cache-Status: HIT.*SEE/ms, 'request cached');
like(http_get('/t2'), qr/X-Cache-Status: MISS.*SEE/ms, '2nd request');
like(http_get('/t2'), qr/X-Cache-Status: HIT.*SEE/ms, '2nd request cached');
like(http_get('/etag/t'), qr/X-Cache-Status: MISS.*SEE/ms, 'etag');
like(http_get('/etag/t'), qr/X-Cache-Status: HIT.*SEE/ms, 'etag cached');
like(http_get('/etag/t2'), qr/X-Cache-Status: MISS.*SEE/ms, 'etag2');
like(http_get('/etag/t2'), qr/X-Cache-Status: HIT.*SEE/ms, 'etag2 cached');
like(http_get('/201'), qr/X-Cache-Status: MISS/, 'other status');
like(http_get('/201'), qr/X-Cache-Status: HIT/, 'other status cached');
like(http_get('/t3'), qr/SEE/, 'cache before 404');
# wait for a while for cached responses to expire
select undef, undef, undef, 3.5;
# 1st document isn't modified, and should be revalidated on first request
# (a 304 status code will appear in backend's logs), then cached again
like(http_get('/t'), qr/X-Cache-Status: REVALIDATED.*SEE/ms, 'revalidated');
like(http_get('/t'), qr/X-Cache-Status: HIT.*SEE/ms, 'cached again');
rename("$d/t3", "$d/t3_moved");
like(http_get('/t3'), qr/ 404 /, 'cache 404 response');
select undef, undef, undef, 0.1;
like($t->read_file('access.log'), qr/ 304 /, 'not modified');
# 2nd document is recreated with a new content
$t->write_file('t2', 'NEW');
like(http_get('/t2'), qr/X-Cache-Status: EXPIRED.*NEW/ms, 'revalidate failed');
like(http_get('/t2'), qr/X-Cache-Status: HIT.*NEW/ms, 'new response cached');
# the same for etag:
# 1st document isn't modified
# 2nd document is recreated
like(http_get('/etag/t'), qr/X-Cache-Status: REVALIDATED.*SEE/ms,
'etag revalidated');
like(http_get('/etag/t'), qr/X-Cache-Status: HIT.*SEE/ms,
'etag cached again');
like(http_get('/etag/t2'), qr/X-Cache-Status: EXPIRED.*NEW/ms,
'etag2 revalidate failed');
like(http_get('/etag/t2'), qr/X-Cache-Status: HIT.*NEW/ms,
'etag2 new response cached');
# check that conditional requests are only used for 200/206 responses
# d0ce06cb9be1 in 1.7.3 changed to ignore header filter's work to strip
# the Last-Modified header when storing non-200/206 in cache;
# 1573fc7875fa in 1.7.9 effectively turned it back.
unlike(http_get('/201'), qr/X-If-Modified/, 'other status no revalidation');
# wait for a while for a cached 404 response to expire
select undef, undef, undef, 3.5;
# check that conditional requests are not used to revalidate 404 response
# before fd283aa92e04 introduced in 1.7.7, this test passed by chance because
# of the If-Modified-Since header that was sent with Epoch in revalidation
# of responses cached without the Last-Modified header;
# fd283aa92e04 leaved (an legitimate) successful revalidation of 404 by ETag
# (introduced by 44b9ab7752e3 in 1.7.3), which caused the test to fail;
# 1573fc7875fa in 1.7.9 changed to not revalidate non-200/206 responses but
# leaked Last-Modified and ETag into 404 inherited from stale 200/206 response;
# 174512857ccf in 1.7.11 fixed the leak and allowed the test to pass.
rename("$d/t3_moved", "$d/t3");
like(http_get('/t3'), qr/SEE/, 'no 404 revalidation after stale 200');
###############################################################################