0
Fork 0
mirror of https://github.com/ninenines/cowboy.git synced 2025-07-16 05:00:24 +00:00

Tests and fixes for the generate_etag/2 callback

The return value from the generate_etag/2 callback is expected to be a
binary tagged with either weak or strong. This binary is quoted, and
may be prefixed with W/ before it is set as the value of the ETag header
in the response.

For backwards compatibility with older handlers where the return value
was expected to be a quoted binary a function has been added to parse any
return values that are untagged binaries. All untagged binaries are expected
to be a valid value for the ETag header.
This commit is contained in:
Magnus Klaar 2012-02-28 18:35:26 +01:00
parent 8f0a78f52f
commit 9922de6d9e
4 changed files with 130 additions and 36 deletions

View file

@ -29,7 +29,8 @@
file_200/1, file_403/1, dir_403/1, file_404/1,
file_400/1]). %% http and https.
-export([http_10_hostless/1]). %% misc.
-export([rest_simple/1, rest_keepalive/1, rest_keepalive_post/1, rest_nodelete/1]). %% rest.
-export([rest_simple/1, rest_keepalive/1, rest_keepalive_post/1,
rest_nodelete/1, rest_resource_etags/1]). %% rest.
%% ct.
@ -47,7 +48,8 @@ groups() ->
static_function_etag, multipart] ++ BaseTests},
{https, [], BaseTests},
{misc, [], [http_10_hostless]},
{rest, [], [rest_simple, rest_keepalive, rest_keepalive_post, rest_nodelete]}].
{rest, [], [rest_simple, rest_keepalive, rest_keepalive_post,
rest_nodelete, rest_resource_etags]}].
init_per_suite(Config) ->
application:start(inets),
@ -75,7 +77,7 @@ init_per_group(https, Config) ->
application:start(public_key),
application:start(ssl),
DataDir = ?config(data_dir, Config),
cowboy:start_listener(https, 100,
{ok,_} = cowboy:start_listener(https, 100,
cowboy_ssl_transport, [
{port, Port}, {certfile, DataDir ++ "cert.pem"},
{keyfile, DataDir ++ "key.pem"}, {password, "cowboy"}],
@ -84,7 +86,7 @@ init_per_group(https, Config) ->
[{scheme, "https"}, {port, Port}|Config1];
init_per_group(misc, Config) ->
Port = 33082,
cowboy:start_listener(misc, 100,
{ok,_} = cowboy:start_listener(misc, 100,
cowboy_tcp_transport, [{port, Port}],
cowboy_http_protocol, [{dispatch, [{'_', [
{[], http_handler, []}
@ -92,15 +94,16 @@ init_per_group(misc, Config) ->
[{port, Port}|Config];
init_per_group(rest, Config) ->
Port = 33083,
cowboy:start_listener(reset, 100,
{ok,_} = cowboy:start_listener(rest, 100,
cowboy_tcp_transport, [{port, Port}],
cowboy_http_protocol, [{dispatch, [{'_', [
{[<<"simple">>], rest_simple_resource, []},
{[<<"forbidden_post">>], rest_forbidden_resource, [true]},
{[<<"simple_post">>], rest_forbidden_resource, [false]},
{[<<"nodelete">>], rest_nodelete_resource, []}
{[<<"nodelete">>], rest_nodelete_resource, []},
{[<<"resetags">>], rest_resource_etags, []}
]}]}]),
[{port, Port}|Config].
[{scheme, "http"},{port, Port}|Config].
end_per_group(https, Config) ->
cowboy:stop_listener(https),
@ -512,7 +515,7 @@ static_function_etag(Arguments, etag_data) ->
{_, _Modified} = lists:keyfind(mtime, 1, Arguments),
ChecksumCommand = lists:flatten(io_lib:format("sha1sum ~s", [Filepath])),
[Checksum|_] = string:tokens(os:cmd(ChecksumCommand), " "),
iolist_to_binary(Checksum).
{strong, iolist_to_binary(Checksum)}.
%% http and https.
@ -623,3 +626,55 @@ rest_nodelete(Config) ->
{ok, Data} = gen_tcp:recv(Socket, 0, 6000),
{0, 12} = binary:match(Data, <<"HTTP/1.1 500">>),
ok = gen_tcp:close(Socket).
rest_resource_etags(Config) ->
%% The Etag header should be set to the return value of generate_etag/2.
fun() ->
%% Correct return values from generate_etag/2.
{Packet1, 200} = raw_resp([
"GET /resetags?type=tuple-weak HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n", "\r\n"], Config),
{_,_} = binary:match(Packet1, <<"ETag: W/\"etag-header-value\"\r\n">>),
{Packet2, 200} = raw_resp([
"GET /resetags?type=tuple-strong HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n", "\r\n"], Config),
{_,_} = binary:match(Packet2, <<"ETag: \"etag-header-value\"\r\n">>),
%% Backwards compatible return values from generate_etag/2.
{Packet3, 200} = raw_resp([
"GET /resetags?type=binary-weak-quoted HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n", "\r\n"], Config),
{_,_} = binary:match(Packet3, <<"ETag: W/\"etag-header-value\"\r\n">>),
{Packet4, 200} = raw_resp([
"GET /resetags?type=binary-strong-quoted HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n", "\r\n"], Config),
{_,_} = binary:match(Packet4, <<"ETag: \"etag-header-value\"\r\n">>),
%% Invalid return values from generate_etag/2.
{_Packet5, 500} = raw_resp([
"GET /resetags?type=binary-strong-unquoted HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n", "\r\n"], Config),
{_Packet6, 500} = raw_resp([
"GET /resetags?type=binary-weak-unquoted HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n", "\r\n"], Config)
end(),
%% The return value of generate_etag/2 should match the request header.
fun() ->
%% Correct return values from generate_etag/2.
{_Packet1, 304} = raw_resp([
"GET /resetags?type=tuple-weak HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n",
"If-None-Match: W/\"etag-header-value\"\r\n", "\r\n"], Config),
{_Packet2, 304} = raw_resp([
"GET /resetags?type=tuple-strong HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n",
"If-None-Match: \"etag-header-value\"\r\n", "\r\n"], Config),
%% Backwards compatible return values from generate_etag/2.
{_Packet3, 304} = raw_resp([
"GET /resetags?type=binary-weak-quoted HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n",
"If-None-Match: W/\"etag-header-value\"\r\n", "\r\n"], Config),
{_Packet4, 304} = raw_resp([
"GET /resetags?type=binary-strong-quoted HTTP/1.1\r\n",
"Host: localhost\r\n", "Connection: close\r\n",
"If-None-Match: \"etag-header-value\"\r\n", "\r\n"], Config)
end().

View file

@ -0,0 +1,30 @@
-module(rest_resource_etags).
-export([init/3, generate_etag/2, content_types_provided/2, get_text_plain/2]).
init(_Transport, _Req, _Opts) ->
{upgrade, protocol, cowboy_http_rest}.
generate_etag(Req, State) ->
case cowboy_http_req:qs_val(<<"type">>, Req) of
%% Correct return values from generate_etag/2.
{<<"tuple-weak">>, Req2} ->
{{weak, <<"etag-header-value">>}, Req2, State};
{<<"tuple-strong">>, Req2} ->
{{strong, <<"etag-header-value">>}, Req2, State};
%% Backwards compatible return values from generate_etag/2.
{<<"binary-weak-quoted">>, Req2} ->
{<<"W/\"etag-header-value\"">>, Req2, State};
{<<"binary-strong-quoted">>, Req2} ->
{<<"\"etag-header-value\"">>, Req2, State};
%% Invalid return values from generate_etag/2.
{<<"binary-strong-unquoted">>, Req2} ->
{<<"etag-header-value">>, Req2, State};
{<<"binary-weak-unquoted">>, Req2} ->
{<<"W/etag-header-value">>, Req2, State}
end.
content_types_provided(Req, State) ->
{[{{<<"text">>, <<"plain">>, []}, get_text_plain}], Req, State}.
get_text_plain(Req, State) ->
{<<"This is REST!">>, Req, State}.