mirror of
https://github.com/ninenines/cowboy.git
synced 2025-07-14 12:20:24 +00:00
Add a REST example that streams its response
This commit is contained in:
parent
bce7b860ee
commit
f9dd3c9e54
9 changed files with 208 additions and 0 deletions
|
@ -34,6 +34,9 @@ Cowboy Examples
|
||||||
* [rest_pastebin](./rest_pastebin):
|
* [rest_pastebin](./rest_pastebin):
|
||||||
create text objects and return the data type that matches the request type (html, text)
|
create text objects and return the data type that matches the request type (html, text)
|
||||||
|
|
||||||
|
* [rest_stream_response](./rest_stream_response):
|
||||||
|
stream results from a data store
|
||||||
|
|
||||||
* [static_world](./static_world):
|
* [static_world](./static_world):
|
||||||
static file handler
|
static file handler
|
||||||
|
|
||||||
|
|
66
examples/rest_stream_response/README.md
Normal file
66
examples/rest_stream_response/README.md
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
Cowboy REST Streaming Responses
|
||||||
|
===============================
|
||||||
|
|
||||||
|
To compile this example you need rebar in your PATH.
|
||||||
|
|
||||||
|
Type the following command:
|
||||||
|
```
|
||||||
|
$ rebar get-deps compile
|
||||||
|
```
|
||||||
|
|
||||||
|
You can then start the Erlang node with the following command:
|
||||||
|
```
|
||||||
|
./start.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
This example simulates streaming a large amount of data from a data store one
|
||||||
|
record at a time in CSV format. It also uses a constraint to ensure that the
|
||||||
|
last segment of the route is an integer.
|
||||||
|
|
||||||
|
Examples
|
||||||
|
--------
|
||||||
|
|
||||||
|
### Get records with a field 2 value of 1
|
||||||
|
|
||||||
|
``` bash
|
||||||
|
$ curl -i localhost:8080
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
transfer-encoding: identity
|
||||||
|
server: Cowboy
|
||||||
|
date: Sun, 10 Feb 2013 19:32:16 GMT
|
||||||
|
connection: close
|
||||||
|
content-type: text/csv
|
||||||
|
|
||||||
|
DBUZGQ0C,1,28
|
||||||
|
BgoQAxMV,1,6
|
||||||
|
DAYEFxER,1,18
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
### Get records with a field 2 value of 4
|
||||||
|
|
||||||
|
``` bash
|
||||||
|
$ curl -i localhost:8080/4
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
transfer-encoding: identity
|
||||||
|
server: Cowboy
|
||||||
|
date: Sun, 10 Feb 2013 19:34:31 GMT
|
||||||
|
connection: close
|
||||||
|
content-type: text/csv
|
||||||
|
|
||||||
|
ABcFDxcE,4,42
|
||||||
|
DgYQCgEE,4,5
|
||||||
|
CA8BBhYD,4,10
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
### Get a 404
|
||||||
|
|
||||||
|
``` bash
|
||||||
|
$ curl -i localhost:8080/foo
|
||||||
|
HTTP/1.1 404 Not Found
|
||||||
|
connection: keep-alive
|
||||||
|
server: Cowboy
|
||||||
|
date: Sun, 10 Feb 2013 19:36:16 GMT
|
||||||
|
content-length: 0
|
||||||
|
```
|
4
examples/rest_stream_response/rebar.config
Normal file
4
examples/rest_stream_response/rebar.config
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
{deps, [
|
||||||
|
{cowboy, ".*",
|
||||||
|
{git, "git://github.com/extend/cowboy.git", "master"}}
|
||||||
|
]}.
|
|
@ -0,0 +1,15 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
{application, rest_stream_response, [
|
||||||
|
{description, "Cowboy REST with streaming."},
|
||||||
|
{vsn, "1"},
|
||||||
|
{modules, []},
|
||||||
|
{registered, []},
|
||||||
|
{applications, [
|
||||||
|
kernel,
|
||||||
|
stdlib,
|
||||||
|
cowboy
|
||||||
|
]},
|
||||||
|
{mod, {rest_stream_response_app, []}},
|
||||||
|
{env, []}
|
||||||
|
]}.
|
14
examples/rest_stream_response/src/rest_stream_response.erl
Normal file
14
examples/rest_stream_response/src/rest_stream_response.erl
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
-module(rest_stream_response).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
-export([start/0]).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
|
||||||
|
start() ->
|
||||||
|
ok = application:start(crypto),
|
||||||
|
ok = application:start(ranch),
|
||||||
|
ok = application:start(cowboy),
|
||||||
|
ok = application:start(rest_stream_response).
|
|
@ -0,0 +1,38 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
-module(rest_stream_response_app).
|
||||||
|
-behaviour(application).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
-export([start/2]).
|
||||||
|
-export([stop/1]).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
|
||||||
|
start(_Type, _Args) ->
|
||||||
|
Table = ets:new(stream_tab, []),
|
||||||
|
generate_rows(Table, 1000),
|
||||||
|
Dispatch = cowboy_router:compile([
|
||||||
|
{'_', [
|
||||||
|
{"/[:v1]", [{v1, int}], toppage_handler, Table}
|
||||||
|
]}
|
||||||
|
]),
|
||||||
|
{ok, _} = cowboy:start_http(http, 100, [{port, 8080}], [
|
||||||
|
{env, [{dispatch, Dispatch}]}
|
||||||
|
]),
|
||||||
|
rest_stream_response_sup:start_link().
|
||||||
|
|
||||||
|
stop(_State) ->
|
||||||
|
ok.
|
||||||
|
|
||||||
|
generate_rows(_Table, 0) -> ok;
|
||||||
|
generate_rows(Table, N) ->
|
||||||
|
ets:insert(Table, {key(), val(), val()}),
|
||||||
|
generate_rows(Table, N - 1).
|
||||||
|
|
||||||
|
key() -> key(10).
|
||||||
|
key(N) -> key(<< (random:uniform(26) - 1) >>, N - 1).
|
||||||
|
key(Acc, 0) -> binary_part(base64:encode(Acc), 0, 8);
|
||||||
|
key(Acc, N) -> key(<< Acc/binary, (random:uniform(26) - 1) >>, N - 1).
|
||||||
|
val() -> random:uniform(50).
|
|
@ -0,0 +1,23 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
-module(rest_stream_response_sup).
|
||||||
|
-behaviour(supervisor).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
-export([start_link/0]).
|
||||||
|
|
||||||
|
%% supervisor.
|
||||||
|
-export([init/1]).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
|
||||||
|
-spec start_link() -> {ok, pid()}.
|
||||||
|
start_link() ->
|
||||||
|
supervisor:start_link({local, ?MODULE}, ?MODULE, []).
|
||||||
|
|
||||||
|
%% supervisor.
|
||||||
|
|
||||||
|
init([]) ->
|
||||||
|
Procs = [],
|
||||||
|
{ok, {{one_for_one, 10, 10}, Procs}}.
|
42
examples/rest_stream_response/src/toppage_handler.erl
Normal file
42
examples/rest_stream_response/src/toppage_handler.erl
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
%% @doc Streaming handler.
|
||||||
|
-module(toppage_handler).
|
||||||
|
|
||||||
|
-export([init/3]).
|
||||||
|
-export([rest_init/2]).
|
||||||
|
-export([content_types_provided/2]).
|
||||||
|
-export([streaming_csv/2]).
|
||||||
|
|
||||||
|
init(_Transport, _Req, _Table) ->
|
||||||
|
{upgrade, protocol, cowboy_rest}.
|
||||||
|
|
||||||
|
rest_init(Req, Table) ->
|
||||||
|
{ok, Req, Table}.
|
||||||
|
|
||||||
|
content_types_provided(Req, State) ->
|
||||||
|
{[
|
||||||
|
{{<<"text">>, <<"csv">>, []}, streaming_csv}
|
||||||
|
], Req, State}.
|
||||||
|
|
||||||
|
streaming_csv(Req, Table) ->
|
||||||
|
{N, Req1} = cowboy_req:binding(v1, Req, 1),
|
||||||
|
MS = [{{'$1', '$2', '$3'}, [{'==', '$2', N}], ['$$']}],
|
||||||
|
|
||||||
|
{{stream, result_streamer(Table, MS)}, Req1, Table}.
|
||||||
|
|
||||||
|
result_streamer(Table, MS) ->
|
||||||
|
fun (Socket, Transport) ->
|
||||||
|
send_records(Socket, Transport, ets:select(Table, MS, 1))
|
||||||
|
end.
|
||||||
|
|
||||||
|
send_records(Socket, Transport, {[Rec], Cont}) ->
|
||||||
|
timer:sleep(500),
|
||||||
|
send_line(Socket, Transport, Rec),
|
||||||
|
send_records(Socket, Transport, ets:select(Cont));
|
||||||
|
send_records(_Socket, _Transport, '$end_of_table') ->
|
||||||
|
ok.
|
||||||
|
|
||||||
|
send_line(Socket, Transport, [Key, V1, V2]) ->
|
||||||
|
Transport:send(Socket,
|
||||||
|
[Key, $,, integer_to_list(V1), $,, integer_to_list(V2), $\r, $\n]).
|
3
examples/rest_stream_response/start.sh
Executable file
3
examples/rest_stream_response/start.sh
Executable file
|
@ -0,0 +1,3 @@
|
||||||
|
#!/bin/sh
|
||||||
|
erl -pa ebin deps/*/ebin -s rest_stream_response \
|
||||||
|
-eval "io:format(\"Streaming results: curl -i http://localhost:8080~n\")."
|
Loading…
Add table
Add a link
Reference in a new issue