2011-03-07 22:59:22 +01:00
|
|
|
%% Copyright (c) 2011, Loïc Hoguin <essen@dev-extend.eu>
|
|
|
|
%%
|
|
|
|
%% 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.
|
|
|
|
|
|
|
|
-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-03-07 22:59:22 +01:00
|
|
|
host/1, raw_host/1,
|
|
|
|
path/1, raw_path/1,
|
|
|
|
qs_val/2, qs_val/3, qs_vals/1, raw_qs/1,
|
|
|
|
binding/2, binding/3, bindings/1,
|
|
|
|
header/2, header/3, headers/1
|
|
|
|
%% cookie/2, cookie/3, cookies/1 @todo
|
2011-03-18 22:38:26 +01:00
|
|
|
]). %% Request API.
|
|
|
|
|
|
|
|
-export([
|
|
|
|
reply/4
|
|
|
|
]). %% Response API.
|
2011-03-07 22:59:22 +01:00
|
|
|
|
|
|
|
-include("include/types.hrl").
|
|
|
|
-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
|
|
|
|
|
|
|
-spec method(Req::#http_req{}) -> {Method::http_method(), Req::#http_req{}}.
|
|
|
|
method(Req) ->
|
|
|
|
{Req#http_req.method, Req}.
|
|
|
|
|
|
|
|
-spec version(Req::#http_req{}) -> {Version::http_version(), Req::#http_req{}}.
|
|
|
|
version(Req) ->
|
|
|
|
{Req#http_req.version, Req}.
|
|
|
|
|
|
|
|
-spec peer(Req::#http_req{})
|
|
|
|
-> {{Address::ip_address(), Port::port_number()}, Req::#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}.
|
|
|
|
|
|
|
|
-spec host(Req::#http_req{}) -> {Host::path_tokens(), Req::#http_req{}}.
|
|
|
|
host(Req) ->
|
|
|
|
{Req#http_req.host, Req}.
|
|
|
|
|
|
|
|
-spec raw_host(Req::#http_req{}) -> {RawHost::string(), Req::#http_req{}}.
|
|
|
|
raw_host(Req) ->
|
|
|
|
{Req#http_req.raw_host, Req}.
|
|
|
|
|
|
|
|
-spec path(Req::#http_req{}) -> {Path::path_tokens(), Req::#http_req{}}.
|
|
|
|
path(Req) ->
|
|
|
|
{Req#http_req.path, Req}.
|
|
|
|
|
|
|
|
-spec raw_path(Req::#http_req{}) -> {RawPath::string(), Req::#http_req{}}.
|
|
|
|
raw_path(Req) ->
|
|
|
|
{Req#http_req.raw_path, Req}.
|
|
|
|
|
|
|
|
-spec qs_val(Name::atom(), Req::#http_req{})
|
|
|
|
-> {Value::string(), Req::#http_req{}}.
|
|
|
|
qs_val(Name, Req=#http_req{raw_qs=RawQs, qs_vals=undefined}) ->
|
|
|
|
QsVals = parse_qs(RawQs),
|
|
|
|
qs_val(Name, Req#http_req{qs_vals=QsVals});
|
|
|
|
qs_val(Name, Req) ->
|
|
|
|
{Name, Value} = lists:keyfind(Name, 1, Req#http_req.qs_vals),
|
|
|
|
{Value, Req}.
|
|
|
|
|
|
|
|
-spec qs_val(Name::atom(), Default::term(), Req::#http_req{})
|
|
|
|
-> {Value::string() | term(), Req::#http_req{}}.
|
|
|
|
qs_val(Name, Default, Req=#http_req{raw_qs=RawQs, qs_vals=undefined}) ->
|
|
|
|
QsVals = parse_qs(RawQs),
|
|
|
|
qs_val(Name, Default, Req#http_req{qs_vals=QsVals});
|
|
|
|
qs_val(Name, Default, Req) ->
|
|
|
|
Value = proplists:get_value(Name, Req#http_req.qs_vals, Default),
|
|
|
|
{Value, Req}.
|
|
|
|
|
|
|
|
-spec qs_vals(Req::#http_req{}) -> list({Name::atom(), Value::string()}).
|
|
|
|
qs_vals(Req=#http_req{raw_qs=RawQs, qs_vals=undefined}) ->
|
|
|
|
QsVals = parse_qs(RawQs),
|
|
|
|
qs_vals(Req#http_req{qs_vals=QsVals});
|
|
|
|
qs_vals(Req=#http_req{qs_vals=QsVals}) ->
|
|
|
|
{QsVals, Req}.
|
|
|
|
|
|
|
|
-spec raw_qs(Req::#http_req{}) -> {RawQs::string(), Req::#http_req{}}.
|
|
|
|
raw_qs(Req) ->
|
|
|
|
{Req#http_req.raw_qs, Req}.
|
|
|
|
|
|
|
|
-spec binding(Name::atom(), Req::#http_req{})
|
|
|
|
-> {Value::string(), Req::#http_req{}}.
|
|
|
|
binding(Name, Req) ->
|
|
|
|
{Name, Value} = lists:keyfind(Name, 1, Req#http_req.bindings),
|
|
|
|
{Value, Req}.
|
|
|
|
|
|
|
|
-spec binding(Name::atom(), Default::term(), Req::#http_req{})
|
|
|
|
-> {Value::string() | term(), Req::#http_req{}}.
|
|
|
|
binding(Name, Default, Req) ->
|
|
|
|
Value = proplists:get_value(Name, Req#http_req.bindings, Default),
|
|
|
|
{Value, Req}.
|
|
|
|
|
|
|
|
-spec bindings(Req::#http_req{})
|
|
|
|
-> {list({Name::atom(), Value::string()}), Req::#http_req{}}.
|
|
|
|
bindings(Req) ->
|
|
|
|
{Req#http_req.bindings, Req}.
|
|
|
|
|
|
|
|
-spec header(Name::atom() | string(), Req::#http_req{})
|
|
|
|
-> {Value::string(), Req::#http_req{}}.
|
|
|
|
header(Name, Req) ->
|
|
|
|
{Name, Value} = lists:keyfind(Name, 1, Req#http_req.headers),
|
|
|
|
{Value, Req}.
|
|
|
|
|
|
|
|
-spec header(Name::atom() | string(), Default::term(), Req::#http_req{})
|
|
|
|
-> {Value::string() | term(), Req::#http_req{}}.
|
|
|
|
header(Name, Default, Req) ->
|
|
|
|
Value = proplists:get_value(Name, Req#http_req.headers, Default),
|
|
|
|
{Value, Req}.
|
|
|
|
|
|
|
|
-spec headers(Req::#http_req{})
|
|
|
|
-> {list({Name::atom() | string(), Value::string()}), Req::#http_req{}}.
|
|
|
|
headers(Req) ->
|
|
|
|
{Req#http_req.headers, Req}.
|
|
|
|
|
2011-03-18 22:38:26 +01:00
|
|
|
%% Response API.
|
|
|
|
|
|
|
|
-spec reply(Code::http_status(), Headers::http_headers(),
|
|
|
|
Body::iolist(), Req::#http_req{}) -> ok.
|
|
|
|
%% @todo Don't be naive about the headers!
|
|
|
|
reply(Code, Headers, Body, Req=#http_req{socket=Socket,
|
|
|
|
transport=Transport, connection=Connection}) ->
|
|
|
|
StatusLine = ["HTTP/1.1 ", status(Code), "\r\n"],
|
|
|
|
BaseHeaders = ["Connection: ", atom_to_connection(Connection),
|
|
|
|
"\r\nContent-Length: ", integer_to_list(iolist_size(Body)), "\r\n"],
|
|
|
|
Transport:send(Socket,
|
|
|
|
[StatusLine, BaseHeaders, Headers, "\r\n", Body]),
|
|
|
|
{ok, Req}.
|
|
|
|
|
2011-03-07 22:59:22 +01:00
|
|
|
%% Internal.
|
|
|
|
|
|
|
|
-spec parse_qs(Qs::string()) -> list({Name::string(), Value::string()}).
|
|
|
|
parse_qs(Qs) ->
|
|
|
|
Tokens = string:tokens(Qs, "&"),
|
|
|
|
[case string:chr(Token, $=) of
|
|
|
|
0 ->
|
|
|
|
{Token, true};
|
|
|
|
N ->
|
|
|
|
{Name, [$=|Value]} = lists:split(N - 1, Token),
|
|
|
|
{Name, Value}
|
|
|
|
end || Token <- Tokens].
|
|
|
|
|
2011-03-18 22:38:26 +01:00
|
|
|
-spec atom_to_connection(Atom::keepalive | close) -> string().
|
|
|
|
atom_to_connection(keepalive) ->
|
|
|
|
"keep-alive";
|
|
|
|
atom_to_connection(close) ->
|
|
|
|
"close".
|
|
|
|
|
|
|
|
-spec status(Code::http_status()) -> string().
|
|
|
|
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(L) when is_list(L) -> L.
|
|
|
|
|
2011-03-07 22:59:22 +01:00
|
|
|
%% Tests.
|
|
|
|
|
|
|
|
-ifdef(TEST).
|
|
|
|
|
|
|
|
parse_qs_test_() ->
|
|
|
|
%% {Qs, Result}
|
|
|
|
Tests = [
|
|
|
|
{"", []},
|
|
|
|
{"a=b", [{"a", "b"}]},
|
|
|
|
{"aaa=bbb", [{"aaa", "bbb"}]},
|
|
|
|
{"a&b", [{"a", true}, {"b", true}]},
|
|
|
|
{"a=b&c&d=e", [{"a", "b"}, {"c", true}, {"d", "e"}]},
|
|
|
|
{"a=b=c=d=e&f=g", [{"a", "b=c=d=e"}, {"f", "g"}]}
|
|
|
|
],
|
|
|
|
[{Qs, fun() -> R = parse_qs(Qs) end} || {Qs, R} <- Tests].
|
|
|
|
|
|
|
|
-endif.
|