blob: f6a578420bb3511c7560873589c7fc278c1b3c24 [file] [log] [blame] [raw]
# vim:set ft= ts=4 sw=4 et fdm=marker:
use lib 'lib';
use Test::Nginx::Socket::Lua;
#worker_connections(1014);
#master_on();
#workers(2);
#log_level('warn');
repeat_each(2);
plan tests => repeat_each() * (blocks() * 2 + 13);
#no_diff();
no_long_string();
run_tests();
__DATA__
=== TEST 1: matched but w/o variables
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("hello, world", "[a-z]+", "howdy")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
howdy, world
1
=== TEST 2: not matched
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("hello, world", "[A-Z]+", "howdy")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
hello, world
0
=== TEST 3: matched and with variables
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("a b c d", "(b) (c)", "[$0] [$1] [$2] [$3] [$134]")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
a [b c] [b] [c] [] [] d
1
=== TEST 4: matched and with named variables
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("a b c d",
"(b) (c)", "[$0] [$1] [$2] [$3] [$hello]")
if s then
ngx.say(s, ": ", n)
else
ngx.say("error: ", err)
end
';
}
--- request
GET /re
--- response_body
error: failed to compile the replacement template
--- error_log
attempt to use named capturing variable "hello" (named captures not supported yet)
=== TEST 5: matched and with named variables (bracketed)
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("a b c d",
"(b) (c)", "[$0] [$1] [$2] [$3] [${hello}]")
if s then
ngx.say(s, ": ", n)
else
ngx.say("error: ", err)
end
';
}
--- request
GET /re
--- response_body
error: failed to compile the replacement template
--- error_log
attempt to use named capturing variable "hello" (named captures not supported yet)
=== TEST 6: matched and with bracketed variables
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("b c d", "(b) (c)", "[$0] [$1] [${2}] [$3] [${134}]")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
[b c] [b] [c] [] [] d
1
=== TEST 7: matched and with bracketed variables (unmatched brackets)
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("b c d", "(b) (c)", "[$0] [$1] [${2}] [$3] [${134]")
if s then
ngx.say(s, ": ", n)
else
ngx.say("error: ", err)
end
';
}
--- request
GET /re
--- response_body
error: failed to compile the replacement template
--- error_log
the closing bracket in "134" variable is missing
=== TEST 8: matched and with bracketed variables (unmatched brackets)
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("b c d", "(b) (c)", "[$0] [$1] [${2}] [$3] [${134")
if s then
ngx.say(s, ": ", n)
else
ngx.say("error: ", err)
end
';
}
--- request
GET /re
--- response_body
error: failed to compile the replacement template
--- error_log
the closing bracket in "134" variable is missing
=== TEST 9: matched and with bracketed variables (unmatched brackets)
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("b c d", "(b) (c)", "[$0] [$1] [${2}] [$3] [${")
if s then
ngx.say(s, ": ", n)
else
ngx.say("error: ", err)
end
';
}
--- request
GET /re
--- response_body
error: failed to compile the replacement template
--- error_log
lua script: invalid capturing variable name found in "[$0] [$1] [${2}] [$3] [${"
=== TEST 10: trailing $
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("b c d", "(b) (c)", "[$0] [$1] [${2}] [$3] [$")
if s then
ngx.say(s, ": ", n)
else
ngx.say("error: ", err)
end
';
}
--- request
GET /re
--- response_body
error: failed to compile the replacement template
--- error_log
lua script: invalid capturing variable name found in "[$0] [$1] [${2}] [$3] [$"
=== TEST 11: matched but w/o variables and with literal $
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("hello, world", "[a-z]+", "ho$$wdy")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
ho$wdy, world
1
=== TEST 12: non-anchored match
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("hello, 1234", "[0-9]", "x")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
hello, x234
1
=== TEST 13: anchored match
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("hello, 1234", "[0-9]", "x", "a")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
hello, 1234
0
=== TEST 14: function replace
--- config
location /re {
content_by_lua '
local repl = function (m)
return "[" .. m[0] .. "] [" .. m[1] .. "]"
end
local s, n = ngx.re.sub("hello, 34", "([0-9])", repl)
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
hello, [3] [3]4
1
=== TEST 15: function replace (failed)
--- config
location /re {
content_by_lua '
local repl = function (m)
return "[" .. m[0] .. "] [" .. m[1] .. "]"
end
local s, n = ngx.re.sub("hello, 34", "([A-Z])", repl)
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
hello, 34
0
=== TEST 16: bad repl arg type
--- config
location /re {
content_by_lua '
local rc, s, n = pcall(ngx.re.sub, "hello, 34", "([A-Z])", true)
ngx.say(rc)
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
false
bad argument #3 to '?' (string, number, or function expected, got boolean)
nil
--- SKIP
=== TEST 17: use number to replace
--- config
location /re {
content_by_lua '
local rc, s, n = pcall(ngx.re.sub, "hello, 34", "([0-9])", 72)
ngx.say(rc)
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
true
hello, 724
1
=== TEST 18: bad function return value type
--- SKIP
--- config
location /re {
content_by_lua '
local f = function (m) end
local rc, s, n = pcall(ngx.re.sub, "hello, 34", "([0-9])", f)
ngx.say(rc)
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
false
bad argument #3 to '?' (string or number expected to be returned by the replace function, got nil)
nil
=== TEST 19: matched but w/o variables (set_by_lua)
--- config
location /re {
set_by_lua $res '
local s, n = ngx.re.sub("hello, world", "[a-z]+", "howdy")
return s
';
echo $res;
}
--- request
GET /re
--- response_body
howdy, world
=== TEST 20: matched and with variables w/o using named patterns in sub
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("a b c d", "(?<first>b) (?<second>c)", "[$0] [$1] [$2] [$3] [$134]")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
a [b c] [b] [c] [] [] d
1
=== TEST 21: matched and with variables using named patterns in func
--- config
error_log /tmp/nginx_error debug;
location /re {
content_by_lua '
local repl = function (m)
return "[" .. m[0] .. "] [" .. m["first"] .. "] [" .. m[2] .. "]"
end
local s, n = ngx.re.sub("a b c d", "(?<first>b) (?<second>c)", repl)
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
a [b c] [b] [c] d
1
=== TEST 22: matched and with variables w/ using named patterns in sub
This is still a TODO
--- SKIP
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("a b c d", "(?<first>b) (?<second>c)", "[$0] [${first}] [${second}] [$3] [$134]")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
a [b c] [b] [c] [] [] d
1
--- no_error_log
[error]
=== TEST 23: $0 without parens
--- config
location /re {
content_by_lua '
local s, n = ngx.re.sub("a b c d", [[\w]], "[$0]")
ngx.say(s)
ngx.say(n)
';
}
--- request
GET /re
--- response_body
[a] b c d
1
--- no_error_log
[error]
=== TEST 24: bad pattern
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("hello\\nworld", "(abc", "")
if s then
ngx.say("subs: ", n)
else
ngx.say("error: ", err)
end
';
}
--- request
GET /re
--- response_body
error: pcre_compile() failed: missing ) in "(abc"
--- no_error_log
[error]
=== TEST 25: bad UTF-8
--- config
location = /t {
content_by_lua '
local target = "你好"
local regex = "你好"
-- Note the D here
local s, n, err = ngx.re.sub(string.sub(target, 1, 4), regex, "", "u")
if s then
ngx.say(s, ": ", n)
else
ngx.say("error: ", err)
end
';
}
--- request
GET /t
--- response_body_like chop
error: pcre_exec\(\) failed: -10
--- no_error_log
[error]
=== TEST 26: UTF-8 mode without UTF-8 sequence checks
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("你好", ".", "a", "U")
if s then
ngx.say("s: ", s)
end
';
}
--- stap
probe process("$LIBPCRE_PATH").function("pcre_compile") {
printf("compile opts: %x\n", $options)
}
probe process("$LIBPCRE_PATH").function("pcre_exec") {
printf("exec opts: %x\n", $options)
}
--- stap_out
compile opts: 800
exec opts: 2000
--- request
GET /re
--- response_body
s: a
--- no_error_log
[error]
=== TEST 27: UTF-8 mode with UTF-8 sequence checks
--- config
location /re {
content_by_lua '
local s, n, err = ngx.re.sub("你好", ".", "a", "u")
if s then
ngx.say("s: ", s)
end
';
}
--- stap
probe process("$LIBPCRE_PATH").function("pcre_compile") {
printf("compile opts: %x\n", $options)
}
probe process("$LIBPCRE_PATH").function("pcre_exec") {
printf("exec opts: %x\n", $options)
}
--- stap_out
compile opts: 800
exec opts: 0
--- request
GET /re
--- response_body
s: a
--- no_error_log
[error]
=== TEST 28: just hit match limit
--- http_config
lua_regex_match_limit 5600;
--- config
location /re {
content_by_lua_file html/a.lua;
}
--- user_files
>>> a.lua
local re = [==[(?i:([\s'\"`´’‘\(\)]*)?([\d\w]+)([\s'\"`´’‘\(\)]*)?(?:=|<=>|r?like|sounds\s+like|regexp)([\s'\"`´’‘\(\)]*)?\2|([\s'\"`´’‘\(\)]*)?([\d\w]+)([\s'\"`´’‘\(\)]*)?(?:!=|<=|>=|<>|<|>|\^|is\s+not|not\s+like|not\s+regexp)([\s'\"`´’‘\(\)]*)?(?!\6)([\d\w]+))]==]
s = string.rep([[ABCDEFG]], 10)
local start = ngx.now()
local res, cnt, err = ngx.re.sub(s, re, "", "o")
--[[
ngx.update_time()
local elapsed = ngx.now() - start
ngx.say(elapsed, " sec elapsed.")
]]
if err then
ngx.say("error: ", err)
return
end
ngx.say("sub: ", cnt)
--- request
GET /re
--- response_body
error: pcre_exec() failed: -8
=== TEST 29: just not hit match limit
--- http_config
lua_regex_match_limit 5700;
--- config
location /re {
content_by_lua_file html/a.lua;
}
--- user_files
>>> a.lua
local re = [==[(?i:([\s'\"`´’‘\(\)]*)?([\d\w]+)([\s'\"`´’‘\(\)]*)?(?:=|<=>|r?like|sounds\s+like|regexp)([\s'\"`´’‘\(\)]*)?\2|([\s'\"`´’‘\(\)]*)?([\d\w]+)([\s'\"`´’‘\(\)]*)?(?:!=|<=|>=|<>|<|>|\^|is\s+not|not\s+like|not\s+regexp)([\s'\"`´’‘\(\)]*)?(?!\6)([\d\w]+))]==]
local s = string.rep([[ABCDEFG]], 10)
local start = ngx.now()
local res, cnt, err = ngx.re.sub(s, re, "", "o")
--[[
ngx.update_time()
local elapsed = ngx.now() - start
ngx.say(elapsed, " sec elapsed.")
]]
if err then
ngx.say("error: ", err)
return
end
ngx.say("sub: ", cnt)
--- request
GET /re
--- response_body
sub: 0