mirror of
https://github.com/ninenines/cowboy.git
synced 2025-07-14 12:20:24 +00:00
Merge bd6b17e799
into a8c717718a
This commit is contained in:
commit
7179955556
8 changed files with 281 additions and 0 deletions
|
@ -9,6 +9,9 @@
|
||||||
* link:cookie[]:
|
* link:cookie[]:
|
||||||
set cookies from server and client side
|
set cookies from server and client side
|
||||||
|
|
||||||
|
* link:cors_hello_world[]:
|
||||||
|
simplest CORS middleware example
|
||||||
|
|
||||||
* link:echo_get[]:
|
* link:echo_get[]:
|
||||||
parse and echo a GET query string
|
parse and echo a GET query string
|
||||||
|
|
||||||
|
|
8
examples/cors_hello_world/Makefile
Normal file
8
examples/cors_hello_world/Makefile
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
PROJECT = cors_hello_world
|
||||||
|
PROJECT_DESCRIPTION = Cowboy CORS Hello World example
|
||||||
|
PROJECT_VERSION = 1
|
||||||
|
|
||||||
|
DEPS = cowboy
|
||||||
|
dep_cowboy_commit = master
|
||||||
|
|
||||||
|
include ../../erlang.mk
|
159
examples/cors_hello_world/README.asciidoc
Normal file
159
examples/cors_hello_world/README.asciidoc
Normal file
|
@ -0,0 +1,159 @@
|
||||||
|
= Hello world example
|
||||||
|
|
||||||
|
To try this example, you need GNU `make` and `git` in your PATH.
|
||||||
|
|
||||||
|
To build and run the example, use the following command:
|
||||||
|
|
||||||
|
[source,bash]
|
||||||
|
$ make run
|
||||||
|
|
||||||
|
== HTTP/1.1 example output
|
||||||
|
|
||||||
|
[source,bash]
|
||||||
|
----
|
||||||
|
$ curl -i \
|
||||||
|
-XOPTIONS \
|
||||||
|
-H'Origin:http://example.org' \
|
||||||
|
-H'Access-Control-Request-Method:GET' \
|
||||||
|
-H'Access-Control-Request-Headers:Authorization' \
|
||||||
|
'http://localhost:8080'
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
access-control-allow-headers: Authorization
|
||||||
|
access-control-allow-methods: GET, PUT
|
||||||
|
access-control-allow-origin: http://example.org
|
||||||
|
access-control-max-age: 0
|
||||||
|
content-length: 0
|
||||||
|
date: Sat, 02 Jul 2016 14:56:30 GMT
|
||||||
|
server: Cowboy
|
||||||
|
vary: Origin
|
||||||
|
|
||||||
|
$ curl -i \
|
||||||
|
-XGET \
|
||||||
|
-H'Origin:http://example.org' \
|
||||||
|
'http://localhost:8080'
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
access-control-allow-origin: http://example.org
|
||||||
|
content-length: 12
|
||||||
|
content-type: text/plain
|
||||||
|
date: Sat, 02 Jul 2016 14:46:42 GMT
|
||||||
|
server: Cowboy
|
||||||
|
vary: Origin
|
||||||
|
|
||||||
|
Hello world!
|
||||||
|
----
|
||||||
|
|
||||||
|
== HTTP/2 example output
|
||||||
|
|
||||||
|
[source,bash]
|
||||||
|
----
|
||||||
|
$ nghttp -v \
|
||||||
|
-H':method: OPTIONS' \
|
||||||
|
-H'Origin:http://example.org' \
|
||||||
|
-H'Access-Control-Request-Method:GET' \
|
||||||
|
-H'Access-Control-Request-Headers:Authorization' \
|
||||||
|
'http://localhost:8080'
|
||||||
|
[ 0.002] Connected
|
||||||
|
[ 0.002] send SETTINGS frame <length=12, flags=0x00, stream_id=0>
|
||||||
|
(niv=2)
|
||||||
|
[SETTINGS_MAX_CONCURRENT_STREAMS(0x03):100]
|
||||||
|
[SETTINGS_INITIAL_WINDOW_SIZE(0x04):65535]
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=3>
|
||||||
|
(dep_stream_id=0, weight=201, exclusive=0)
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=5>
|
||||||
|
(dep_stream_id=0, weight=101, exclusive=0)
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=7>
|
||||||
|
(dep_stream_id=0, weight=1, exclusive=0)
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=9>
|
||||||
|
(dep_stream_id=7, weight=1, exclusive=0)
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=11>
|
||||||
|
(dep_stream_id=3, weight=1, exclusive=0)
|
||||||
|
[ 0.002] send HEADERS frame <length=126, flags=0x25, stream_id=13>
|
||||||
|
; END_STREAM | END_HEADERS | PRIORITY
|
||||||
|
(padlen=0, dep_stream_id=11, weight=16, exclusive=0)
|
||||||
|
; Open new stream
|
||||||
|
:method: OPTIONS
|
||||||
|
:path: /
|
||||||
|
:scheme: http
|
||||||
|
:authority: localhost:8080
|
||||||
|
accept: */*
|
||||||
|
accept-encoding: gzip, deflate
|
||||||
|
user-agent: nghttp2/1.11.1
|
||||||
|
origin: http://example.org
|
||||||
|
access-control-request-method: GET
|
||||||
|
access-control-request-headers: Authorization
|
||||||
|
[ 0.007] recv SETTINGS frame <length=0, flags=0x00, stream_id=0>
|
||||||
|
(niv=0)
|
||||||
|
[ 0.007] recv SETTINGS frame <length=0, flags=0x01, stream_id=0>
|
||||||
|
; ACK
|
||||||
|
(niv=0)
|
||||||
|
[ 0.007] send SETTINGS frame <length=0, flags=0x01, stream_id=0>
|
||||||
|
; ACK
|
||||||
|
(niv=0)
|
||||||
|
[ 0.007] recv (stream_id=13) :status: 200
|
||||||
|
[ 0.007] recv (stream_id=13) access-control-allow-headers: Authorization
|
||||||
|
[ 0.007] recv (stream_id=13) access-control-allow-methods: GET, PUT
|
||||||
|
[ 0.007] recv (stream_id=13) access-control-allow-origin: http://example.org
|
||||||
|
[ 0.007] recv (stream_id=13) access-control-max-age: 0
|
||||||
|
[ 0.007] recv (stream_id=13) content-length: 0
|
||||||
|
[ 0.008] recv (stream_id=13) date: Sat, 02 Jul 2016 15:06:07 GMT
|
||||||
|
[ 0.008] recv (stream_id=13) server: Cowboy
|
||||||
|
[ 0.008] recv (stream_id=13) vary: Origin
|
||||||
|
[ 0.008] recv HEADERS frame <length=138, flags=0x05, stream_id=13>
|
||||||
|
; END_STREAM | END_HEADERS
|
||||||
|
(padlen=0)
|
||||||
|
; First response header
|
||||||
|
[ 0.008] send GOAWAY frame <length=8, flags=0x00, stream_id=0>
|
||||||
|
(last_stream_id=0, error_code=NO_ERROR(0x00), opaque_data(0)=[])
|
||||||
|
|
||||||
|
$ nghttp -v \
|
||||||
|
-H':method:GET' \
|
||||||
|
-H'Origin:http://example.org' \
|
||||||
|
'http://localhost:8080'
|
||||||
|
[ 0.002] Connected
|
||||||
|
[ 0.002] send SETTINGS frame <length=12, flags=0x00, stream_id=0>
|
||||||
|
(niv=2)
|
||||||
|
[SETTINGS_MAX_CONCURRENT_STREAMS(0x03):100]
|
||||||
|
[SETTINGS_INITIAL_WINDOW_SIZE(0x04):65535]
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=3>
|
||||||
|
(dep_stream_id=0, weight=201, exclusive=0)
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=5>
|
||||||
|
(dep_stream_id=0, weight=101, exclusive=0)
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=7>
|
||||||
|
(dep_stream_id=0, weight=1, exclusive=0)
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=9>
|
||||||
|
(dep_stream_id=7, weight=1, exclusive=0)
|
||||||
|
[ 0.002] send PRIORITY frame <length=5, flags=0x00, stream_id=11>
|
||||||
|
(dep_stream_id=3, weight=1, exclusive=0)
|
||||||
|
[ 0.002] send HEADERS frame <length=59, flags=0x25, stream_id=13>
|
||||||
|
; END_STREAM | END_HEADERS | PRIORITY
|
||||||
|
(padlen=0, dep_stream_id=11, weight=16, exclusive=0)
|
||||||
|
; Open new stream
|
||||||
|
:method: GET
|
||||||
|
:path: /
|
||||||
|
:scheme: http
|
||||||
|
:authority: localhost:8080
|
||||||
|
accept: */*
|
||||||
|
accept-encoding: gzip, deflate
|
||||||
|
user-agent: nghttp2/1.11.1
|
||||||
|
origin: http://example.org
|
||||||
|
[ 0.004] recv SETTINGS frame <length=0, flags=0x00, stream_id=0>
|
||||||
|
(niv=0)
|
||||||
|
[ 0.005] recv SETTINGS frame <length=0, flags=0x01, stream_id=0>
|
||||||
|
; ACK
|
||||||
|
(niv=0)
|
||||||
|
[ 0.005] recv (stream_id=13) :status: 200
|
||||||
|
[ 0.005] recv (stream_id=13) access-control-allow-origin: http://example.org
|
||||||
|
[ 0.005] recv (stream_id=13) content-length: 12
|
||||||
|
[ 0.005] recv (stream_id=13) content-type: text/plain
|
||||||
|
[ 0.005] recv (stream_id=13) date: Sat, 02 Jul 2016 15:07:01 GMT
|
||||||
|
[ 0.005] recv (stream_id=13) server: Cowboy
|
||||||
|
[ 0.005] recv (stream_id=13) vary: Origin
|
||||||
|
[ 0.005] recv HEADERS frame <length=67, flags=0x04, stream_id=13>
|
||||||
|
; END_HEADERS
|
||||||
|
(padlen=0)
|
||||||
|
; First response header
|
||||||
|
Hello world![ 0.009] recv DATA frame <length=12, flags=0x01, stream_id=13>
|
||||||
|
; END_STREAM
|
||||||
|
[ 0.009] send GOAWAY frame <length=8, flags=0x00, stream_id=0>
|
||||||
|
(last_stream_id=0, error_code=NO_ERROR(0x00), opaque_data(0)=[])
|
||||||
|
----
|
2
examples/cors_hello_world/relx.config
Normal file
2
examples/cors_hello_world/relx.config
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
{release, {cors_hello_world_example, "1"}, [cors_hello_world]}.
|
||||||
|
{extended_start_script, true}.
|
26
examples/cors_hello_world/src/cors_hello_world_app.erl
Normal file
26
examples/cors_hello_world/src/cors_hello_world_app.erl
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
-module(cors_hello_world_app).
|
||||||
|
-behaviour(application).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
-export([start/2]).
|
||||||
|
-export([stop/1]).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
|
||||||
|
start(_Type, _Args) ->
|
||||||
|
Dispatch = cowboy_router:compile([
|
||||||
|
{'_', [
|
||||||
|
{"/", toppage_handler, []}
|
||||||
|
]}
|
||||||
|
]),
|
||||||
|
{ok, _} = cowboy:start_clear(http, 100, [{port, 8080}], #{
|
||||||
|
middlewares => [cors_hello_world_middleware, cowboy_router, cowboy_handler],
|
||||||
|
env => #{dispatch => Dispatch}
|
||||||
|
}),
|
||||||
|
cors_hello_world_sup:start_link().
|
||||||
|
|
||||||
|
stop(_State) ->
|
||||||
|
ok.
|
|
@ -0,0 +1,48 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
-module(cors_hello_world_middleware).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
-export([execute/2]).
|
||||||
|
|
||||||
|
%% API.
|
||||||
|
|
||||||
|
-spec execute(Req, Env) -> {ok | stop, Req, Env} when Req :: cowboy_req:req(), Env :: any().
|
||||||
|
execute(#{headers := #{<<"origin">> := HeaderVal}} = Req, Env) ->
|
||||||
|
%%AllowedOrigins = '*',
|
||||||
|
AllowedOrigins = [
|
||||||
|
{<<"http">>, <<"example.org">>, 80},
|
||||||
|
{<<"https">>, <<"example.org">>, 443}
|
||||||
|
],
|
||||||
|
|
||||||
|
%% NOTE: we assume we always deal with single origin
|
||||||
|
[Val] = cow_http_hd:parse_origin(HeaderVal),
|
||||||
|
case check_origin(Val, AllowedOrigins) of
|
||||||
|
true -> handle_cors_request(HeaderVal, Req, Env);
|
||||||
|
_ -> {ok, Req, Env}
|
||||||
|
end;
|
||||||
|
execute(Req, Env) ->
|
||||||
|
{ok, Req, Env}.
|
||||||
|
|
||||||
|
-spec handle_cors_request(binary(), cowboy_req:req(), any()) -> cowboy_req:req().
|
||||||
|
handle_cors_request(Origin, #{method := Method} = Req, Env) ->
|
||||||
|
Req2 = cowboy_req:set_resp_header(<<"access-control-allow-origin">>, Origin, Req),
|
||||||
|
Req3 = cowboy_req:set_resp_header(<<"vary">>, <<"Origin">>, Req2),
|
||||||
|
case Method of
|
||||||
|
<<"OPTIONS">> ->
|
||||||
|
Req4 = cowboy_req:set_resp_header(<<"access-control-allow-methods">>, <<"GET, PUT">>, Req3),
|
||||||
|
Req5 = cowboy_req:set_resp_header(<<"access-control-allow-headers">>, <<"Authorization">>, Req4),
|
||||||
|
Req6 = cowboy_req:set_resp_header(<<"access-control-max-age">>, <<"0">>, Req5),
|
||||||
|
Req7 = cowboy_req:reply(200, Req6),
|
||||||
|
{stop, Req7};
|
||||||
|
_ ->
|
||||||
|
{ok, Req3, Env}
|
||||||
|
end.
|
||||||
|
|
||||||
|
-spec check_origin(Origin, [Origin] | '*') -> boolean() when Origin :: {binary(), binary(), 0..65535} | reference().
|
||||||
|
check_origin(Val, '*') when is_reference(Val) -> true;
|
||||||
|
check_origin(_, '*') -> true;
|
||||||
|
check_origin(Val, Val) -> true;
|
||||||
|
check_origin(Val, L) when is_list(L) -> lists:member(Val, L);
|
||||||
|
check_origin(_, _) -> false.
|
23
examples/cors_hello_world/src/cors_hello_world_sup.erl
Normal file
23
examples/cors_hello_world/src/cors_hello_world_sup.erl
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
%% @private
|
||||||
|
-module(cors_hello_world_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}}.
|
12
examples/cors_hello_world/src/toppage_handler.erl
Normal file
12
examples/cors_hello_world/src/toppage_handler.erl
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
%% Feel free to use, reuse and abuse the code in this file.
|
||||||
|
|
||||||
|
%% @doc Hello world handler.
|
||||||
|
-module(toppage_handler).
|
||||||
|
|
||||||
|
-export([init/2]).
|
||||||
|
|
||||||
|
init(Req, Opts) ->
|
||||||
|
cowboy_req:reply(200, #{
|
||||||
|
<<"content-type">> => <<"text/plain">>
|
||||||
|
}, <<"Hello world!">>, Req),
|
||||||
|
{ok, Req, Opts}.
|
Loading…
Add table
Add a link
Reference in a new issue