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

Add CORS example

This commit is contained in:
Andrei Nesterov 2016-07-02 18:22:56 +03:00
parent 0ba3a9a222
commit bd6b17e799
8 changed files with 281 additions and 0 deletions

View file

@ -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

View 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

View 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)=[])
----

View file

@ -0,0 +1,2 @@
{release, {cors_hello_world_example, "1"}, [cors_hello_world]}.
{extended_start_script, true}.

View 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.

View file

@ -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.

View 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}}.

View 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}.