erlang网络库
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.
 

203 rader
6.6 KiB

-module(ntUdpSrv).
-include("eNet.hrl").
-include("ntCom.hrl").
-compile(inline).
-compile({inline_size, 128}).
-export([
start_link/3
, getOpts/1
, getOpenPort/1
, init_it/3
, system_code_change/4
, system_continue/3
, system_get_state/1
, system_terminate/4
]).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% genActor start %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-spec(start_link(atom(), inet:port_number(), [listenOpt()]) -> {ok, pid()} | ignore | {error, term()}).
start_link(UoName, Port, UoOpts) ->
proc_lib:start_link(?MODULE, init_it, [UoName, self(), {Port, UoOpts}], infinity, []).
init_it(UoName, Parent, Args) ->
case safeRegister(UoName) of
true ->
process_flag(trap_exit, true),
modInit(Parent, Args);
{false, Pid} ->
proc_lib:init_ack(Parent, {error, {already_started, Pid}})
end.
-spec system_code_change(term(), module(), undefined | term(), term()) -> {ok, term()}.
system_code_change(State, _Module, _OldVsn, _Extra) ->
{ok, State}.
-spec system_continue(pid(), [], {module(), atom(), pid(), term()}) -> ok.
system_continue(_Parent, _Debug, {Parent, State}) ->
loop(Parent, State).
-spec system_get_state(term()) -> {ok, term()}.
system_get_state(State) ->
{ok, State}.
-spec system_terminate(term(), pid(), [], term()) -> none().
system_terminate(Reason, _Parent, _Debug, _State) ->
exit(Reason).
safeRegister(Name) ->
try register(Name, self()) of
true -> true
catch
_:_ -> {false, whereis(Name)}
end.
modInit(Parent, Args) ->
case init(Args) of
{ok, State} ->
proc_lib:init_ack(Parent, {ok, self()}),
loop(Parent, State);
{stop, Reason} ->
proc_lib:init_ack(Parent, {error, Reason}),
exit(Reason)
end.
loop(Parent, State) ->
receive
{system, From, Request} ->
sys:handle_system_msg(Request, From, Parent, ?MODULE, [], {Parent, State});
{'EXIT', Parent, Reason} ->
terminate(Reason, State);
Msg ->
case handleMsg(Msg, State) of
kpS ->
loop(Parent, State);
{ok, NewState} ->
loop(Parent, NewState);
{stop, Reason} ->
terminate(Reason, State),
exit(Reason)
end
end.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% genActor end %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-record(state, {
listenAddr :: inet:ip_address()
, listenPort :: inet:port_number()
, oSock :: inet:socket()
, opts :: [listenOpt()]
, conMod :: atom()
, peers = #{} :: map()
}).
-define(DefUdpOpts, [binary, {reuseaddr, true}]).
init({Port, UoOpts}) ->
UdpOpts = ?getLValue(udpOpts, UoOpts, []),
LastUdpOpts = ntCom:mergeOpts(?DefUdpOpts, UdpOpts),
%% Don't active the socket...
case gen_udp:open(Port, lists:keystore(active, 1, LastUdpOpts, {active, false})) of
{ok, OSock} ->
AptCnt = ?getLValue(aptCnt, UoOpts, ?AptCnt),
ConMod = ?getLValue(conMod, UoOpts, undefined),
{ok, {LAddr, LPort}} = inet:sockname(OSock),
?ntInfo("success to open on ~p ~p ~n", [LAddr, LPort]),
ok = inet:setopts(OSock, [{active, 100}]),
{ok, #state{listenAddr = LAddr, listenPort = LPort, oSock = OSock, conMod = ConMod, opts = [{acceptors, AptCnt}, {tcpOpts, LastUdpOpts}]}};
{error, Reason} ->
?ntErr("failed to open on ~p - ~p (~s) ~n", [Port, Reason, inet:format_error(Reason)]),
{stop, Reason}
end.
handleMsg({udp, Sock, IP, InPortNo, AncData, Packet}, #state{oSock = Sock, conMod = ConMod, peers = Peers} = State) ->
case maps:find({IP, InPortNo}, Peers) of
{ok, Pid} ->
Pid ! {datagram, self(), Packet},
kpS;
error ->
try ConMod:datagram(Sock, IP, InPortNo, AncData, Packet) of
{ok, Pid} ->
_Ref = erlang:monitor(process, Pid),
Pid ! {datagram, self(), Packet},
{noreply, addPeer({IP, InPortNo}, Pid, State)};
{error, Reason} ->
?ntErr("Failed to start udp channel for peer ~p ~p reason: ~p", [IP, InPortNo, Reason]),
kpS
catch
C:R:S ->
?ntErr("Exception occurred when starting udp channel for peer ~p ~p, reason: ~p", [IP, InPortNo, {C, R, S}]),
kpS
end
end;
handleMsg({udp, Sock, IP, InPortNo, Packet}, #state{oSock = Sock, conMod = ConMod, peers = Peers} = State) ->
case maps:find({IP, InPortNo}, Peers) of
{ok, Pid} ->
Pid ! {datagram, self(), Packet},
kpS;
error ->
try ConMod:datagram(Sock, IP, InPortNo, undefined, Packet) of
{ok, Pid} ->
_Ref = erlang:monitor(process, Pid),
Pid ! {datagram, self(), Packet},
{ok, addPeer({IP, InPortNo}, Pid, State)};
{error, Reason} ->
?ntErr("Failed to start udp channel for peer ~p ~p reason: ~p", [IP, InPortNo, Reason]),
kpS
catch
C:R:S ->
?ntErr("Exception occurred when starting udp channel for peer ~p ~p, reason: ~p", [IP, InPortNo, {C, R, S}]),
kpS
end
end;
handleMsg({udp_passive, Sock}, #state{oSock = Sock} = State) ->
inet:setopts(Sock, [{active, 100}]),
kpS;
handleMsg({'DOWN', _MRef, process, DownPid, _Reason}, State = #state{peers = Peers}) ->
peerDown(DownPid, Peers, State);
handleMsg({'EXIT', DownPid, _Reason}, State = #state{peers = Peers}) ->
peerDown(DownPid, Peers, State);
handleMsg({'$gen_call', From, miOpts}, #state{opts = Opts} = _State) ->
gen_server:reply(From, Opts),
kpS;
handleMsg({'$gen_call', From, miOpenPort}, #state{listenPort = LPort} = _State) ->
gen_server:reply(From, LPort),
kpS;
handleMsg(_Msg, _State) ->
?ntErr("~p unexpected info: ~p ~n", [?MODULE, _Msg]),
kpS.
terminate(_Reason, #state{oSock = LSock, listenAddr = Addr, listenPort = Port}) ->
?ntInfo("stopped on ~s:~p ~n", [inet:ntoa(Addr), Port]),
catch gen_udp:close(LSock),
ok.
-spec getOpts(pid()) -> [listenOpt()].
getOpts(Listener) ->
gen_server:call(Listener, miOpts).
-spec getOpenPort(pid()) -> inet:port_number().
getOpenPort(Listener) ->
gen_server:call(Listener, miOpenPort).
addPeer(Peer, Pid, State = #state{peers = Peers}) ->
State#state{peers = maps:put(Pid, Peer, maps:put(Peer, Pid, Peers))}.
delPeer(Peer, Pid, State = #state{peers = Peers}) ->
State#state{peers = maps:remove(Peer, maps:remove(Pid, Peers))}.
peerDown(DownPid, Peers, State) ->
case maps:find(DownPid, Peers) of
{ok, Peer} ->
{ok, delPeer(Peer, DownPid, State)};
error ->
kpS
end.