生产中的Elixir httpc错误

时间:2018-03-06 14:57:30

标签: erlang elixir httpc

我想使用:httpc作为我的HTTP客户端。

我使用edeliver进行发布管理等。我的:inets中有:httpc.deliver/config.exs,如:

  set applications: [
    :runtime_tools,
    :inets,
    :httpc
  ]

我还在:inets中将:extra_applications添加到mix.exs

以下是我使用:httpc的方式:

headers =
  if apikey,
    do: [{'Content-Type', 'application/json'}, {'apikey', to_charlist(apikey)}],
    else: [{'Content-Type', 'application/json'}]

http_options = [timeout: @timeout, connect_timeout: @timeout]
options = []

request = {
  to_charlist(url),
  headers,
  'application/json',
  to_charlist(encoded_obj)
}

:post
|> :httpc.request(request, http_options, options)
|> handle_response()

我收到很多错误:

=SUPERVISOR REPORT==== 6-Mar-2018::15:44:11 ===
     Supervisor: {local,httpc_handler_sup}
     Context:    child_terminated
     Reason:     {function_clause,
                     [{http_transport,close,
                          [undefined,#Port<0.21557>],
                          [{file,"http_transport.erl"},{line,346}]},
                      {gen_server,try_terminate,3,
                          [{file,"gen_server.erl"},{line,648}]},
                      {gen_server,terminate,10,
                          [{file,"gen_server.erl"},{line,833}]},
                      {proc_lib,init_p_do_apply,3,
                          [{file,"proc_lib.erl"},{line,247}]}]}
     Offender:   [{pid,<0.2596.1>},
                  {id,undefined},
                  {mfargs,{httpc_handler,start_link,undefined}},
                  {restart_type,temporary},
                  {shutdown,4000},
                  {child_type,worker}]

以及

15:44:11.743 [error] GenServer #PID<0.2595.1> terminating
** (FunctionClauseError) no function clause matching in :http_transport.close/2
    (inets) http_transport.erl:346: :http_transport.close(:undefined, #Port<0.21559>)
    (stdlib) gen_server.erl:648: :gen_server.try_terminate/3
    (stdlib) gen_server.erl:833: :gen_server.terminate/10
    (stdlib) proc_lib.erl:247: :proc_lib.init_p_do_apply/3
Last message: {:init_error, :error_sending, {#Reference<0.18155839.2531262466.203553>, {:error, :einval}}}

报告的错误相同。

这句话说的是我不太了解的事情:

15:44:11.741 [error] Bad value on output port 'tcp_inet'

我实际上并不知道为什么会这样。

我正在使用HTTPotion并且没有这个问题(尽管有其他人)。

问题在于我的开发机器。它也适用于我的机器上的类似生产的VM。但是当它进入真正的生产服务器时会抛出这个错误。

我很困惑!

1 个答案:

答案 0 :(得分:0)

在erlang中,httpc:request()函数有一个Request参数变量:

request(Method, Request, HTTPOptions, Options) -> 

Request参数变量的类型指定为:

Request = request()
request() = {url(), headers()}

其中url() = string()headers() = [header()],即列表,因此Request参数的形状必须是:

{string, []}

但是,你这样做:

request = {
  to_charlist(url),
  headers,
  'application/json',
  to_charlist(encoded_obj)
}

看起来它的形状为:

{string, [], string, string}

我不知道你为什么要复制标题列表之外的'application / json'的Content-Type,而且我不清楚to_charlist(encoded_obj)应该是什么。我会尝试:

request = {
  to_charlist(url),
  headers
}

如果to_charlist(encoded_obj)需要位于http标头中,请在标头列表中添加适当的键/值对。潜在的http标头here。也许你需要:

headers = 
  if api,
     do: [{'Content-Type', 'application/json'}, 
          {'apikey', to_charlist(apikey)}, 
          {'Authorization, to_charlist(encoded_obj)} ],
     else: [{'Content-Type', 'application/json'}, 
            {'Authorization, to_charlist(encoded_obj)} ]

可能更好地写成:

base_headers = [
    {'Content-Type', 'application/json'},
    {'Authorization', to_charlist(encoded_obj)}
]

request_headers = 
    if apikey,
        do: [ {'apikey', to_charlist(apikey)} | base_headers ],
        else: base_headers
  

问题在于我的开发机器。它也适用于   类似生产的VM也在我的机器上。但它抛出了这个错误   当它进入真正的生产服务器时。

那么,请看这里:

Bad value on output port 'tcp_inet'