blob: 62ebfeda9bbec61b9b58f57d247a1f3a35f3cc80 [file] [log] [blame]
#!/usr/bin/perl
# (C) Dmitry Volyntsev
# (C) Nginx, Inc.
# Tests for http njs module, working with headers.
###############################################################################
use warnings;
use strict;
use Test::More;
use Socket qw/ CRLF /;
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 charset/)
->write_file_expand('nginx.conf', <<'EOF');
%%TEST_GLOBALS%%
daemon off;
events {
}
http {
%%TEST_GLOBALS_HTTP%%
js_set $test_foo_in test_foo_in;
js_set $test_ifoo_in test_ifoo_in;
js_include test.js;
server {
listen 127.0.0.1:8080;
server_name localhost;
location /njs {
js_content test_njs;
}
location /content_length {
js_content content_length;
}
location /content_length_arr {
js_content content_length_arr;
}
location /content_length_keys {
js_content content_length_keys;
}
location /content_type {
charset windows-1251;
default_type text/plain;
js_content content_type;
}
location /content_type_arr {
charset windows-1251;
default_type text/plain;
js_content content_type_arr;
}
location /content_encoding {
js_content content_encoding;
}
location /content_encoding_arr {
js_content content_encoding_arr;
}
location /headers_list {
js_content headers_list;
}
location /foo_in {
return 200 $test_foo_in;
}
location /ifoo_in {
return 200 $test_ifoo_in;
}
location /hdr_in {
js_content hdr_in;
}
location /raw_hdr_in {
js_content raw_hdr_in;
}
location /hdr_out {
js_content hdr_out;
}
location /raw_hdr_out {
js_content raw_hdr_out;
}
location /hdr_out_array {
js_content hdr_out_array;
}
location /hdr_out_set_cookie {
js_content hdr_out_set_cookie;
}
location /hdr_out_single {
js_content hdr_out_single;
}
location /ihdr_out {
js_content ihdr_out;
}
location /hdr_sorted_keys {
js_content hdr_sorted_keys;
}
}
}
EOF
$t->write_file('test.js', <<EOF);
function test_njs(r) {
r.return(200, njs.version);
}
function content_length(r) {
r.headersOut['Content-Length'] = '';
r.headersOut['Content-Length'] = 3;
delete r.headersOut['Content-Length'];
r.headersOut['Content-Length'] = 3;
r.sendHeader();
r.send('XXX');
r.finish();
}
function content_length_arr(r) {
r.headersOut['Content-Length'] = [5];
r.headersOut['Content-Length'] = [];
r.headersOut['Content-Length'] = [4,3];
r.sendHeader();
r.send('XXX');
r.finish();
}
function content_length_keys(r) {
r.headersOut['Content-Length'] = 3;
var in_keys = Object.keys(r.headersOut).some(v=>v=='Content-Length');
r.return(200, `B:\${in_keys}`);
}
function content_type(r) {
r.headersOut['Content-Type'] = 'text/xml';
r.headersOut['Content-Type'] = '';
r.headersOut['Content-Type'] = 'text/xml; charset=';
delete r.headersOut['Content-Type'];
r.headersOut['Content-Type'] = 'text/xml; charset=utf-8';
r.headersOut['Content-Type'] = 'text/xml; charset="utf-8"';
var in_keys = Object.keys(r.headersOut).some(v=>v=='Content-Type');
r.return(200, `B:\${in_keys}`);
}
function content_type_arr(r) {
r.headersOut['Content-Type'] = ['text/html'];
r.headersOut['Content-Type'] = [];
r.headersOut['Content-Type'] = [ 'text/xml', 'text/html'];
r.return(200);
}
function content_encoding(r) {
r.headersOut['Content-Encoding'] = '';
r.headersOut['Content-Encoding'] = 'test';
delete r.headersOut['Content-Encoding'];
r.headersOut['Content-Encoding'] = 'gzip';
r.return(200);
}
function content_encoding_arr(r) {
r.headersOut['Content-Encoding'] = 'test';
r.headersOut['Content-Encoding'] = [];
r.headersOut['Content-Encoding'] = ['test', 'gzip'];
r.return(200);
}
function headers_list(r) {
for (var h in {a:1, b:2, c:3}) {
r.headersOut[h] = h;
}
delete r.headersOut.b;
r.headersOut.d = 'd';
var out = "";
for (var h in r.headersOut) {
out += h + ":";
}
r.return(200, out);
}
function hdr_in(r) {
var s = '', h;
for (h in r.headersIn) {
s += `\${h.toLowerCase()}: \${r.headersIn[h]}\n`;
}
r.return(200, s);
}
function raw_hdr_in(r) {
var filtered = r.rawHeadersIn
.filter(v=>v[0].toLowerCase() == r.args.filter);
r.return(200, 'raw:' + filtered.map(v=>v[1]).join('|'));
}
function hdr_sorted_keys(r) {
var s = '';
var hdr = r.args.in ? r.headersIn : r.headersOut;
if (!r.args.in) {
r.headersOut.b = 'b';
r.headersOut.c = 'c';
r.headersOut.a = 'a';
}
r.return(200, Object.keys(hdr).sort());
}
function test_foo_in(r) {
return 'hdr=' + r.headersIn.foo;
}
function test_ifoo_in(r) {
var s = '', h;
for (h in r.headersIn) {
if (h.substr(0, 3) == 'foo') {
s += r.headersIn[h];
}
}
return s;
}
function hdr_out(r) {
r.status = 200;
r.headersOut['Foo'] = r.args.fOO;
if (r.args.bar) {
r.headersOut['Bar'] =
r.headersOut[(r.args.bar == 'empty' ? 'Baz' :'Foo')]
}
r.sendHeader();
r.finish();
}
function raw_hdr_out(r) {
r.headersOut.a = ['foo', 'bar'];
r.headersOut.b = 'b';
var filtered = r.rawHeadersOut
.filter(v=>v[0].toLowerCase() == r.args.filter);
r.return(200, 'raw:' + filtered.map(v=>v[1]).join('|'));
}
function hdr_out_array(r) {
if (!r.args.hidden) {
r.headersOut['Foo'] = [r.args.fOO];
r.headersOut['Foo'] = [];
r.headersOut['Foo'] = ['bar', r.args.fOO];
}
if (r.args.scalar_set) {
r.headersOut['Foo'] = 'xxx';
}
r.return(200, `B:\${njs.dump(r.headersOut.foo)}`);
}
function hdr_out_single(r) {
r.headersOut.ETag = ['a', 'b'];
r.return(200, `B:\${njs.dump(r.headersOut.etag)}`);
}
function hdr_out_set_cookie(r) {
r.headersOut['Set-Cookie'] = [];
r.headersOut['Set-Cookie'] = ['a', 'b'];
delete r.headersOut['Set-Cookie'];
r.headersOut['Set-Cookie'] = 'e';
r.headersOut['Set-Cookie'] = ['c', '', null, 'd', 'f'];
r.return(200, `B:\${njs.dump(r.headersOut['Set-Cookie'])}`);
}
function ihdr_out(r) {
r.status = 200;
r.headersOut['a'] = r.args.a;
r.headersOut['b'] = r.args.b;
var s = '', h;
for (h in r.headersOut) {
s += r.headersOut[h];
}
r.sendHeader();
r.send(s);
r.finish();
}
EOF
$t->try_run('no njs')->plan(39);
###############################################################################
like(http_get('/content_length'), qr/Content-Length: 3/,
'set Content-Length');
like(http_get('/content_type'), qr/Content-Type: text\/xml; charset="utf-8"\r/,
'set Content-Type');
unlike(http_get('/content_type'), qr/Content-Type: text\/plain/,
'set Content-Type 2');
like(http_get('/content_encoding'), qr/Content-Encoding: gzip/,
'set Content-Encoding');
like(http_get('/headers_list'), qr/a:c:d/, 'headers list');
like(http_get('/ihdr_out?a=12&b=34'), qr/^1234$/m, 'r.headersOut iteration');
like(http_get('/ihdr_out'), qr/\x0d\x0a?\x0d\x0a?$/m, 'r.send zero');
like(http_get('/hdr_out?foo=12345'), qr/Foo: 12345/, 'r.headersOut');
like(http_get('/hdr_out?foo=123&bar=copy'), qr/Bar: 123/, 'r.headersOut get');
unlike(http_get('/hdr_out?bar=empty'), qr/Bar:/, 'r.headersOut empty');
unlike(http_get('/hdr_out?foo='), qr/Foo:/, 'r.headersOut no value');
unlike(http_get('/hdr_out?foo'), qr/Foo:/, 'r.headersOut no value 2');
TODO: {
local $TODO = 'not yet'
unless http_get('/njs') =~ /^([.0-9]+)$/m && $1 ge '0.4.0';
like(http_get('/content_length_keys'), qr/B:true/, 'Content-Length in keys');
like(http_get('/content_length_arr'), qr/Content-Length: 3/,
'set Content-Length arr');
like(http_get('/content_type'), qr/B:true/, 'Content-Type in keys');
like(http_get('/content_type_arr'), qr/Content-Type: text\/html/,
'set Content-Type arr');
like(http_get('/content_encoding_arr'), qr/Content-Encoding: gzip/,
'set Content-Encoding arr');
like(http_get('/hdr_out_array?foo=12345'), qr/Foo: bar\r\nFoo: 12345/,
'r.headersOut arr');
like(http_get('/hdr_out_array'), qr/Foo: bar/,
'r.headersOut arr last is empty');
like(http_get('/hdr_out_array?foo=abc'), qr/B:bar,abc/,
'r.headersOut get');
like(http_get('/hdr_out_array'), qr/B:bar/, 'r.headersOut get2');
like(http_get('/hdr_out_array?hidden=1'), qr/B:undefined/,
'r.headersOut get3');
like(http_get('/hdr_out_array?scalar_set=1'), qr/B:xxx/,
'r.headersOut scalar set');
like(http_get('/hdr_out_single'), qr/ETag: a\r\nETag: b/,
'r.headersOut single');
like(http_get('/hdr_out_single'), qr/B:a/,
'r.headersOut single get');
like(http_get('/hdr_out_set_cookie'), qr/Set-Cookie: c\r\nSet-Cookie: d/,
'set_cookie');
like(http_get('/hdr_out_set_cookie'), qr/B:\['c','d','f']/,
'set_cookie2');
unlike(http_get('/hdr_out_set_cookie'), qr/Set-Cookie: [abe]/,
'set_cookie3');
}
like(http(
'GET /hdr_in HTTP/1.0' . CRLF
. 'Cookie: foo' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/cookie: foo/, 'r.headersIn cookie');
like(http(
'GET /hdr_in HTTP/1.0' . CRLF
. 'X-Forwarded-For: foo' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/x-forwarded-for: foo/, 'r.headersIn xff');
like(http(
'GET /hdr_in HTTP/1.0' . CRLF
. 'Cookie: foo1' . CRLF
. 'Cookie: foo2' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/cookie: foo1;\s?foo2/, 'r.headersIn cookie2');
like(http(
'GET /hdr_in HTTP/1.0' . CRLF
. 'X-Forwarded-For: foo1' . CRLF
. 'X-Forwarded-For: foo2' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/x-forwarded-for: foo1,\s?foo2/, 'r.headersIn xff2');
like(http(
'GET /hdr_in HTTP/1.0' . CRLF
. 'ETag: bar1' . CRLF
. 'ETag: bar2' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/etag: bar1(?!,\s?bar2)/, 'r.headersIn duplicate single');
like(http(
'GET /hdr_in HTTP/1.0' . CRLF
. 'Content-Type: bar1' . CRLF
. 'Content-Type: bar2' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/content-type: bar1(?!,\s?bar2)/, 'r.headersIn duplicate single 2');
TODO: {
local $TODO = 'not yet'
unless http_get('/njs') =~ /^([.0-9]+)$/m && $1 ge '0.4.1';
like(http(
'GET /hdr_in HTTP/1.0' . CRLF
. 'Foo: bar1' . CRLF
. 'Foo: bar2' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/foo: bar1,bar2/, 'r.headersIn duplicate generic');
like(http(
'GET /raw_hdr_in?filter=foo HTTP/1.0' . CRLF
. 'foo: bar1' . CRLF
. 'Foo: bar2' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/raw: bar1|bar2/, 'r.rawHeadersIn');
like(http_get('/raw_hdr_out?filter=a'), qr/raw: foo|bar/, 'r.rawHeadersOut');
}
like(http(
'GET /hdr_sorted_keys?in=1 HTTP/1.0' . CRLF
. 'Cookie: foo1' . CRLF
. 'Accept: */*' . CRLF
. 'Cookie: foo2' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/Accept,Cookie,Host/, 'r.headersIn sorted keys');
like(http(
'GET /hdr_sorted_keys HTTP/1.0' . CRLF
. 'Host: localhost' . CRLF . CRLF
), qr/a,b,c/, 'r.headersOut sorted keys');
###############################################################################