123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474 |
- %% Copyright (c) 2016-2017, Loïc Hoguin <essen@ninenines.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(examples_SUITE).
- -compile(export_all).
- -compile(nowarn_export_all).
- -import(ct_helper, [config/2]).
- -import(ct_helper, [doc/1]).
- -import(cowboy_test, [gun_open/1]).
- %% ct.
- all() ->
- ct_helper:all(?MODULE).
- init_per_suite(Config) ->
- %% Remove environment variables inherited from Erlang.mk.
- os:unsetenv("ERLANG_MK_TMP"),
- os:unsetenv("APPS_DIR"),
- os:unsetenv("DEPS_DIR"),
- os:unsetenv("ERL_LIBS"),
- %% Clone and build Cowboy, Cowlib and Ranch only once and
- %% reuse the same build across all tests.
- Make = do_find_make_cmd(),
- CommonDir = config(priv_dir, Config),
- ct:log("~s~n", [os:cmd("git clone --depth 1 https://github.com/ninenines/cowboy "
- ++ CommonDir ++ "cowboy")]),
- ct:log("~s~n", [os:cmd(Make ++ " -C " ++ CommonDir ++ "cowboy distclean")]),
- ct:log("~s~n", [os:cmd(Make ++ " -C " ++ CommonDir ++ "cowboy DEPS_DIR=" ++ CommonDir)]),
- Config.
- end_per_suite(_) ->
- ok.
- %% Find GNU Make.
- do_find_make_cmd() ->
- case os:getenv("MAKE") of
- false ->
- case os:find_executable("gmake") of
- false -> "make";
- Cmd -> Cmd
- end;
- Cmd ->
- Cmd
- end.
- %% Compile, start and stop releases.
- do_get_paths(Example0) ->
- Example = atom_to_list(Example0),
- {ok, CWD} = file:get_cwd(),
- Dir = CWD ++ "/../../examples/" ++ Example,
- Rel = Dir ++ "/_rel/" ++ Example ++ "_example/bin/" ++ Example ++ "_example",
- Log = Dir ++ "/_rel/" ++ Example ++ "_example/log/erlang.log.1",
- {Dir, Rel, Log}.
- do_compile_and_start(Example, Config) ->
- Make = do_find_make_cmd(),
- {Dir, Rel, _} = do_get_paths(Example),
- ct:log("~s~n", [os:cmd(Make ++ " -C " ++ Dir ++ " distclean")]),
- %% We use a common build for Cowboy, Cowlib and Ranch to speed things up.
- CommonDir = config(priv_dir, Config),
- ct:log("~s~n", [os:cmd("mkdir " ++ Dir ++ "/deps")]),
- ct:log("~s~n", [os:cmd("ln -s " ++ CommonDir ++ "cowboy " ++ Dir ++ "/deps/cowboy")]),
- ct:log("~s~n", [os:cmd("ln -s " ++ CommonDir ++ "cowlib " ++ Dir ++ "/deps/cowlib")]),
- ct:log("~s~n", [os:cmd("ln -s " ++ CommonDir ++ "ranch " ++ Dir ++ "/deps/ranch")]),
- %% TERM=dumb disables relx coloring.
- ct:log("~s~n", [os:cmd(Make ++ " -C " ++ Dir ++ " TERM=dumb")]),
- ct:log("~s~n", [os:cmd(Rel ++ " stop")]),
- ct:log("~s~n", [os:cmd(Rel ++ " daemon")]),
- timer:sleep(2000),
- ok.
- do_stop(Example) ->
- {_, Rel, Log} = do_get_paths(Example),
- ct:log("~s~n", [os:cmd(Rel ++ " stop")]),
- ct:log("~s~n", [element(2, file:read_file(Log))]),
- ok.
- %% Fetch a response.
- do_get(Transport, Protocol, Path, Config) ->
- do_get(Transport, Protocol, Path, [], Config).
- do_get(Transport, Protocol, Path, ReqHeaders, Config) ->
- Port = case Transport of
- tcp -> 8080;
- ssl -> 8443
- end,
- ConnPid = gun_open([{port, Port}, {type, Transport}, {protocol, Protocol}|Config]),
- Ref = gun:get(ConnPid, Path, ReqHeaders),
- case gun:await(ConnPid, Ref) of
- {response, nofin, Status, RespHeaders} ->
- {ok, Body} = gun:await_body(ConnPid, Ref),
- {Status, RespHeaders, Body};
- {response, fin, Status, RespHeaders} ->
- {Status, RespHeaders, <<>>}
- end.
- %% TCP and SSL Hello World.
- hello_world(Config) ->
- doc("Hello World example."),
- try
- do_compile_and_start(hello_world, Config),
- do_hello_world(tcp, http, Config),
- do_hello_world(tcp, http2, Config)
- after
- do_stop(hello_world)
- end.
- ssl_hello_world(Config) ->
- doc("SSL Hello World example."),
- try
- do_compile_and_start(ssl_hello_world, Config),
- do_hello_world(ssl, http, Config),
- do_hello_world(ssl, http2, Config)
- after
- do_stop(ssl_hello_world)
- end.
- do_hello_world(Transport, Protocol, Config) ->
- {200, _, <<"Hello world!">>} = do_get(Transport, Protocol, "/", Config),
- ok.
- %% Chunked Hello World.
- chunked_hello_world(Config) ->
- doc("Chunked Hello World example."),
- try
- do_compile_and_start(chunked_hello_world, Config),
- do_chunked_hello_world(tcp, http, Config),
- do_chunked_hello_world(tcp, http2, Config)
- after
- do_stop(chunked_hello_world)
- end.
- do_chunked_hello_world(Transport, Protocol, Config) ->
- ConnPid = gun_open([{port, 8080}, {type, Transport}, {protocol, Protocol}|Config]),
- Ref = gun:get(ConnPid, "/"),
- {response, nofin, 200, _} = gun:await(ConnPid, Ref),
- %% We expect to receive a chunk every second, three total.
- {data, nofin, <<"Hello\r\n">>} = gun:await(ConnPid, Ref, 2000),
- {data, nofin, <<"World\r\n">>} = gun:await(ConnPid, Ref, 2000),
- {data, IsFin, <<"Chunked!\r\n">>} = gun:await(ConnPid, Ref, 2000),
- %% We may get an extra empty chunk (last chunk for HTTP/1.1,
- %% empty DATA frame with the FIN bit set for HTTP/2).
- case IsFin of
- fin -> ok;
- nofin ->
- {data, fin, <<>>} = gun:await(ConnPid, Ref, 500),
- ok
- end.
- %% Compressed responses.
- compress_response(Config) ->
- doc("Compressed response example."),
- try
- do_compile_and_start(compress_response, Config),
- do_compress_response(tcp, http, Config),
- do_compress_response(tcp, http2, Config)
- after
- do_stop(compress_response)
- end.
- do_compress_response(Transport, Protocol, Config) ->
- {200, Headers, Body} = do_get(Transport, Protocol, "/",
- [{<<"accept-encoding">>, <<"gzip">>}], Config),
- {_, <<"gzip">>} = lists:keyfind(<<"content-encoding">>, 1, Headers),
- _ = zlib:gunzip(Body),
- ok.
- %% Cookie.
- cookie(Config) ->
- doc("Cookie example."),
- try
- do_compile_and_start(cookie, Config),
- do_cookie(tcp, http, Config),
- do_cookie(tcp, http2, Config)
- after
- do_stop(cookie)
- end.
- do_cookie(Transport, Protocol, Config) ->
- {200, _, One} = do_get(Transport, Protocol, "/", Config),
- {200, _, Two} = do_get(Transport, Protocol, "/", [{<<"cookie">>, <<"server=abcdef">>}], Config),
- true = One =/= Two,
- ok.
- %% Echo GET.
- echo_get(Config) ->
- doc("GET parameter echo example."),
- try
- do_compile_and_start(echo_get, Config),
- do_echo_get(tcp, http, Config),
- do_echo_get(tcp, http2, Config)
- after
- do_stop(echo_get)
- end.
- do_echo_get(Transport, Protocol, Config) ->
- {200, _, <<"this is fun">>} = do_get(Transport, Protocol, "/?echo=this+is+fun", Config),
- {400, _, _} = do_get(Transport, Protocol, "/", Config),
- ok.
- %% Echo POST.
- echo_post(Config) ->
- doc("POST parameter echo example."),
- try
- do_compile_and_start(echo_post, Config),
- do_echo_post(tcp, http, Config),
- do_echo_post(tcp, http2, Config)
- after
- do_stop(echo_post)
- end.
- do_echo_post(Transport, Protocol, Config) ->
- ConnPid = gun_open([{port, 8080}, {type, Transport}, {protocol, Protocol}|Config]),
- Ref = gun:post(ConnPid, "/", [
- {<<"content-type">>, <<"application/octet-stream">>}
- ], <<"echo=this+is+fun">>),
- {response, nofin, 200, _} = gun:await(ConnPid, Ref),
- {ok, <<"this is fun">>} = gun:await_body(ConnPid, Ref),
- ok.
- %% Eventsource.
- eventsource(Config) ->
- doc("Eventsource example."),
- try
- do_compile_and_start(eventsource, Config),
- do_eventsource(tcp, http, Config),
- do_eventsource(tcp, http2, Config)
- after
- do_stop(eventsource)
- end.
- do_eventsource(Transport, Protocol, Config) ->
- ConnPid = gun_open([{port, 8080}, {type, Transport}, {protocol, Protocol}|Config]),
- Ref = gun:get(ConnPid, "/eventsource"),
- {response, nofin, 200, Headers} = gun:await(ConnPid, Ref),
- {_, <<"text/event-stream">>} = lists:keyfind(<<"content-type">>, 1, Headers),
- %% Receive a few events.
- {data, nofin, << "id: ", _/bits >>} = gun:await(ConnPid, Ref, 2000),
- {data, nofin, << "id: ", _/bits >>} = gun:await(ConnPid, Ref, 2000),
- {data, nofin, << "id: ", _/bits >>} = gun:await(ConnPid, Ref, 2000),
- gun:close(ConnPid).
- %% REST Hello World.
- rest_hello_world(Config) ->
- doc("REST Hello World example."),
- try
- do_compile_and_start(rest_hello_world, Config),
- do_rest_hello_world(tcp, http, Config),
- do_rest_hello_world(tcp, http2, Config)
- after
- do_stop(rest_hello_world)
- end.
- do_rest_hello_world(Transport, Protocol, Config) ->
- << "<html>", _/bits >> = do_rest_get(Transport, Protocol,
- "/", undefined, undefined, Config),
- << "REST Hello World as text!" >> = do_rest_get(Transport, Protocol,
- "/", <<"text/plain">>, undefined, Config),
- << "{\"rest\": \"Hello World!\"}" >> = do_rest_get(Transport, Protocol,
- "/", <<"application/json">>, undefined, Config),
- not_acceptable = do_rest_get(Transport, Protocol,
- "/", <<"text/css">>, undefined, Config),
- ok.
- do_rest_get(Transport, Protocol, Path, Accept, Auth, Config) ->
- ReqHeaders0 = case Accept of
- undefined -> [];
- _ -> [{<<"accept">>, Accept}]
- end,
- ReqHeaders = case Auth of
- undefined -> ReqHeaders0;
- _ -> [{<<"authorization">>, [<<"Basic ">>, base64:encode(Auth)]}|ReqHeaders0]
- end,
- case do_get(Transport, Protocol, Path, ReqHeaders, Config) of
- {200, RespHeaders, Body} ->
- Accept = case Accept of
- undefined -> undefined;
- _ ->
- {_, ContentType} = lists:keyfind(<<"content-type">>, 1, RespHeaders),
- ContentType
- end,
- Body;
- {401, _, _} ->
- unauthorized;
- {406, _, _} ->
- not_acceptable
- end.
- %% REST basic auth.
- rest_basic_auth(Config) ->
- doc("REST basic authorization example."),
- try
- do_compile_and_start(rest_basic_auth, Config),
- do_rest_basic_auth(tcp, http, Config),
- do_rest_basic_auth(tcp, http2, Config)
- after
- do_stop(rest_basic_auth)
- end.
- do_rest_basic_auth(Transport, Protocol, Config) ->
- unauthorized = do_rest_get(Transport, Protocol, "/", undefined, undefined, Config),
- <<"Hello, Alladin!\n">> = do_rest_get(Transport, Protocol, "/", undefined, "Alladin:open sesame", Config),
- ok.
- %% REST pastebin.
- rest_pastebin(Config) ->
- doc("REST pastebin example."),
- try
- do_compile_and_start(rest_pastebin, Config),
- do_rest_pastebin(tcp, http, Config),
- do_rest_pastebin(tcp, http2, Config)
- after
- do_stop(rest_pastebin)
- end.
- do_rest_pastebin(Transport, Protocol, Config) ->
- %% Existing files.
- _ = do_rest_get(Transport, Protocol, "/", <<"text/html">>, undefined, Config),
- _ = do_rest_get(Transport, Protocol, "/", <<"text/plain">>, undefined, Config),
- %% Use POST to upload a new file and download it back.
- ConnPid = gun_open([{port, 8080}, {type, Transport}, {protocol, Protocol}|Config]),
- Ref = gun:post(ConnPid, "/", [
- {<<"content-type">>, <<"application/x-www-form-urlencoded">>}
- ], <<"paste=this+is+fun">>),
- %% @todo Not too happy about 303 here,
- %% will need to revisit this example.
- {response, _, 303, Headers} = gun:await(ConnPid, Ref),
- {_, Location} = lists:keyfind(<<"location">>, 1, Headers),
- <<"this is fun">> = do_rest_get(Transport, Protocol, Location, <<"text/plain">>, undefined, Config),
- << "<!DOCTYPE html><html>", _/bits >>
- = do_rest_get(Transport, Protocol, Location, <<"text/html">>, undefined, Config),
- ok.
- %% File server.
- file_server(Config) ->
- doc("File server example with directory listing."),
- try
- do_compile_and_start(file_server, Config),
- do_file_server(tcp, http, Config),
- do_file_server(tcp, http2, Config)
- after
- do_stop(file_server)
- end.
- do_file_server(Transport, Protocol, Config) ->
- %% Directory.
- {200, DirHeaders, <<"<!DOCTYPE html><html>", _/bits >>} = do_get(Transport, Protocol, "/", Config),
- {_, <<"text/html; charset=utf-8">>} = lists:keyfind(<<"content-type">>, 1, DirHeaders),
- _ = do_rest_get(Transport, Protocol, "/", <<"application/json">>, undefined, Config),
- %% Files.
- {200, _, _} = do_get(Transport, Protocol, "/small.mp4", Config),
- {200, _, _} = do_get(Transport, Protocol, "/small.ogv", Config),
- {200, _, _} = do_get(Transport, Protocol, "/test.txt", Config),
- {200, _, _} = do_get(Transport, Protocol, "/video.html", Config),
- {200, _, _} = do_get(Transport, Protocol,
- ["/", cow_uri:urlencode(<<"中文"/utf8>>), "/", cow_uri:urlencode(<<"中文.html"/utf8>>)],
- Config),
- ok.
- %% Markdown middleware.
- markdown_middleware(Config) ->
- doc("Markdown middleware example."),
- try
- do_compile_and_start(markdown_middleware, Config),
- do_markdown_middleware(tcp, http, Config),
- do_markdown_middleware(tcp, http2, Config)
- after
- do_stop(markdown_middleware)
- end.
- do_markdown_middleware(Transport, Protocol, Config) ->
- {200, Headers, <<"<h1>", _/bits >>} = do_get(Transport, Protocol, "/video.html", Config),
- {_, <<"text/html">>} = lists:keyfind(<<"content-type">>, 1, Headers),
- ok.
- %% Upload.
- upload(Config) ->
- doc("Upload example."),
- try
- do_compile_and_start(upload, Config),
- do_upload(tcp, http, Config),
- do_upload(tcp, http2, Config)
- after
- do_stop(upload)
- end.
- do_upload(Transport, Protocol, Config) ->
- {200, _, << "<html>", _/bits >>} = do_get(Transport, Protocol, "/", Config),
- %% Use POST to upload a file using multipart.
- ConnPid = gun_open([{port, 8080}, {type, Transport}, {protocol, Protocol}|Config]),
- Ref = gun:post(ConnPid, "/upload", [
- {<<"content-type">>, <<"multipart/form-data;boundary=deadbeef">>}
- ], <<
- "--deadbeef\r\n"
- "Content-Disposition: form-data; name=\"inputfile\"; filename=\"test.txt\"\r\n"
- "Content-Type: text/plain\r\n"
- "\r\n"
- "Cowboy upload example!\r\n"
- "--deadbeef--">>),
- {response, fin, 204, _} = gun:await(ConnPid, Ref),
- ok.
- %% Websocket.
- websocket(Config) ->
- doc("Websocket example."),
- try
- do_compile_and_start(websocket, Config),
- %% We can only initiate a Websocket connection from HTTP/1.1.
- {ok, Pid} = gun:open("127.0.0.1", 8080, #{protocols => [http], retry => 0}),
- {ok, http} = gun:await_up(Pid),
- _ = monitor(process, Pid),
- StreamRef = gun:ws_upgrade(Pid, "/websocket", [], #{compress => true}),
- receive
- {gun_upgrade, Pid, StreamRef, _, _} ->
- ok;
- Msg1 ->
- exit({connection_failed, Msg1})
- end,
- %% Check that we receive the message sent on timer on init.
- receive
- {gun_ws, Pid, StreamRef, {text, <<"Hello!">>}} ->
- ok
- after 2000 ->
- exit(timeout)
- end,
- %% Check that we receive subsequent messages sent on timer.
- receive
- {gun_ws, Pid, StreamRef, {text, <<"How' you doin'?">>}} ->
- ok
- after 2000 ->
- exit(timeout)
- end,
- %% Check that we receive the echoed message.
- gun:ws_send(Pid, StreamRef, {text, <<"hello">>}),
- receive
- {gun_ws, Pid, StreamRef, {text, <<"That's what she said! hello">>}} ->
- ok
- after 500 ->
- exit(timeout)
- end,
- gun:ws_send(Pid, StreamRef, close)
- after
- do_stop(websocket)
- end.
|