2011-03-07 22:59:22 +01:00
|
|
|
%% Copyright (c) 2011, Loïc Hoguin <essen@dev-extend.eu>
|
2011-03-29 13:49:48 +02:00
|
|
|
%% Copyright (c) 2011, Anthony Ramine <nox@dev-extend.eu>
|
2011-03-07 22:59:22 +01:00
|
|
|
%%
|
|
|
|
%% Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
%% purpose with or without fee is hereby granted, provided that the above
|
|
|
|
%% copyright notice and this permission notice appear in all copies.
|
|
|
|
%%
|
|
|
|
%% THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
%% WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
%% MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
%% ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
%% WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
%% ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
%% OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc HTTP request manipulation API.
|
|
|
|
%%
|
|
|
|
%% Almost all functions in this module return a new <em>Req</em> variable.
|
|
|
|
%% It should always be used instead of the one used in your function call
|
|
|
|
%% because it keeps the state of the request. It also allows Cowboy to do
|
|
|
|
%% some lazy evaluation and cache results where possible.
|
2011-03-07 22:59:22 +01:00
|
|
|
-module(cowboy_http_req).
|
2011-03-18 22:38:26 +01:00
|
|
|
|
2011-03-07 22:59:22 +01:00
|
|
|
-export([
|
2011-03-18 22:38:26 +01:00
|
|
|
method/1, version/1, peer/1,
|
2011-05-09 14:31:06 +02:00
|
|
|
host/1, host_info/1, raw_host/1, port/1,
|
|
|
|
path/1, path_info/1, raw_path/1,
|
2011-03-07 22:59:22 +01:00
|
|
|
qs_val/2, qs_val/3, qs_vals/1, raw_qs/1,
|
|
|
|
binding/2, binding/3, bindings/1,
|
2011-07-08 13:41:30 -05:00
|
|
|
header/2, header/3, headers/1,
|
2011-10-05 03:17:13 +02:00
|
|
|
parse_header/2, parse_header/3,
|
2011-07-08 13:41:30 -05:00
|
|
|
cookie/2, cookie/3, cookies/1
|
2011-03-18 22:38:26 +01:00
|
|
|
]). %% Request API.
|
|
|
|
|
2011-03-21 17:26:00 +01:00
|
|
|
-export([
|
2011-03-22 13:20:21 +01:00
|
|
|
body/1, body/2, body_qs/1
|
2011-03-21 17:26:00 +01:00
|
|
|
]). %% Request Body API.
|
|
|
|
|
2011-03-18 22:38:26 +01:00
|
|
|
-export([
|
2011-12-07 11:54:57 +01:00
|
|
|
set_resp_cookie/4, set_resp_header/3, set_resp_body/2,
|
2011-11-28 09:09:41 +01:00
|
|
|
has_resp_header/2, has_resp_body/1,
|
2011-10-13 16:16:53 +02:00
|
|
|
reply/2, reply/3, reply/4,
|
2011-10-20 14:11:17 +02:00
|
|
|
chunked_reply/2, chunked_reply/3, chunk/2,
|
|
|
|
upgrade_reply/3
|
2011-03-18 22:38:26 +01:00
|
|
|
]). %% Response API.
|
2011-03-07 22:59:22 +01:00
|
|
|
|
2011-06-01 12:49:03 +02:00
|
|
|
-export([
|
|
|
|
compact/1
|
|
|
|
]). %% Misc API.
|
|
|
|
|
2011-03-07 22:59:22 +01:00
|
|
|
-include("include/http.hrl").
|
|
|
|
-include_lib("eunit/include/eunit.hrl").
|
|
|
|
|
2011-03-18 22:38:26 +01:00
|
|
|
%% Request API.
|
2011-03-07 22:59:22 +01:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the HTTP method of the request.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec method(#http_req{}) -> {http_method(), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
method(Req) ->
|
|
|
|
{Req#http_req.method, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the HTTP version used for the request.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec version(#http_req{}) -> {http_version(), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
version(Req) ->
|
|
|
|
{Req#http_req.version, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the peer address and port number of the remote host.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec peer(#http_req{}) -> {{inet:ip_address(), inet:ip_port()}, #http_req{}}.
|
2011-03-20 16:09:05 +01:00
|
|
|
peer(Req=#http_req{socket=Socket, transport=Transport, peer=undefined}) ->
|
|
|
|
{ok, Peer} = Transport:peername(Socket),
|
|
|
|
{Peer, Req#http_req{peer=Peer}};
|
2011-03-07 22:59:22 +01:00
|
|
|
peer(Req) ->
|
|
|
|
{Req#http_req.peer, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the tokens for the hostname requested.
|
2011-10-01 02:32:20 +02:00
|
|
|
-spec host(#http_req{}) -> {cowboy_dispatcher:tokens(), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
host(Req) ->
|
|
|
|
{Req#http_req.host, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the extra host information obtained from partially matching
|
|
|
|
%% the hostname using <em>'...'</em>.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec host_info(#http_req{})
|
2011-10-01 02:32:20 +02:00
|
|
|
-> {cowboy_dispatcher:tokens() | undefined, #http_req{}}.
|
2011-05-09 14:31:06 +02:00
|
|
|
host_info(Req) ->
|
|
|
|
{Req#http_req.host_info, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the raw host directly taken from the request.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec raw_host(#http_req{}) -> {binary(), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
raw_host(Req) ->
|
|
|
|
{Req#http_req.raw_host, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the port used for this request.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec port(#http_req{}) -> {inet:ip_port(), #http_req{}}.
|
2011-05-04 12:05:57 +02:00
|
|
|
port(Req) ->
|
|
|
|
{Req#http_req.port, Req}.
|
|
|
|
|
2011-09-29 16:53:47 +02:00
|
|
|
%% @doc Return the path segments for the path requested.
|
|
|
|
%%
|
2011-10-01 02:17:50 +02:00
|
|
|
%% Following RFC2396, this function may return path segments containing any
|
2011-09-29 16:53:47 +02:00
|
|
|
%% character, including <em>/</em> if, and only if, a <em>/</em> was escaped
|
|
|
|
%% and part of a path segment in the path requested.
|
2011-10-01 02:32:20 +02:00
|
|
|
-spec path(#http_req{}) -> {cowboy_dispatcher:tokens(), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
path(Req) ->
|
|
|
|
{Req#http_req.path, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the extra path information obtained from partially matching
|
|
|
|
%% the patch using <em>'...'</em>.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec path_info(#http_req{})
|
2011-10-01 02:32:20 +02:00
|
|
|
-> {cowboy_dispatcher:tokens() | undefined, #http_req{}}.
|
2011-05-09 14:31:06 +02:00
|
|
|
path_info(Req) ->
|
|
|
|
{Req#http_req.path_info, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the raw path directly taken from the request.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec raw_path(#http_req{}) -> {binary(), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
raw_path(Req) ->
|
|
|
|
{Req#http_req.raw_path, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @equiv qs_val(Name, Req, undefined)
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec qs_val(binary(), #http_req{})
|
|
|
|
-> {binary() | true | undefined, #http_req{}}.
|
2011-07-18 14:21:45 +02:00
|
|
|
qs_val(Name, Req) when is_binary(Name) ->
|
2011-03-29 13:57:21 +02:00
|
|
|
qs_val(Name, Req, undefined).
|
2011-03-07 22:59:22 +01:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the query string value for the given key, or a default if
|
|
|
|
%% missing.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec qs_val(binary(), #http_req{}, Default)
|
|
|
|
-> {binary() | true | Default, #http_req{}} when Default::any().
|
2011-12-05 01:08:38 +01:00
|
|
|
qs_val(Name, Req=#http_req{raw_qs=RawQs, qs_vals=undefined,
|
|
|
|
urldecode={URLDecFun, URLDecArg}}, Default) when is_binary(Name) ->
|
|
|
|
QsVals = parse_qs(RawQs, fun(Bin) -> URLDecFun(Bin, URLDecArg) end),
|
2011-03-29 13:57:21 +02:00
|
|
|
qs_val(Name, Req#http_req{qs_vals=QsVals}, Default);
|
|
|
|
qs_val(Name, Req, Default) ->
|
2011-03-29 13:49:48 +02:00
|
|
|
case lists:keyfind(Name, 1, Req#http_req.qs_vals) of
|
|
|
|
{Name, Value} -> {Value, Req};
|
|
|
|
false -> {Default, Req}
|
|
|
|
end.
|
2011-03-07 22:59:22 +01:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the full list of query string values.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec qs_vals(#http_req{}) -> {list({binary(), binary() | true}), #http_req{}}.
|
2011-12-05 01:08:38 +01:00
|
|
|
qs_vals(Req=#http_req{raw_qs=RawQs, qs_vals=undefined,
|
|
|
|
urldecode={URLDecFun, URLDecArg}}) ->
|
|
|
|
QsVals = parse_qs(RawQs, fun(Bin) -> URLDecFun(Bin, URLDecArg) end),
|
2011-03-07 22:59:22 +01:00
|
|
|
qs_vals(Req#http_req{qs_vals=QsVals});
|
|
|
|
qs_vals(Req=#http_req{qs_vals=QsVals}) ->
|
|
|
|
{QsVals, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the raw query string directly taken from the request.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec raw_qs(#http_req{}) -> {binary(), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
raw_qs(Req) ->
|
|
|
|
{Req#http_req.raw_qs, Req}.
|
|
|
|
|
2011-06-30 01:11:11 +02:00
|
|
|
%% @equiv binding(Name, Req, undefined)
|
2011-07-06 17:42:20 +02:00
|
|
|
-spec binding(atom(), #http_req{}) -> {binary() | undefined, #http_req{}}.
|
2011-07-18 14:21:45 +02:00
|
|
|
binding(Name, Req) when is_atom(Name) ->
|
2011-03-29 13:57:21 +02:00
|
|
|
binding(Name, Req, undefined).
|
2011-03-29 13:49:48 +02:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the binding value for the given key obtained when matching
|
|
|
|
%% the host and path against the dispatch list, or a default if missing.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec binding(atom(), #http_req{}, Default)
|
|
|
|
-> {binary() | Default, #http_req{}} when Default::any().
|
2011-07-18 14:21:45 +02:00
|
|
|
binding(Name, Req, Default) when is_atom(Name) ->
|
2011-03-27 13:11:57 +02:00
|
|
|
case lists:keyfind(Name, 1, Req#http_req.bindings) of
|
|
|
|
{Name, Value} -> {Value, Req};
|
2011-03-29 13:49:48 +02:00
|
|
|
false -> {Default, Req}
|
2011-03-27 13:11:57 +02:00
|
|
|
end.
|
2011-03-07 22:59:22 +01:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the full list of binding values.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec bindings(#http_req{}) -> {list({atom(), binary()}), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
bindings(Req) ->
|
|
|
|
{Req#http_req.bindings, Req}.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @equiv header(Name, Req, undefined)
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec header(atom() | binary(), #http_req{})
|
|
|
|
-> {binary() | undefined, #http_req{}}.
|
2011-07-18 14:21:45 +02:00
|
|
|
header(Name, Req) when is_atom(Name) orelse is_binary(Name) ->
|
2011-03-29 13:57:21 +02:00
|
|
|
header(Name, Req, undefined).
|
2011-03-29 13:49:48 +02:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the header value for the given key, or a default if missing.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec header(atom() | binary(), #http_req{}, Default)
|
|
|
|
-> {binary() | Default, #http_req{}} when Default::any().
|
2011-07-18 14:21:45 +02:00
|
|
|
header(Name, Req, Default) when is_atom(Name) orelse is_binary(Name) ->
|
2011-03-21 17:26:00 +01:00
|
|
|
case lists:keyfind(Name, 1, Req#http_req.headers) of
|
|
|
|
{Name, Value} -> {Value, Req};
|
2011-03-29 13:49:48 +02:00
|
|
|
false -> {Default, Req}
|
2011-03-21 17:26:00 +01:00
|
|
|
end.
|
2011-03-07 22:59:22 +01:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the full list of headers.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec headers(#http_req{}) -> {http_headers(), #http_req{}}.
|
2011-03-07 22:59:22 +01:00
|
|
|
headers(Req) ->
|
|
|
|
{Req#http_req.headers, Req}.
|
|
|
|
|
2011-10-05 03:17:13 +02:00
|
|
|
%% @doc Semantically parse headers.
|
|
|
|
%%
|
|
|
|
%% When the value isn't found, a proper default value for the type
|
|
|
|
%% returned is used as a return value.
|
|
|
|
%% @see parse_header/3
|
|
|
|
-spec parse_header(http_header(), #http_req{})
|
2011-10-26 04:07:08 +02:00
|
|
|
-> {any(), #http_req{}} | {error, badarg}.
|
|
|
|
parse_header(Name, Req=#http_req{p_headers=PHeaders}) ->
|
|
|
|
case lists:keyfind(Name, 1, PHeaders) of
|
|
|
|
false -> parse_header(Name, Req, parse_header_default(Name));
|
|
|
|
{Name, Value} -> {Value, Req}
|
|
|
|
end.
|
|
|
|
|
|
|
|
%% @doc Default values for semantic header parsing.
|
|
|
|
-spec parse_header_default(http_header()) -> any().
|
2011-11-24 21:33:54 +01:00
|
|
|
parse_header_default('Accept') -> undefined;
|
|
|
|
parse_header_default('Accept-Charset') -> undefined;
|
|
|
|
parse_header_default('Accept-Encoding') -> undefined;
|
|
|
|
parse_header_default('Accept-Language') -> undefined;
|
2011-10-26 04:07:08 +02:00
|
|
|
parse_header_default('Connection') -> [];
|
2011-12-05 10:28:45 +01:00
|
|
|
parse_header_default('If-Match') -> undefined;
|
|
|
|
parse_header_default('If-None-Match') -> undefined;
|
2011-10-26 04:07:08 +02:00
|
|
|
parse_header_default(_Name) -> undefined.
|
2011-10-05 03:17:13 +02:00
|
|
|
|
|
|
|
%% @doc Semantically parse headers.
|
|
|
|
%%
|
2011-10-26 04:07:08 +02:00
|
|
|
%% When the header is unknown, the value is returned directly without parsing.
|
2011-10-05 03:17:13 +02:00
|
|
|
-spec parse_header(http_header(), #http_req{}, any())
|
2011-10-26 04:07:08 +02:00
|
|
|
-> {any(), #http_req{}} | {error, badarg}.
|
|
|
|
parse_header(Name, Req, Default) when Name =:= 'Accept' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
|
|
|
cowboy_http:list(Value, fun cowboy_http:media_range/2)
|
|
|
|
end);
|
2011-10-26 20:54:21 +02:00
|
|
|
parse_header(Name, Req, Default) when Name =:= 'Accept-Charset' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
2011-11-04 10:38:04 +01:00
|
|
|
cowboy_http:nonempty_list(Value, fun cowboy_http:conneg/2)
|
2011-10-26 20:54:21 +02:00
|
|
|
end);
|
2011-10-26 19:23:06 +02:00
|
|
|
parse_header(Name, Req, Default) when Name =:= 'Accept-Encoding' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
2011-11-04 10:38:04 +01:00
|
|
|
cowboy_http:list(Value, fun cowboy_http:conneg/2)
|
2011-10-26 19:23:06 +02:00
|
|
|
end);
|
2011-11-07 00:53:19 +01:00
|
|
|
parse_header(Name, Req, Default) when Name =:= 'Accept-Language' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
|
|
|
cowboy_http:nonempty_list(Value, fun cowboy_http:language_range/2)
|
|
|
|
end);
|
2011-10-26 04:07:08 +02:00
|
|
|
parse_header(Name, Req, Default) when Name =:= 'Connection' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
|
|
|
cowboy_http:nonempty_list(Value, fun cowboy_http:token_ci/2)
|
|
|
|
end);
|
2011-11-04 11:42:36 +01:00
|
|
|
parse_header(Name, Req, Default) when Name =:= 'Content-Length' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
|
|
|
cowboy_http:digits(Value)
|
|
|
|
end);
|
2011-11-04 15:55:55 +01:00
|
|
|
parse_header(Name, Req, Default) when Name =:= 'Content-Type' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
|
|
|
cowboy_http:content_type(Value)
|
|
|
|
end);
|
2011-11-09 21:41:33 +01:00
|
|
|
parse_header(Name, Req, Default)
|
|
|
|
when Name =:= 'If-Match'; Name =:= 'If-None-Match' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
|
|
|
cowboy_http:entity_tag_match(Value)
|
|
|
|
end);
|
2011-11-05 01:49:03 +01:00
|
|
|
parse_header(Name, Req, Default)
|
|
|
|
when Name =:= 'If-Modified-Since'; Name =:= 'If-Unmodified-Since' ->
|
|
|
|
parse_header(Name, Req, Default,
|
|
|
|
fun (Value) ->
|
|
|
|
cowboy_http:http_date(Value)
|
|
|
|
end);
|
2011-10-26 04:07:08 +02:00
|
|
|
parse_header(Name, Req, Default) ->
|
|
|
|
{Value, Req2} = header(Name, Req, Default),
|
|
|
|
{undefined, Value, Req2}.
|
|
|
|
|
|
|
|
parse_header(Name, Req=#http_req{p_headers=PHeaders}, Default, Fun) ->
|
2011-10-05 03:17:13 +02:00
|
|
|
case header(Name, Req) of
|
2011-10-26 04:07:08 +02:00
|
|
|
{undefined, Req2} ->
|
|
|
|
{Default, Req2#http_req{p_headers=[{Name, Default}|PHeaders]}};
|
2011-10-05 03:17:13 +02:00
|
|
|
{Value, Req2} ->
|
2011-10-26 04:07:08 +02:00
|
|
|
case Fun(Value) of
|
2011-10-05 03:17:13 +02:00
|
|
|
{error, badarg} ->
|
|
|
|
{error, badarg};
|
|
|
|
P ->
|
2011-10-26 04:07:08 +02:00
|
|
|
{P, Req2#http_req{p_headers=[{Name, P}|PHeaders]}}
|
2011-10-05 03:17:13 +02:00
|
|
|
end
|
2011-10-26 04:07:08 +02:00
|
|
|
end.
|
2011-10-05 03:17:13 +02:00
|
|
|
|
2011-07-08 13:41:30 -05:00
|
|
|
%% @equiv cookie(Name, Req, undefined)
|
|
|
|
-spec cookie(binary(), #http_req{})
|
|
|
|
-> {binary() | true | undefined, #http_req{}}.
|
2011-07-18 15:50:29 -05:00
|
|
|
cookie(Name, Req) when is_binary(Name) ->
|
2011-07-08 13:41:30 -05:00
|
|
|
cookie(Name, Req, undefined).
|
|
|
|
|
|
|
|
%% @doc Return the cookie value for the given key, or a default if
|
|
|
|
%% missing.
|
|
|
|
-spec cookie(binary(), #http_req{}, Default)
|
|
|
|
-> {binary() | true | Default, #http_req{}} when Default::any().
|
2011-07-18 15:50:29 -05:00
|
|
|
cookie(Name, Req=#http_req{cookies=undefined}, Default) when is_binary(Name) ->
|
2011-07-08 13:41:30 -05:00
|
|
|
case header('Cookie', Req) of
|
|
|
|
{undefined, Req2} ->
|
|
|
|
{Default, Req2#http_req{cookies=[]}};
|
|
|
|
{RawCookie, Req2} ->
|
|
|
|
Cookies = cowboy_cookies:parse_cookie(RawCookie),
|
|
|
|
cookie(Name, Req2#http_req{cookies=Cookies}, Default)
|
|
|
|
end;
|
|
|
|
cookie(Name, Req, Default) ->
|
|
|
|
case lists:keyfind(Name, 1, Req#http_req.cookies) of
|
|
|
|
{Name, Value} -> {Value, Req};
|
|
|
|
false -> {Default, Req}
|
|
|
|
end.
|
|
|
|
|
|
|
|
%% @doc Return the full list of cookie values.
|
|
|
|
-spec cookies(#http_req{}) -> {list({binary(), binary() | true}), #http_req{}}.
|
|
|
|
cookies(Req=#http_req{cookies=undefined}) ->
|
|
|
|
case header('Cookie', Req) of
|
|
|
|
{undefined, Req2} ->
|
|
|
|
{[], Req2#http_req{cookies=[]}};
|
|
|
|
{RawCookie, Req2} ->
|
|
|
|
Cookies = cowboy_cookies:parse_cookie(RawCookie),
|
|
|
|
cookies(Req2#http_req{cookies=Cookies})
|
|
|
|
end;
|
|
|
|
cookies(Req=#http_req{cookies=Cookies}) ->
|
|
|
|
{Cookies, Req}.
|
|
|
|
|
2011-03-21 17:26:00 +01:00
|
|
|
%% Request Body API.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the full body sent with the request, or <em>{error, badarg}</em>
|
|
|
|
%% if no <em>Content-Length</em> is available.
|
2011-03-21 17:26:00 +01:00
|
|
|
%% @todo We probably want to allow a max length.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec body(#http_req{}) -> {ok, binary(), #http_req{}} | {error, atom()}.
|
2011-03-21 21:28:24 +01:00
|
|
|
body(Req) ->
|
2011-11-04 11:42:36 +01:00
|
|
|
{Length, Req2} = cowboy_http_req:parse_header('Content-Length', Req),
|
2011-03-21 21:28:24 +01:00
|
|
|
case Length of
|
2011-03-29 13:55:53 +02:00
|
|
|
undefined -> {error, badarg};
|
2011-11-04 11:42:36 +01:00
|
|
|
{error, badarg} -> {error, badarg};
|
2011-03-21 21:28:24 +01:00
|
|
|
_Any ->
|
2011-11-04 11:42:36 +01:00
|
|
|
body(Length, Req2)
|
2011-03-21 21:28:24 +01:00
|
|
|
end.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return <em>Length</em> bytes of the request body.
|
|
|
|
%%
|
|
|
|
%% You probably shouldn't be calling this function directly, as it expects the
|
|
|
|
%% <em>Length</em> argument to be the full size of the body, and will consider
|
|
|
|
%% the body to be fully read from the socket.
|
2011-03-21 21:28:24 +01:00
|
|
|
%% @todo We probably want to configure the timeout.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec body(non_neg_integer(), #http_req{})
|
|
|
|
-> {ok, binary(), #http_req{}} | {error, atom()}.
|
2011-05-05 14:03:39 +02:00
|
|
|
body(Length, Req=#http_req{body_state=waiting, buffer=Buffer})
|
2011-12-06 12:53:56 +01:00
|
|
|
when is_integer(Length) andalso Length =< byte_size(Buffer) ->
|
2011-11-07 19:17:49 +01:00
|
|
|
<< Body:Length/binary, Rest/bits >> = Buffer,
|
|
|
|
{ok, Body, Req#http_req{body_state=done, buffer=Rest}};
|
2011-03-22 13:23:40 +01:00
|
|
|
body(Length, Req=#http_req{socket=Socket, transport=Transport,
|
2011-12-06 12:53:56 +01:00
|
|
|
body_state=waiting, buffer=Buffer}) ->
|
2011-05-05 14:03:39 +02:00
|
|
|
case Transport:recv(Socket, Length - byte_size(Buffer), 5000) of
|
2011-05-25 23:02:40 +02:00
|
|
|
{ok, Body} -> {ok, << Buffer/binary, Body/binary >>,
|
|
|
|
Req#http_req{body_state=done, buffer= <<>>}};
|
2011-03-21 17:26:00 +01:00
|
|
|
{error, Reason} -> {error, Reason}
|
|
|
|
end.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Return the full body sent with the reqest, parsed as an
|
|
|
|
%% application/x-www-form-urlencoded string. Essentially a POST query string.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec body_qs(#http_req{}) -> {list({binary(), binary() | true}), #http_req{}}.
|
2011-12-05 01:08:38 +01:00
|
|
|
body_qs(Req=#http_req{urldecode={URLDecFun, URLDecArg}}) ->
|
2011-03-22 13:20:21 +01:00
|
|
|
{ok, Body, Req2} = body(Req),
|
2011-12-05 01:08:38 +01:00
|
|
|
{parse_qs(Body, fun(Bin) -> URLDecFun(Bin, URLDecArg) end), Req2}.
|
2011-03-22 13:20:21 +01:00
|
|
|
|
2011-03-18 22:38:26 +01:00
|
|
|
%% Response API.
|
|
|
|
|
2011-12-07 11:54:57 +01:00
|
|
|
%% @doc Add a cookie header to the response.
|
|
|
|
-spec set_resp_cookie(binary(), binary(), [cowboy_cookies:cookie_option()],
|
|
|
|
#http_req{}) -> {ok, #http_req{}}.
|
|
|
|
set_resp_cookie(Name, Value, Options, Req) ->
|
|
|
|
{HeaderName, HeaderValue} = cowboy_cookies:cookie(Name, Value, Options),
|
|
|
|
set_resp_header(HeaderName, HeaderValue, Req).
|
|
|
|
|
2011-11-28 09:09:41 +01:00
|
|
|
%% @doc Add a header to the response.
|
2011-12-05 07:54:56 +01:00
|
|
|
-spec set_resp_header(http_header(), iodata(), #http_req{})
|
2011-11-28 09:09:41 +01:00
|
|
|
-> {ok, #http_req{}}.
|
|
|
|
set_resp_header(Name, Value, Req=#http_req{resp_headers=RespHeaders}) ->
|
|
|
|
NameBin = header_to_binary(Name),
|
|
|
|
{ok, Req#http_req{resp_headers=[{NameBin, Value}|RespHeaders]}}.
|
|
|
|
|
|
|
|
%% @doc Add a body to the response.
|
|
|
|
%%
|
|
|
|
%% The body set here is ignored if the response is later sent using
|
|
|
|
%% anything other than reply/2 or reply/3.
|
2011-12-05 07:54:56 +01:00
|
|
|
-spec set_resp_body(iodata(), #http_req{}) -> {ok, #http_req{}}.
|
2011-11-28 09:09:41 +01:00
|
|
|
set_resp_body(Body, Req) ->
|
|
|
|
{ok, Req#http_req{resp_body=Body}}.
|
|
|
|
|
|
|
|
%% @doc Return whether the given header has been set for the response.
|
|
|
|
-spec has_resp_header(http_header(), #http_req{}) -> boolean().
|
|
|
|
has_resp_header(Name, #http_req{resp_headers=RespHeaders}) ->
|
|
|
|
NameBin = header_to_binary(Name),
|
|
|
|
lists:keymember(NameBin, 1, RespHeaders).
|
|
|
|
|
|
|
|
%% @doc Return whether a body has been set for the response.
|
|
|
|
-spec has_resp_body(#http_req{}) -> boolean().
|
|
|
|
has_resp_body(#http_req{resp_body=RespBody}) ->
|
|
|
|
byte_size(RespBody) > 0.
|
|
|
|
|
2011-10-13 16:16:53 +02:00
|
|
|
%% @equiv reply(Status, [], [], Req)
|
|
|
|
-spec reply(http_status(), #http_req{}) -> {ok, #http_req{}}.
|
2011-11-28 09:09:41 +01:00
|
|
|
reply(Status, Req=#http_req{resp_body=Body}) ->
|
|
|
|
reply(Status, [], Body, Req).
|
2011-10-13 16:16:53 +02:00
|
|
|
|
|
|
|
%% @equiv reply(Status, Headers, [], Req)
|
|
|
|
-spec reply(http_status(), http_headers(), #http_req{}) -> {ok, #http_req{}}.
|
2011-11-28 09:09:41 +01:00
|
|
|
reply(Status, Headers, Req=#http_req{resp_body=Body}) ->
|
|
|
|
reply(Status, Headers, Body, Req).
|
2011-10-13 16:16:53 +02:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Send a reply to the client.
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec reply(http_status(), http_headers(), iodata(), #http_req{})
|
|
|
|
-> {ok, #http_req{}}.
|
2011-10-13 16:01:55 +02:00
|
|
|
reply(Status, Headers, Body, Req=#http_req{socket=Socket,
|
2011-03-20 18:03:11 +01:00
|
|
|
transport=Transport, connection=Connection,
|
2011-11-28 09:09:41 +01:00
|
|
|
method=Method, resp_state=waiting, resp_headers=RespHeaders}) ->
|
2011-10-06 12:40:04 +02:00
|
|
|
RespConn = response_connection(Headers, Connection),
|
2011-11-28 09:09:41 +01:00
|
|
|
Head = response_head(Status, Headers, RespHeaders, [
|
2011-05-05 14:03:39 +02:00
|
|
|
{<<"Connection">>, atom_to_connection(Connection)},
|
2011-05-06 00:27:51 +02:00
|
|
|
{<<"Content-Length">>,
|
2011-05-14 15:20:12 +02:00
|
|
|
list_to_binary(integer_to_list(iolist_size(Body)))},
|
2011-05-14 18:46:50 +02:00
|
|
|
{<<"Date">>, cowboy_clock:rfc1123()},
|
|
|
|
{<<"Server">>, <<"Cowboy">>}
|
2011-05-08 14:40:58 +02:00
|
|
|
]),
|
2011-09-06 12:11:44 +02:00
|
|
|
case Method of
|
|
|
|
'HEAD' -> Transport:send(Socket, Head);
|
|
|
|
_ -> Transport:send(Socket, [Head, Body])
|
|
|
|
end,
|
2011-11-28 09:09:41 +01:00
|
|
|
{ok, Req#http_req{connection=RespConn, resp_state=done,
|
|
|
|
resp_headers=[], resp_body= <<>>}}.
|
2011-03-18 22:38:26 +01:00
|
|
|
|
2011-10-13 16:16:53 +02:00
|
|
|
%% @equiv chunked_reply(Status, [], Req)
|
|
|
|
-spec chunked_reply(http_status(), #http_req{}) -> {ok, #http_req{}}.
|
|
|
|
chunked_reply(Status, Req) ->
|
|
|
|
chunked_reply(Status, [], Req).
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Initiate the sending of a chunked reply to the client.
|
|
|
|
%% @see cowboy_http_req:chunk/2
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec chunked_reply(http_status(), http_headers(), #http_req{})
|
|
|
|
-> {ok, #http_req{}}.
|
2011-10-13 16:01:55 +02:00
|
|
|
chunked_reply(Status, Headers, Req=#http_req{socket=Socket, transport=Transport,
|
2011-11-28 09:09:41 +01:00
|
|
|
connection=Connection, resp_state=waiting, resp_headers=RespHeaders}) ->
|
2011-10-06 12:40:04 +02:00
|
|
|
RespConn = response_connection(Headers, Connection),
|
2011-11-28 09:09:41 +01:00
|
|
|
Head = response_head(Status, Headers, RespHeaders, [
|
2011-10-06 12:40:04 +02:00
|
|
|
{<<"Connection">>, atom_to_connection(Connection)},
|
2011-05-14 15:20:12 +02:00
|
|
|
{<<"Transfer-Encoding">>, <<"chunked">>},
|
2011-05-14 18:46:50 +02:00
|
|
|
{<<"Date">>, cowboy_clock:rfc1123()},
|
|
|
|
{<<"Server">>, <<"Cowboy">>}
|
2011-05-08 17:26:21 +02:00
|
|
|
]),
|
|
|
|
Transport:send(Socket, Head),
|
2011-11-28 09:09:41 +01:00
|
|
|
{ok, Req#http_req{connection=RespConn, resp_state=chunks,
|
|
|
|
resp_headers=[], resp_body= <<>>}}.
|
2011-05-08 17:26:21 +02:00
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Send a chunk of data.
|
|
|
|
%%
|
|
|
|
%% A chunked reply must have been initiated before calling this function.
|
2011-10-29 15:48:43 +01:00
|
|
|
-spec chunk(iodata(), #http_req{}) -> ok | {error, atom()}.
|
2011-09-06 12:11:44 +02:00
|
|
|
chunk(_Data, #http_req{socket=_Socket, transport=_Transport, method='HEAD'}) ->
|
|
|
|
ok;
|
2011-05-08 17:26:21 +02:00
|
|
|
chunk(Data, #http_req{socket=Socket, transport=Transport, resp_state=chunks}) ->
|
|
|
|
Transport:send(Socket, [integer_to_list(iolist_size(Data), 16),
|
|
|
|
<<"\r\n">>, Data, <<"\r\n">>]).
|
|
|
|
|
2011-10-20 14:11:17 +02:00
|
|
|
%% @doc Send an upgrade reply.
|
|
|
|
-spec upgrade_reply(http_status(), http_headers(), #http_req{})
|
|
|
|
-> {ok, #http_req{}}.
|
|
|
|
upgrade_reply(Status, Headers, Req=#http_req{socket=Socket, transport=Transport,
|
2011-11-28 09:09:41 +01:00
|
|
|
resp_state=waiting, resp_headers=RespHeaders}) ->
|
|
|
|
Head = response_head(Status, Headers, RespHeaders, [
|
2011-10-20 14:11:17 +02:00
|
|
|
{<<"Connection">>, <<"Upgrade">>}
|
|
|
|
]),
|
|
|
|
Transport:send(Socket, Head),
|
2011-11-28 09:09:41 +01:00
|
|
|
{ok, Req#http_req{resp_state=done, resp_headers=[], resp_body= <<>>}}.
|
2011-10-20 14:11:17 +02:00
|
|
|
|
2011-06-01 12:49:03 +02:00
|
|
|
%% Misc API.
|
|
|
|
|
2011-07-06 17:42:20 +02:00
|
|
|
%% @doc Compact the request data by removing all non-system information.
|
|
|
|
%%
|
|
|
|
%% This essentially removes the host, path, query string, bindings and headers.
|
|
|
|
%% Use it when you really need to save up memory, for example when having
|
|
|
|
%% many concurrent long-running connections.
|
2011-06-01 12:49:03 +02:00
|
|
|
-spec compact(#http_req{}) -> #http_req{}.
|
|
|
|
compact(Req) ->
|
|
|
|
Req#http_req{host=undefined, host_info=undefined, path=undefined,
|
2011-10-06 01:07:49 +02:00
|
|
|
path_info=undefined, qs_vals=undefined,
|
2011-06-01 12:49:03 +02:00
|
|
|
bindings=undefined, headers=[]}.
|
|
|
|
|
2011-03-07 22:59:22 +01:00
|
|
|
%% Internal.
|
|
|
|
|
2011-12-05 01:08:38 +01:00
|
|
|
-spec parse_qs(binary(), fun((binary()) -> binary())) ->
|
|
|
|
list({binary(), binary() | true}).
|
|
|
|
parse_qs(<<>>, _URLDecode) ->
|
2011-05-05 14:03:39 +02:00
|
|
|
[];
|
2011-12-05 01:08:38 +01:00
|
|
|
parse_qs(Qs, URLDecode) ->
|
2011-05-05 14:03:39 +02:00
|
|
|
Tokens = binary:split(Qs, <<"&">>, [global, trim]),
|
|
|
|
[case binary:split(Token, <<"=">>) of
|
2011-12-05 01:08:38 +01:00
|
|
|
[Token] -> {URLDecode(Token), true};
|
|
|
|
[Name, Value] -> {URLDecode(Name), URLDecode(Value)}
|
2011-03-07 22:59:22 +01:00
|
|
|
end || Token <- Tokens].
|
|
|
|
|
2011-10-06 12:40:04 +02:00
|
|
|
-spec response_connection(http_headers(), keepalive | close)
|
|
|
|
-> keepalive | close.
|
|
|
|
response_connection([], Connection) ->
|
|
|
|
Connection;
|
|
|
|
response_connection([{Name, Value}|Tail], Connection) ->
|
|
|
|
case Name of
|
|
|
|
'Connection' -> response_connection_parse(Value);
|
2011-10-07 16:14:39 +02:00
|
|
|
Name when is_atom(Name) -> response_connection(Tail, Connection);
|
2011-10-06 12:40:04 +02:00
|
|
|
Name ->
|
|
|
|
Name2 = cowboy_bstr:to_lower(Name),
|
|
|
|
case Name2 of
|
|
|
|
<<"connection">> -> response_connection_parse(Value);
|
|
|
|
_Any -> response_connection(Tail, Connection)
|
|
|
|
end
|
|
|
|
end.
|
|
|
|
|
|
|
|
-spec response_connection_parse(binary()) -> keepalive | close.
|
|
|
|
response_connection_parse(ReplyConn) ->
|
2011-10-17 09:15:01 +02:00
|
|
|
Tokens = cowboy_http:nonempty_list(ReplyConn, fun cowboy_http:token/2),
|
2011-10-06 12:40:04 +02:00
|
|
|
cowboy_http:connection_to_atom(Tokens).
|
|
|
|
|
2011-11-28 09:09:41 +01:00
|
|
|
-spec response_head(http_status(), http_headers(), http_headers(),
|
|
|
|
http_headers()) -> iolist().
|
|
|
|
response_head(Status, Headers, RespHeaders, DefaultHeaders) ->
|
2011-10-13 16:01:55 +02:00
|
|
|
StatusLine = <<"HTTP/1.1 ", (status(Status))/binary, "\r\n">>,
|
2011-05-08 14:40:58 +02:00
|
|
|
Headers2 = [{header_to_binary(Key), Value} || {Key, Value} <- Headers],
|
2011-11-28 09:09:41 +01:00
|
|
|
Headers3 = merge_headers(
|
|
|
|
merge_headers(Headers2, RespHeaders),
|
|
|
|
DefaultHeaders),
|
|
|
|
Headers4 = [[Key, <<": ">>, Value, <<"\r\n">>]
|
|
|
|
|| {Key, Value} <- Headers3],
|
|
|
|
[StatusLine, Headers4, <<"\r\n">>].
|
|
|
|
|
|
|
|
-spec merge_headers(http_headers(), http_headers()) -> http_headers().
|
|
|
|
merge_headers(Headers, []) ->
|
|
|
|
Headers;
|
|
|
|
merge_headers(Headers, [{Name, Value}|Tail]) ->
|
|
|
|
Headers2 = case lists:keymember(Name, 1, Headers) of
|
|
|
|
true -> Headers;
|
|
|
|
false -> Headers ++ [{Name, Value}]
|
|
|
|
end,
|
|
|
|
merge_headers(Headers2, Tail).
|
2011-05-08 14:40:58 +02:00
|
|
|
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec atom_to_connection(keepalive) -> <<_:80>>;
|
|
|
|
(close) -> <<_:40>>.
|
2011-03-18 22:38:26 +01:00
|
|
|
atom_to_connection(keepalive) ->
|
2011-05-05 14:03:39 +02:00
|
|
|
<<"keep-alive">>;
|
2011-03-18 22:38:26 +01:00
|
|
|
atom_to_connection(close) ->
|
2011-05-05 14:03:39 +02:00
|
|
|
<<"close">>.
|
|
|
|
|
2011-05-25 23:02:40 +02:00
|
|
|
-spec status(http_status()) -> binary().
|
2011-05-05 14:03:39 +02:00
|
|
|
status(100) -> <<"100 Continue">>;
|
|
|
|
status(101) -> <<"101 Switching Protocols">>;
|
|
|
|
status(102) -> <<"102 Processing">>;
|
|
|
|
status(200) -> <<"200 OK">>;
|
|
|
|
status(201) -> <<"201 Created">>;
|
|
|
|
status(202) -> <<"202 Accepted">>;
|
|
|
|
status(203) -> <<"203 Non-Authoritative Information">>;
|
|
|
|
status(204) -> <<"204 No Content">>;
|
|
|
|
status(205) -> <<"205 Reset Content">>;
|
|
|
|
status(206) -> <<"206 Partial Content">>;
|
|
|
|
status(207) -> <<"207 Multi-Status">>;
|
|
|
|
status(226) -> <<"226 IM Used">>;
|
|
|
|
status(300) -> <<"300 Multiple Choices">>;
|
|
|
|
status(301) -> <<"301 Moved Permanently">>;
|
|
|
|
status(302) -> <<"302 Found">>;
|
|
|
|
status(303) -> <<"303 See Other">>;
|
|
|
|
status(304) -> <<"304 Not Modified">>;
|
|
|
|
status(305) -> <<"305 Use Proxy">>;
|
|
|
|
status(306) -> <<"306 Switch Proxy">>;
|
|
|
|
status(307) -> <<"307 Temporary Redirect">>;
|
|
|
|
status(400) -> <<"400 Bad Request">>;
|
|
|
|
status(401) -> <<"401 Unauthorized">>;
|
|
|
|
status(402) -> <<"402 Payment Required">>;
|
|
|
|
status(403) -> <<"403 Forbidden">>;
|
|
|
|
status(404) -> <<"404 Not Found">>;
|
|
|
|
status(405) -> <<"405 Method Not Allowed">>;
|
|
|
|
status(406) -> <<"406 Not Acceptable">>;
|
|
|
|
status(407) -> <<"407 Proxy Authentication Required">>;
|
|
|
|
status(408) -> <<"408 Request Timeout">>;
|
|
|
|
status(409) -> <<"409 Conflict">>;
|
|
|
|
status(410) -> <<"410 Gone">>;
|
|
|
|
status(411) -> <<"411 Length Required">>;
|
|
|
|
status(412) -> <<"412 Precondition Failed">>;
|
|
|
|
status(413) -> <<"413 Request Entity Too Large">>;
|
|
|
|
status(414) -> <<"414 Request-URI Too Long">>;
|
|
|
|
status(415) -> <<"415 Unsupported Media Type">>;
|
|
|
|
status(416) -> <<"416 Requested Range Not Satisfiable">>;
|
|
|
|
status(417) -> <<"417 Expectation Failed">>;
|
|
|
|
status(418) -> <<"418 I'm a teapot">>;
|
|
|
|
status(422) -> <<"422 Unprocessable Entity">>;
|
|
|
|
status(423) -> <<"423 Locked">>;
|
|
|
|
status(424) -> <<"424 Failed Dependency">>;
|
|
|
|
status(425) -> <<"425 Unordered Collection">>;
|
|
|
|
status(426) -> <<"426 Upgrade Required">>;
|
|
|
|
status(500) -> <<"500 Internal Server Error">>;
|
|
|
|
status(501) -> <<"501 Not Implemented">>;
|
|
|
|
status(502) -> <<"502 Bad Gateway">>;
|
|
|
|
status(503) -> <<"503 Service Unavailable">>;
|
|
|
|
status(504) -> <<"504 Gateway Timeout">>;
|
|
|
|
status(505) -> <<"505 HTTP Version Not Supported">>;
|
|
|
|
status(506) -> <<"506 Variant Also Negotiates">>;
|
|
|
|
status(507) -> <<"507 Insufficient Storage">>;
|
|
|
|
status(510) -> <<"510 Not Extended">>;
|
|
|
|
status(B) when is_binary(B) -> B.
|
2011-03-18 22:38:26 +01:00
|
|
|
|
2011-05-06 00:27:51 +02:00
|
|
|
-spec header_to_binary(http_header()) -> binary().
|
|
|
|
header_to_binary('Cache-Control') -> <<"Cache-Control">>;
|
|
|
|
header_to_binary('Connection') -> <<"Connection">>;
|
|
|
|
header_to_binary('Date') -> <<"Date">>;
|
|
|
|
header_to_binary('Pragma') -> <<"Pragma">>;
|
|
|
|
header_to_binary('Transfer-Encoding') -> <<"Transfer-Encoding">>;
|
|
|
|
header_to_binary('Upgrade') -> <<"Upgrade">>;
|
|
|
|
header_to_binary('Via') -> <<"Via">>;
|
|
|
|
header_to_binary('Accept') -> <<"Accept">>;
|
|
|
|
header_to_binary('Accept-Charset') -> <<"Accept-Charset">>;
|
|
|
|
header_to_binary('Accept-Encoding') -> <<"Accept-Encoding">>;
|
|
|
|
header_to_binary('Accept-Language') -> <<"Accept-Language">>;
|
|
|
|
header_to_binary('Authorization') -> <<"Authorization">>;
|
|
|
|
header_to_binary('From') -> <<"From">>;
|
|
|
|
header_to_binary('Host') -> <<"Host">>;
|
|
|
|
header_to_binary('If-Modified-Since') -> <<"If-Modified-Since">>;
|
|
|
|
header_to_binary('If-Match') -> <<"If-Match">>;
|
|
|
|
header_to_binary('If-None-Match') -> <<"If-None-Match">>;
|
|
|
|
header_to_binary('If-Range') -> <<"If-Range">>;
|
|
|
|
header_to_binary('If-Unmodified-Since') -> <<"If-Unmodified-Since">>;
|
|
|
|
header_to_binary('Max-Forwards') -> <<"Max-Forwards">>;
|
|
|
|
header_to_binary('Proxy-Authorization') -> <<"Proxy-Authorization">>;
|
|
|
|
header_to_binary('Range') -> <<"Range">>;
|
|
|
|
header_to_binary('Referer') -> <<"Referer">>;
|
|
|
|
header_to_binary('User-Agent') -> <<"User-Agent">>;
|
|
|
|
header_to_binary('Age') -> <<"Age">>;
|
|
|
|
header_to_binary('Location') -> <<"Location">>;
|
|
|
|
header_to_binary('Proxy-Authenticate') -> <<"Proxy-Authenticate">>;
|
|
|
|
header_to_binary('Public') -> <<"Public">>;
|
|
|
|
header_to_binary('Retry-After') -> <<"Retry-After">>;
|
|
|
|
header_to_binary('Server') -> <<"Server">>;
|
|
|
|
header_to_binary('Vary') -> <<"Vary">>;
|
|
|
|
header_to_binary('Warning') -> <<"Warning">>;
|
|
|
|
header_to_binary('Www-Authenticate') -> <<"Www-Authenticate">>;
|
|
|
|
header_to_binary('Allow') -> <<"Allow">>;
|
|
|
|
header_to_binary('Content-Base') -> <<"Content-Base">>;
|
|
|
|
header_to_binary('Content-Encoding') -> <<"Content-Encoding">>;
|
|
|
|
header_to_binary('Content-Language') -> <<"Content-Language">>;
|
|
|
|
header_to_binary('Content-Length') -> <<"Content-Length">>;
|
|
|
|
header_to_binary('Content-Location') -> <<"Content-Location">>;
|
|
|
|
header_to_binary('Content-Md5') -> <<"Content-Md5">>;
|
|
|
|
header_to_binary('Content-Range') -> <<"Content-Range">>;
|
|
|
|
header_to_binary('Content-Type') -> <<"Content-Type">>;
|
|
|
|
header_to_binary('Etag') -> <<"Etag">>;
|
|
|
|
header_to_binary('Expires') -> <<"Expires">>;
|
|
|
|
header_to_binary('Last-Modified') -> <<"Last-Modified">>;
|
|
|
|
header_to_binary('Accept-Ranges') -> <<"Accept-Ranges">>;
|
|
|
|
header_to_binary('Set-Cookie') -> <<"Set-Cookie">>;
|
|
|
|
header_to_binary('Set-Cookie2') -> <<"Set-Cookie2">>;
|
|
|
|
header_to_binary('X-Forwarded-For') -> <<"X-Forwarded-For">>;
|
|
|
|
header_to_binary('Cookie') -> <<"Cookie">>;
|
|
|
|
header_to_binary('Keep-Alive') -> <<"Keep-Alive">>;
|
|
|
|
header_to_binary('Proxy-Connection') -> <<"Proxy-Connection">>;
|
|
|
|
header_to_binary(B) when is_binary(B) -> B.
|
|
|
|
|
2011-03-07 22:59:22 +01:00
|
|
|
%% Tests.
|
|
|
|
|
|
|
|
-ifdef(TEST).
|
|
|
|
|
|
|
|
parse_qs_test_() ->
|
|
|
|
%% {Qs, Result}
|
|
|
|
Tests = [
|
2011-05-05 14:03:39 +02:00
|
|
|
{<<"">>, []},
|
|
|
|
{<<"a=b">>, [{<<"a">>, <<"b">>}]},
|
|
|
|
{<<"aaa=bbb">>, [{<<"aaa">>, <<"bbb">>}]},
|
|
|
|
{<<"a&b">>, [{<<"a">>, true}, {<<"b">>, true}]},
|
2011-05-25 23:02:40 +02:00
|
|
|
{<<"a=b&c&d=e">>, [{<<"a">>, <<"b">>},
|
|
|
|
{<<"c">>, true}, {<<"d">>, <<"e">>}]},
|
2011-07-20 17:38:10 +02:00
|
|
|
{<<"a=b=c=d=e&f=g">>, [{<<"a">>, <<"b=c=d=e">>}, {<<"f">>, <<"g">>}]},
|
|
|
|
{<<"a+b=c+d">>, [{<<"a b">>, <<"c d">>}]}
|
2011-03-07 22:59:22 +01:00
|
|
|
],
|
2011-12-05 01:08:38 +01:00
|
|
|
URLDecode = fun cowboy_http:urldecode/1,
|
|
|
|
[{Qs, fun() -> R = parse_qs(Qs, URLDecode) end} || {Qs, R} <- Tests].
|
2011-03-07 22:59:22 +01:00
|
|
|
|
|
|
|
-endif.
|