#!/usr/bin/perl # (C) Roman Arutyunyan # (C) Dmitry Volyntsev # (C) Nginx, Inc. # Tests for http njs module. ############################################################################### 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 rewrite/) ->write_file_expand('nginx.conf', <<'EOF'); %%TEST_GLOBALS%% daemon off; events { } http { %%TEST_GLOBALS_HTTP%% js_set $test_method test.method; js_set $test_version test.version; js_set $test_addr test.addr; js_set $test_uri test.uri; js_set $test_var test.variable; js_set $test_type test.type; js_set $test_global test.global_obj; js_set $test_log test.log; js_set $test_internal test.sub_internal; js_set $buffer test.buffer; js_set $test_except test.except; js_import test.js; server { listen 127.0.0.1:8080; server_name localhost; location /njs { js_content test.njs; } location /method { return 200 $test_method; } location /version { return 200 $test_version; } location /addr { return 200 $test_addr; } location /uri { return 200 $test_uri; } location /var { return 200 $test_var; } location /global { return 200 $test_global; } location /status { js_content test.status; } location /buffer_variable { js_content test.buffer_variable; } location /send { js_content test.send; } location /send_buffer { js_content test.send_buffer; } location /return_method { js_content test.return_method; } location /type { js_content test.type; } location /log { return 200 $test_log; } location /internal { js_content test.internal; } location /sub_internal { internal; return 200 $test_internal; } location /except { return 200 $test_except; } location /content_except { js_content test.content_except; } location /content_empty { js_content test.content_empty; } } } EOF $t->write_file('test.js', < a[v], r); var typ = Buffer.isBuffer(p) ? 'buffer' : (typeof p); r.return(200, `type: \${typ}`); } function log(r) { r.log('SEE-LOG'); } function buffer_variable(r) { r.return(200, r.rawVariables.buffer.toString('hex')); } async function internal(r) { let reply = await r.subrequest('/sub_internal'); r.return(200, `parent: \${r.internal} sub: \${reply.responseText}`); } function sub_internal(r) { return r.internal; } function except(r) { decodeURI("%E0"); } function content_except(r) { return {}.a.a; } function content_empty(r) { } export default {njs:test_njs, method, version, addr, uri, buffer, variable, global_obj, status, internal, send, return_method, sub_internal, type, log, buffer_variable, except, content_except, content_empty, send_buffer}; EOF $t->try_run('no njs available')->plan(25); ############################################################################### like(http_get('/method'), qr/method=GET/, 'r.method'); like(http_get('/version'), qr/version=1.0/, 'r.httpVersion'); like(http_get('/addr'), qr/addr=127.0.0.1/, 'r.remoteAddress'); like(http_get('/uri'), qr/uri=\/uri/, 'r.uri'); like(http_get('/status'), qr/204 No Content/, 'r.status'); like(http_get('/send?foo=12345&n=11&foo-2=bar&ndd=&foo-3=z'), qr/n=foo, v=12 n=foo-2, v=ba n=foo-3, v=z/, 'r.send'); TODO: { local $TODO = 'not yet' unless has_version('0.8.4'); like(http_get('/send_buffer'), qr/send_buffer/, 'r.send accepts buffer'); } like(http_get('/return_method?c=200'), qr/200 OK.*\x0d\x0a?\x0d\x0a?$/s, 'return code'); like(http_get('/return_method?c=200&t=SEE-THIS'), qr/200 OK.*^SEE-THIS$/ms, 'return text'); like(http_get('/return_method?c=301&t=path'), qr/ 301 .*Location: path/s, 'return redirect'); like(http_get('/return_method?c=404'), qr/404 Not.*html/s, 'return error page'); like(http_get('/return_method?c=inv'), qr/ 500 /, 'return invalid'); like(http_get('/type?path=variables.host'), qr/200 OK.*type: string$/s, 'variables type'); like(http_get('/type?path=rawVariables.host'), qr/200 OK.*type: buffer$/s, 'rawVariables type'); like(http_post('/type?path=requestText'), qr/200 OK.*type: string$/s, 'requestText type'); like(http_post('/type?path=requestBuffer'), qr/200 OK.*type: buffer$/s, 'requestBuffer type'); like(http_get('/var'), qr/variable=127.0.0.1/, 'r.variables'); like(http_get('/global'), qr/global=njs/, 'global code'); like(http_get('/log'), qr/200 OK/, 'r.log'); TODO: { local $TODO = 'not yet' unless has_version('0.7.7'); like(http_get('/internal'), qr/parent: false sub: true/, 'r.internal'); } TODO: { local $TODO = 'not yet' unless has_version('0.8.3'); like(http_get('/buffer_variable'), qr/aabbccdd/, 'buffer variable'); } http_get('/except'); http_get('/content_except'); like(http_get('/content_empty'), qr/500 Internal Server Error/, 'empty handler'); $t->stop(); ok(index($t->read_file('error.log'), 'SEE-LOG') > 0, 'log js'); ok(index($t->read_file('error.log'), 'at decodeURI') > 0, 'js_set backtrace'); ok(index($t->read_file('error.log'), 'at content_except') > 0, 'js_content backtrace'); ############################################################################### sub has_version { my $need = shift; http_get('/njs') =~ /^([.0-9]+)$/m; my @v = split(/\./, $1); my ($n, $v); for $n (split(/\./, $need)) { $v = shift @v || 0; return 0 if $n > $v; return 1 if $v > $n; } return 1; } ############################################################################### sub http_get_hdr { my ($url, %extra) = @_; return http(<