aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorjuhlig <[email protected]>2019-06-21 16:01:22 +0200
committerLoïc Hoguin <[email protected]>2019-06-21 17:10:16 +0200
commite8d6401741021f76988091e6dc633afb65ffbe7b (patch)
treeba4a55fb0a3b7c461cd71986c689db3009af9648 /src
parentae84436f7ceed06a09e3fe1afb30e675579b7621 (diff)
downloadranch-e8d6401741021f76988091e6dc633afb65ffbe7b.tar.gz
ranch-e8d6401741021f76988091e6dc633afb65ffbe7b.tar.bz2
ranch-e8d6401741021f76988091e6dc633afb65ffbe7b.zip
Embedded listeners depending on ranch_server
Diffstat (limited to 'src')
-rw-r--r--src/ranch.erl10
-rw-r--r--src/ranch_embedded_sup.erl34
-rw-r--r--src/ranch_server_proxy.erl61
3 files changed, 101 insertions, 4 deletions
diff --git a/src/ranch.erl b/src/ranch.erl
index b26f3a6..0e3541f 100644
--- a/src/ranch.erl
+++ b/src/ranch.erl
@@ -73,8 +73,10 @@ start_listener(Ref, Transport, TransOpts0, Protocol, ProtoOpts)
_ = code:ensure_loaded(Transport),
case {erlang:function_exported(Transport, name, 0), validate_transport_opts(TransOpts)} of
{true, ok} ->
- maybe_started(supervisor:start_child(ranch_sup, child_spec(Ref,
- Transport, TransOpts, Protocol, ProtoOpts)));
+ ChildSpec = #{id => {ranch_listener_sup, Ref}, start => {ranch_listener_sup, start_link, [
+ Ref, Transport, TransOpts, Protocol, ProtoOpts
+ ]}, type => supervisor},
+ maybe_started(supervisor:start_child(ranch_sup, ChildSpec));
{false, _} ->
{error, {bad_transport, Transport}};
{_, TransOptsError} ->
@@ -191,9 +193,9 @@ maybe_resumed(Res) ->
-> supervisor:child_spec().
child_spec(Ref, Transport, TransOpts0, Protocol, ProtoOpts) ->
TransOpts = normalize_opts(TransOpts0),
- {{ranch_listener_sup, Ref}, {ranch_listener_sup, start_link, [
+ #{id => {ranch_embedded_sup, Ref}, start => {ranch_embedded_sup, start_link, [
Ref, Transport, TransOpts, Protocol, ProtoOpts
- ]}, permanent, infinity, supervisor, [ranch_listener_sup]}.
+ ]}, type => supervisor}.
-spec handshake(ref()) -> {ok, ranch_transport:socket()}.
handshake(Ref) ->
diff --git a/src/ranch_embedded_sup.erl b/src/ranch_embedded_sup.erl
new file mode 100644
index 0000000..aa157e0
--- /dev/null
+++ b/src/ranch_embedded_sup.erl
@@ -0,0 +1,34 @@
+%% Copyright (c) 2019, Jan Uhlig <[email protected]>
+%%
+%% 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(ranch_embedded_sup).
+
+-behavior(supervisor).
+
+-export([start_link/5]).
+-export([init/1]).
+
+-spec start_link(ranch:ref(), module(), any(), module(), any())
+ -> {ok, pid()}.
+start_link(Ref, Transport, TransOpts, Protocol, ProtoOpts) ->
+ supervisor:start_link(?MODULE, {Ref, Transport, TransOpts, Protocol, ProtoOpts}).
+
+init({Ref, Transport, TransOpts, Protocol, ProtoOpts}) ->
+ Proxy = #{id => ranch_server_proxy,
+ start => {ranch_server_proxy, start_link, []},
+ shutdown => brutal_kill},
+ Listener = #{id => {ranch_listener_sup, Ref},
+ start => {ranch_listener_sup, start_link, [Ref, Transport, TransOpts, Protocol, ProtoOpts]},
+ type => supervisor},
+ {ok, {#{strategy => rest_for_one}, [Proxy, Listener]}}.
diff --git a/src/ranch_server_proxy.erl b/src/ranch_server_proxy.erl
new file mode 100644
index 0000000..949ac33
--- /dev/null
+++ b/src/ranch_server_proxy.erl
@@ -0,0 +1,61 @@
+%% Copyright (c) 2019, Jan Uhlig <[email protected]>
+%%
+%% 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(ranch_server_proxy).
+
+-behavior(gen_server).
+
+-export([start_link/0]).
+-export([init/1]).
+-export([handle_call/3]).
+-export([handle_cast/2]).
+-export([handle_info/2]).
+-export([code_change/3]).
+
+start_link() ->
+ gen_server:start_link(?MODULE, [], []).
+
+init([]) ->
+ case wait_ranch_server(50) of
+ {ok, Monitor} ->
+ {ok, Monitor, hibernate};
+ {error, Reason} ->
+ {stop, Reason}
+ end.
+
+handle_call(_, _, Monitor) ->
+ {noreply, Monitor, hibernate}.
+
+handle_cast(_, Monitor) ->
+ {noreply, Monitor, hibernate}.
+
+handle_info({'DOWN', Monitor, process, _, Reason}, Monitor) ->
+ {stop, Reason, Monitor};
+handle_info(_, Monitor) ->
+ {noreply, Monitor, hibernate}.
+
+code_change(_, Monitor, _) ->
+ {ok, Monitor}.
+
+wait_ranch_server(N) ->
+ case whereis(ranch_server) of
+ undefined when N > 0 ->
+ receive after 100 -> ok end,
+ wait_ranch_server(N - 1);
+ undefined ->
+ {error, noproc};
+ Pid ->
+ Monitor = monitor(process, Pid),
+ {ok, Monitor}
+ end.