Merge pull request #123 from f/master
Upgrade ParamParser to make it more convinient.
This commit is contained in:
commit
a194bebe31
7 changed files with 92 additions and 56 deletions
|
@ -3,19 +3,31 @@ require "./spec_helper"
|
||||||
describe "ParamParser" do
|
describe "ParamParser" do
|
||||||
it "parses query params" do
|
it "parses query params" do
|
||||||
route = Route.new "POST", "/" do |env|
|
route = Route.new "POST", "/" do |env|
|
||||||
hasan = env.params["hasan"]
|
hasan = env.params.query["hasan"]
|
||||||
"Hello #{hasan}"
|
"Hello #{hasan}"
|
||||||
end
|
end
|
||||||
request = HTTP::Request.new("POST", "/?hasan=cemal")
|
request = HTTP::Request.new("POST", "/?hasan=cemal")
|
||||||
params = Kemal::ParamParser.new(request).parse
|
query_params = Kemal::ParamParser.new(request).query
|
||||||
params["hasan"].should eq "cemal"
|
query_params["hasan"].should eq "cemal"
|
||||||
|
end
|
||||||
|
|
||||||
|
it "parses url params" do
|
||||||
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
|
kemal.add_route "POST", "/hello/:hasan" do |env|
|
||||||
|
"hello #{env.params.url["hasan"]}"
|
||||||
|
end
|
||||||
|
request = HTTP::Request.new("POST", "/hello/cemal")
|
||||||
|
# Radix tree MUST be run to parse url params.
|
||||||
|
io_with_context = create_request_and_return_io(kemal, request)
|
||||||
|
url_params = Kemal::ParamParser.new(request).url
|
||||||
|
url_params["hasan"].should eq "cemal"
|
||||||
end
|
end
|
||||||
|
|
||||||
it "parses request body" do
|
it "parses request body" do
|
||||||
route = Route.new "POST", "/" do |env|
|
route = Route.new "POST", "/" do |env|
|
||||||
name = env.params["name"]
|
name = env.params.query["name"]
|
||||||
age = env.params["age"]
|
age = env.params.query["age"]
|
||||||
hasan = env.params["hasan"]
|
hasan = env.params.body["hasan"]
|
||||||
"Hello #{name} #{hasan} #{age}"
|
"Hello #{name} #{hasan} #{age}"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -26,8 +38,11 @@ describe "ParamParser" do
|
||||||
headers: HTTP::Headers{"Content-Type": "application/x-www-form-urlencoded"},
|
headers: HTTP::Headers{"Content-Type": "application/x-www-form-urlencoded"},
|
||||||
)
|
)
|
||||||
|
|
||||||
params = Kemal::ParamParser.new(request).parse
|
query_params = Kemal::ParamParser.new(request).query
|
||||||
params.should eq({"hasan" => "cemal", "name" => "serdar", "age" => "99"})
|
query_params.should eq({"hasan" => "cemal"})
|
||||||
|
|
||||||
|
body_params = Kemal::ParamParser.new(request).body
|
||||||
|
body_params.should eq({"name" => "serdar", "age" => "99"})
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when content type is application/json" do
|
context "when content type is application/json" do
|
||||||
|
@ -41,8 +56,8 @@ describe "ParamParser" do
|
||||||
headers: HTTP::Headers{"Content-Type": "application/json"},
|
headers: HTTP::Headers{"Content-Type": "application/json"},
|
||||||
)
|
)
|
||||||
|
|
||||||
params = Kemal::ParamParser.new(request).parse
|
json_params = Kemal::ParamParser.new(request).json
|
||||||
params.should eq({"name": "Serdar"})
|
json_params.should eq({"name": "Serdar"})
|
||||||
end
|
end
|
||||||
|
|
||||||
it "parses request body for array" do
|
it "parses request body for array" do
|
||||||
|
@ -55,8 +70,8 @@ describe "ParamParser" do
|
||||||
headers: HTTP::Headers{"Content-Type": "application/json"},
|
headers: HTTP::Headers{"Content-Type": "application/json"},
|
||||||
)
|
)
|
||||||
|
|
||||||
params = Kemal::ParamParser.new(request).parse
|
json_params = Kemal::ParamParser.new(request).json
|
||||||
params.should eq({"_json": [1]})
|
json_params.should eq({"_json": [1]})
|
||||||
end
|
end
|
||||||
|
|
||||||
it "parses request body and query params" do
|
it "parses request body and query params" do
|
||||||
|
@ -69,8 +84,11 @@ describe "ParamParser" do
|
||||||
headers: HTTP::Headers{"Content-Type": "application/json"},
|
headers: HTTP::Headers{"Content-Type": "application/json"},
|
||||||
)
|
)
|
||||||
|
|
||||||
params = Kemal::ParamParser.new(request).parse
|
query_params = Kemal::ParamParser.new(request).query
|
||||||
params.should eq({"foo": "bar", "_json": [1]})
|
query_params.should eq({"foo": "bar"})
|
||||||
|
|
||||||
|
json_params = Kemal::ParamParser.new(request).json
|
||||||
|
json_params.should eq({"_json": [1]})
|
||||||
end
|
end
|
||||||
|
|
||||||
it "handles no request body" do
|
it "handles no request body" do
|
||||||
|
@ -82,17 +100,26 @@ describe "ParamParser" do
|
||||||
headers: HTTP::Headers{"Content-Type": "application/json"},
|
headers: HTTP::Headers{"Content-Type": "application/json"},
|
||||||
)
|
)
|
||||||
|
|
||||||
params = Kemal::ParamParser.new(request).parse
|
url_params = Kemal::ParamParser.new(request).url
|
||||||
params.should eq({} of String => AllParamTypes)
|
url_params.should eq({} of String => String)
|
||||||
|
|
||||||
|
query_params = Kemal::ParamParser.new(request).query
|
||||||
|
query_params.should eq({} of String => String)
|
||||||
|
|
||||||
|
body_params = Kemal::ParamParser.new(request).body
|
||||||
|
body_params.should eq({} of String => String)
|
||||||
|
|
||||||
|
json_params = Kemal::ParamParser.new(request).json
|
||||||
|
json_params.should eq({} of String => AllParamTypes)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when content type is incorrect" do
|
context "when content type is incorrect" do
|
||||||
it "does not parse request body" do
|
it "does not parse request body" do
|
||||||
route = Route.new "POST", "/" do |env|
|
route = Route.new "POST", "/" do |env|
|
||||||
name = env.params["name"]
|
name = env.params.body["name"]
|
||||||
age = env.params["age"]
|
age = env.params.body["age"]
|
||||||
hasan = env.params["hasan"]
|
hasan = env.params.query["hasan"]
|
||||||
"Hello #{name} #{hasan} #{age}"
|
"Hello #{name} #{hasan} #{age}"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -103,8 +130,11 @@ describe "ParamParser" do
|
||||||
headers: HTTP::Headers{"Content-Type": "text/plain"},
|
headers: HTTP::Headers{"Content-Type": "text/plain"},
|
||||||
)
|
)
|
||||||
|
|
||||||
params = Kemal::ParamParser.new(request).parse
|
query_params = Kemal::ParamParser.new(request).query
|
||||||
params.should eq({"hasan" => "cemal"})
|
query_params.should eq({"hasan" => "cemal"})
|
||||||
|
|
||||||
|
body_params = Kemal::ParamParser.new(request).body
|
||||||
|
body_params.should eq({} of String => String)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -15,7 +15,7 @@ describe "Kemal::RouteHandler" do
|
||||||
it "routes request with query string" do
|
it "routes request with query string" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "GET", "/" do |env|
|
kemal.add_route "GET", "/" do |env|
|
||||||
"hello #{env.params["message"]}"
|
"hello #{env.params.query["message"]}"
|
||||||
end
|
end
|
||||||
request = HTTP::Request.new("GET", "/?message=world")
|
request = HTTP::Request.new("GET", "/?message=world")
|
||||||
io_with_context = create_request_and_return_io(kemal, request)
|
io_with_context = create_request_and_return_io(kemal, request)
|
||||||
|
@ -26,7 +26,7 @@ describe "Kemal::RouteHandler" do
|
||||||
it "routes request with multiple query strings" do
|
it "routes request with multiple query strings" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "GET", "/" do |env|
|
kemal.add_route "GET", "/" do |env|
|
||||||
"hello #{env.params["message"]} time #{env.params["time"]}"
|
"hello #{env.params.query["message"]} time #{env.params.query["time"]}"
|
||||||
end
|
end
|
||||||
request = HTTP::Request.new("GET", "/?message=world&time=now")
|
request = HTTP::Request.new("GET", "/?message=world&time=now")
|
||||||
io_with_context = create_request_and_return_io(kemal, request)
|
io_with_context = create_request_and_return_io(kemal, request)
|
||||||
|
@ -37,7 +37,7 @@ describe "Kemal::RouteHandler" do
|
||||||
it "route parameter has more precedence than query string arguments" do
|
it "route parameter has more precedence than query string arguments" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "GET", "/:message" do |env|
|
kemal.add_route "GET", "/:message" do |env|
|
||||||
"hello #{env.params["message"]}"
|
"hello #{env.params.url["message"]}"
|
||||||
end
|
end
|
||||||
request = HTTP::Request.new("GET", "/world?message=coco")
|
request = HTTP::Request.new("GET", "/world?message=coco")
|
||||||
io_with_context = create_request_and_return_io(kemal, request)
|
io_with_context = create_request_and_return_io(kemal, request)
|
||||||
|
@ -48,8 +48,8 @@ describe "Kemal::RouteHandler" do
|
||||||
it "parses simple JSON body" do
|
it "parses simple JSON body" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "POST", "/" do |env|
|
kemal.add_route "POST", "/" do |env|
|
||||||
name = env.params["name"]
|
name = env.params.json["name"]
|
||||||
age = env.params["age"]
|
age = env.params.json["age"]
|
||||||
"Hello #{name} Age #{age}"
|
"Hello #{name} Age #{age}"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -68,7 +68,7 @@ describe "Kemal::RouteHandler" do
|
||||||
it "parses JSON with string array" do
|
it "parses JSON with string array" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "POST", "/" do |env|
|
kemal.add_route "POST", "/" do |env|
|
||||||
skills = env.params["skills"] as Array
|
skills = env.params.json["skills"] as Array
|
||||||
"Skills #{skills.each.join(',')}"
|
"Skills #{skills.each.join(',')}"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -87,7 +87,7 @@ describe "Kemal::RouteHandler" do
|
||||||
it "parses JSON with json object array" do
|
it "parses JSON with json object array" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "POST", "/" do |env|
|
kemal.add_route "POST", "/" do |env|
|
||||||
skills = env.params["skills"] as Array
|
skills = env.params.json["skills"] as Array
|
||||||
skills_from_languages = skills.map do |skill|
|
skills_from_languages = skills.map do |skill|
|
||||||
skill = skill as Hash
|
skill = skill as Hash
|
||||||
skill["language"]
|
skill["language"]
|
||||||
|
@ -170,8 +170,8 @@ describe "Kemal::RouteHandler" do
|
||||||
request = HTTP::Request.new(
|
request = HTTP::Request.new(
|
||||||
"POST",
|
"POST",
|
||||||
"/",
|
"/",
|
||||||
body: json_payload.to_json,
|
body: "_method=DELETE",
|
||||||
headers: HTTP::Headers{"Content-Type": "application/json"}
|
headers: HTTP::Headers{"Content-Type": "application/x-www-form-urlencoded; charset=UTF-8"}
|
||||||
)
|
)
|
||||||
io_with_context = create_request_and_return_io(kemal, request)
|
io_with_context = create_request_and_return_io(kemal, request)
|
||||||
client_response = HTTP::Client::Response.from_io(io_with_context, decompress: false)
|
client_response = HTTP::Client::Response.from_io(io_with_context, decompress: false)
|
||||||
|
|
|
@ -8,7 +8,7 @@ describe "Views" do
|
||||||
it "renders file" do
|
it "renders file" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "GET", "/view/:name" do |env|
|
kemal.add_route "GET", "/view/:name" do |env|
|
||||||
name = env.params["name"]
|
name = env.params.url["name"]
|
||||||
render "spec/asset/hello.ecr"
|
render "spec/asset/hello.ecr"
|
||||||
end
|
end
|
||||||
request = HTTP::Request.new("GET", "/view/world")
|
request = HTTP::Request.new("GET", "/view/world")
|
||||||
|
@ -20,7 +20,7 @@ describe "Views" do
|
||||||
it "renders file with dynamic variables" do
|
it "renders file with dynamic variables" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "GET", "/view/:name" do |env|
|
kemal.add_route "GET", "/view/:name" do |env|
|
||||||
name = env.params["name"]
|
name = env.params.url["name"]
|
||||||
render_with_base_and_layout "hello.ecr"
|
render_with_base_and_layout "hello.ecr"
|
||||||
end
|
end
|
||||||
request = HTTP::Request.new("GET", "/view/world")
|
request = HTTP::Request.new("GET", "/view/world")
|
||||||
|
@ -32,7 +32,7 @@ describe "Views" do
|
||||||
it "renders layout" do
|
it "renders layout" do
|
||||||
kemal = Kemal::RouteHandler::INSTANCE
|
kemal = Kemal::RouteHandler::INSTANCE
|
||||||
kemal.add_route "GET", "/view/:name" do |env|
|
kemal.add_route "GET", "/view/:name" do |env|
|
||||||
name = env.params["name"]
|
name = env.params.url["name"]
|
||||||
render "spec/asset/hello.ecr", "spec/asset/layout.ecr"
|
render "spec/asset/hello.ecr", "spec/asset/layout.ecr"
|
||||||
end
|
end
|
||||||
request = HTTP::Request.new("GET", "/view/world")
|
request = HTTP::Request.new("GET", "/view/world")
|
||||||
|
|
|
@ -18,7 +18,7 @@ at_exit do
|
||||||
|
|
||||||
# This route serves the built-in images for not_found and exceptions.
|
# This route serves the built-in images for not_found and exceptions.
|
||||||
get "/__kemal__/:image" do |env|
|
get "/__kemal__/:image" do |env|
|
||||||
image = env.params["image"]
|
image = env.params.url["image"]
|
||||||
file_path = File.expand_path("libs/kemal/images/#{image}", Dir.current)
|
file_path = File.expand_path("libs/kemal/images/#{image}", Dir.current)
|
||||||
if File.exists? file_path
|
if File.exists? file_path
|
||||||
env.response.headers.add "Content-Type", "application/octet-stream"
|
env.response.headers.add "Content-Type", "application/octet-stream"
|
||||||
|
|
|
@ -4,7 +4,7 @@ class HTTP::Server
|
||||||
class Context
|
class Context
|
||||||
def params
|
def params
|
||||||
@request.url_params = route_lookup.params
|
@request.url_params = route_lookup.params
|
||||||
@params ||= Kemal::ParamParser.new(@request).parse
|
@params ||= Kemal::ParamParser.new(@request)
|
||||||
end
|
end
|
||||||
|
|
||||||
def redirect(url, status_code = 302)
|
def redirect(url, status_code = 302)
|
||||||
|
|
|
@ -10,34 +10,37 @@ class Kemal::ParamParser
|
||||||
APPLICATION_JSON = "application/json"
|
APPLICATION_JSON = "application/json"
|
||||||
|
|
||||||
def initialize(@request)
|
def initialize(@request)
|
||||||
@params = {} of String => AllParamTypes
|
@url = {} of String => String
|
||||||
|
@query = {} of String => String
|
||||||
|
@body = {} of String => String
|
||||||
|
@json = {} of String => AllParamTypes
|
||||||
end
|
end
|
||||||
|
|
||||||
def parse
|
{% for method in %w(url query body json) %}
|
||||||
parse_request
|
def {{method.id}}
|
||||||
end
|
# check memoization
|
||||||
|
return @{{method.id}} if @{{method.id}}_parsed
|
||||||
|
|
||||||
def parse_request
|
parse_{{method.id}}
|
||||||
parse_query
|
# memoize
|
||||||
parse_body
|
@{{method.id}}_parsed = true
|
||||||
parse_json
|
@{{method.id}}
|
||||||
parse_url_params
|
|
||||||
@params
|
|
||||||
end
|
end
|
||||||
|
{% end %}
|
||||||
|
|
||||||
def parse_body
|
def parse_body
|
||||||
return if (@request.headers["Content-Type"]? =~ /#{URL_ENCODED_FORM}/).nil?
|
return if (@request.headers["Content-Type"]? =~ /#{URL_ENCODED_FORM}/).nil?
|
||||||
parse_part(@request.body)
|
@body = parse_part(@request.body)
|
||||||
end
|
end
|
||||||
|
|
||||||
def parse_query
|
def parse_query
|
||||||
parse_part(@request.query)
|
@query = parse_part(@request.query)
|
||||||
end
|
end
|
||||||
|
|
||||||
def parse_url_params
|
def parse_url
|
||||||
if params = @request.url_params
|
if params = @request.url_params
|
||||||
params.each do |key, value|
|
params.each do |key, value|
|
||||||
@params[key] = value
|
@url[key as String] = value as String
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -52,18 +55,21 @@ class Kemal::ParamParser
|
||||||
body = @request.body as String
|
body = @request.body as String
|
||||||
case json = JSON.parse(body).raw
|
case json = JSON.parse(body).raw
|
||||||
when Hash
|
when Hash
|
||||||
json.each do |k, v|
|
json.each do |key, value|
|
||||||
@params[k as String] = v as AllParamTypes
|
@json[key as String] = value as AllParamTypes
|
||||||
end
|
end
|
||||||
when Array
|
when Array
|
||||||
@params["_json"] = json
|
@json["_json"] = json
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def parse_part(part)
|
def parse_part(part)
|
||||||
return unless part
|
part_params = {} of String => String
|
||||||
|
if part
|
||||||
HTTP::Params.parse(part) do |key, value|
|
HTTP::Params.parse(part) do |key, value|
|
||||||
@params[key] ||= value
|
part_params[key as String] ||= value as String
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
part_params
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -11,7 +11,7 @@ class HTTP::Request
|
||||||
private def check_for_method_override!
|
private def check_for_method_override!
|
||||||
@override_method = @method
|
@override_method = @method
|
||||||
if @method == "POST"
|
if @method == "POST"
|
||||||
params = Kemal::ParamParser.new(self).parse_request
|
params = Kemal::ParamParser.new(self).body
|
||||||
if params.has_key?("_method") && HTTP::Request.override_method_valid?(params["_method"])
|
if params.has_key?("_method") && HTTP::Request.override_method_valid?(params["_method"])
|
||||||
@override_method = params["_method"]
|
@override_method = params["_method"]
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in a new issue