【问题标题】:erlang socket (using ranch) close in a short time?erlang套接字(使用牧场)在短时间内关闭?
【发布时间】:2014-07-11 02:29:50
【问题描述】:

我用ranch监听socket,但是在短短五秒左右的时间里,ranch关闭了socket,我的socket设置在上面,怎么回事?

    {ok, _} = ranch:start_listener(server,200, ranch_tcp, [{port, 5555},{active, once},    {max_connections, 1024}], server_protocol, []), %% start the listener

协议文件在下面,牧场监听接受一个套接字,并反向接收数据,但问题是,当向客户端发送数据时,大约五秒钟后,客户端收到消息说socket被服务器关闭了,不知道是不是ranch的默认设置导致的?

-module(reverse_protocol).
-behaviour(gen_server).
-behaviour(ranch_protocol).
%% API.
-export([start_link/4]).
%% gen_server.
-export([init/1]).
-export([init/4]).
-export([handle_call/3]).
-export([handle_cast/2]).
-export([handle_info/2]).
-export([terminate/2]).
-export([code_change/3]).
-define(TIMEOUT, 5000).
-record(state, {socket, transport}).

%% API.
start_link(Ref, Socket, Transport, Opts) ->
proc_lib:start_link(?MODULE, init, [Ref, Socket, Transport, Opts]).
%% gen_server.
%% This function is never called. We only define it so that
%% we can use the -behaviour(gen_server) attribute.

init([]) -> {ok, undefined}.

init(Ref, Socket, Transport, _Opts = []) ->
ok = proc_lib:init_ack({ok, self()}),
ok = ranch:accept_ack(Ref),
ok = Transport:setopts(Socket, [{active, once}]),
gen_server:enter_loop(?MODULE, [],
    #state{socket=Socket, transport=Transport},
    ?TIMEOUT).

handle_info({tcp, Socket, Data}, State=#state{
    socket=Socket, transport=Transport}) ->
Transport:setopts(Socket, [{active, once}]),
Transport:send(Socket, reverse_binary(Data)),
{noreply, State, ?TIMEOUT};

handle_info({tcp_closed, _Socket}, State) ->
{stop, normal, State};

handle_info({tcp_error, _, Reason}, State) ->
{stop, Reason, State};

handle_info(timeout, State) ->
{stop, normal, State};

handle_info(_Info, State) ->
{stop, normal, State}.

handle_call(_Request, _From, State) ->
{reply, ok, State}.

handle_cast(_Msg, State) ->
{noreply, State}.

terminate(_Reason, _State) ->
ok.

code_change(_OldVsn, State, _Extra) ->
{ok, State}.

%% Internal.
reverse_binary(B) when is_binary(B) ->
[list_to_binary(lists:reverse(binary_to_list(
    binary:part(B, {0, byte_size(B)-2})
))), "\r\n"].

【问题讨论】:

  • 你是如何确定牧场关闭了监听套接字的?
  • 客户端打印出原因,我也尝试连接到python套接字服务器,它工作正常! @evnu
  • 你能提供一个最小的例子来说明这个问题吗?
  • 好的,我立即将此添加到问题描述中。@evnu
  • 我为这个问题添加了更多细节,感谢您给我建议。@evnu

标签: sockets erlang ranch


【解决方案1】:

所以你的问题是因为你的 gen_server 进程超时并关闭。关闭套接字是这样做的副作用,因为牧场将套接字链接到生成的处理程序进程。

一旦新进程进入 gen_server 循环并调用gen_server:enter_loop,它在发送timeout 消息之前有?TIMEOUT 毫秒来接收消息。

-define(TIMEOUT, 5000).

init(Ref, Socket, Transport, _Opts = []) ->
  ok = proc_lib:init_ack({ok, self()}),
  ok = ranch:accept_ack(Ref),
  ok = Transport:setopts(Socket, [{active, once}]),
  gen_server:enter_loop(?MODULE, [],
    #state{socket=Socket, transport=Transport},
    ?TIMEOUT). %% timeout because of this!

handle_info({tcp, Socket, Data}, State=#state{
  socket=Socket, transport=Transport}) ->
  Transport:setopts(Socket, [{active, once}]),
  Transport:send(Socket, reverse_binary(Data)),
  {noreply, State, ?TIMEOUT}; %% timeout because of this!

因此,当这五秒钟过去并且 gen_server 在那段时间内没有收到任何消息时,它会向自己发送一条 timeout 消息,然后由 handle_info 处理

handle_info(timeout, State) ->
  {stop, normal, State};

您的handle_info 告诉 gen_server 停止,这会导致 Socket 关闭,因为两者是链接在一起的。

您可以完全删除超时,或者只是阻止超时导致进程关闭。

以下是我将如何更改 handle_info 超时代码:

handle_info(timeout, State) ->
  io:format("the socket is idle~n"),
  {noreply,State};

【讨论】:

    猜你喜欢
    • 2015-12-07
    • 2023-04-07
    • 2015-09-04
    • 2018-06-24
    • 1970-01-01
    • 1970-01-01
    • 2019-06-07
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多