feat: add option to normalize uri like servlet (#6984)

Signed-off-by: spacewander <spacewanderlzx@gmail.com>
This commit is contained in:
罗泽轩 2022-05-07 10:25:01 +08:00 committed by GitHub
parent 2b7bcf0034
commit 63f69fe116
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 165 additions and 5 deletions

View File

@ -50,6 +50,7 @@ local error = error
local ipairs = ipairs
local ngx_now = ngx.now
local ngx_var = ngx.var
local re_split = require("ngx.re").split
local str_byte = string.byte
local str_sub = string.sub
local tonumber = tonumber
@ -262,6 +263,44 @@ local function verify_tls_client(ctx)
end
local function normalize_uri_like_servlet(uri)
local found = core.string.find(uri, ';')
if not found then
return uri
end
local segs, err = re_split(uri, "/", "jo")
if not segs then
return nil, err
end
local len = #segs
for i = 1, len do
local seg = segs[i]
local pos = core.string.find(seg, ';')
if pos then
seg = seg:sub(1, pos - 1)
-- reject bad uri which bypasses with ';'
if seg == "." or seg == ".." then
return nil, "dot segment with parameter"
end
if seg == "" and i < len then
return nil, "empty segment with parameters"
end
segs[i] = seg
seg = seg:lower()
if seg == "%2e" or seg == "%2e%2e" then
return nil, "encoded dot segment"
end
end
end
return core.table.concat(segs, '/')
end
local function common_phase(phase_name)
local api_ctx = ngx.ctx.api_ctx
if not api_ctx then
@ -295,11 +334,25 @@ function _M.http_access_phase()
debug.dynamic_debug(api_ctx)
local uri = api_ctx.var.uri
if local_conf.apisix and local_conf.apisix.delete_uri_tail_slash then
if str_byte(uri, #uri) == str_byte("/") then
api_ctx.var.uri = str_sub(api_ctx.var.uri, 1, #uri - 1)
core.log.info("remove the end of uri '/', current uri: ",
api_ctx.var.uri)
if local_conf.apisix then
if local_conf.apisix.delete_uri_tail_slash then
if str_byte(uri, #uri) == str_byte("/") then
api_ctx.var.uri = str_sub(api_ctx.var.uri, 1, #uri - 1)
core.log.info("remove the end of uri '/', current uri: ", api_ctx.var.uri)
end
end
if local_conf.apisix.normalize_uri_like_servlet then
local new_uri, err = normalize_uri_like_servlet(uri)
if not new_uri then
core.log.error("failed to normalize: ", err)
return core.response.exit(400)
end
api_ctx.var.uri = new_uri
-- forward the original uri so the servlet upstream
-- can consume the param after ';'
api_ctx.var.upstream_uri = uri
end
end

View File

@ -101,6 +101,11 @@ apisix:
role: viewer
delete_uri_tail_slash: false # delete the '/' at the end of the URI
# The URI normalization in servlet is a little different from the RFC's.
# See https://github.com/jakartaee/servlet/blob/master/spec/src/main/asciidoc/servlet-spec-body.adoc#352-uri-path-canonicalization,
# which is used under Tomcat.
# Turn this option on if you want to be compatible with servlet when matching URI path.
normalize_uri_like_servlet: false
router:
http: radixtree_uri # radixtree_uri: match route by uri(base on radixtree)
# radixtree_host_uri: match route by host + uri(base on radixtree)

View File

@ -22,6 +22,13 @@ worker_connections(256);
no_root_location();
no_shuffle();
our $servlet_yaml_config = <<_EOC_;
apisix:
node_listen: 1984
admin_key: null
normalize_uri_like_servlet: true
_EOC_
run_tests();
__DATA__
@ -295,3 +302,98 @@ GET /hello/
--- error_code: 404
--- no_error_log
[error]
--- response_body
{"error_msg":"404 Route Not Found"}
=== TEST 16: match route like servlet
--- yaml_config eval: $::servlet_yaml_config
--- request
GET /hello;world
--- response_body eval
qr/404 Not Found/
--- error_code: 404
--- no_error_log
[error]
=== TEST 17: plugin should work on the normalized url
--- config
location /t {
content_by_lua_block {
local t = require("lib.test_admin").test
local code, body = t('/apisix/admin/routes/1',
ngx.HTTP_PUT,
[[{
"upstream": {
"nodes": {
"127.0.0.1:1980": 1
},
"type": "roundrobin"
},
"uri": "/*",
"plugins": {
"uri-blocker": {
"block_rules": ["/hello/world"]
}
}
}]]
)
if code >= 300 then
ngx.status = code
end
ngx.say(body)
}
}
--- request
GET /t
--- response_body
passed
--- no_error_log
[error]
=== TEST 18: hit
--- yaml_config eval: $::servlet_yaml_config
--- request
GET /hello;a=b/world;a/;
--- error_code: 403
--- no_error_log
[error]
=== TEST 19: reject bad uri
--- yaml_config eval: $::servlet_yaml_config
--- config
location /t {
content_by_lua_block {
local http = require "resty.http"
local uri = "http://127.0.0.1:" .. ngx.var.server_port
for _, path in ipairs({
"/;/a", "/%2e;", "/%2E%2E;", "/.;", "/..;",
"/%2E%2e;", "/b/;/c"
}) do
local httpc = http.new()
local res, err = httpc:request_uri(uri .. path)
if not res then
ngx.say(err)
return
end
if res.status ~= 400 then
ngx.say(path, " ", res.status)
end
end
ngx.say("ok")
}
}
--- request
GET /t
--- response_body
ok