Elixir: Ошибка аргумента при доступе к ETS из задачи Elixir

#elixir #gen-server #ets

#elixir #общий сервер #ets

Вопрос:

В настоящее время я пытаюсь создать клон Redis в Elixir. В рамках этой работы я использую задачу для обработки запроса на получение / установку, и я использую ETS для хранения ключей и значений.

 ** (ArgumentError) argument error
    (stdlib 3.13.2) :ets.insert(:kv, {"doomspork", "Sean", ["Elixir", "Ruby", "Java"]})
 

Вот шаги, которые я предпринял с соответствующими разделами кода:

  1. Я создал общедоступную таблицу ETS
  def init(_) do
    :ets.new(:kv, [:set, :public, :named_table])
  end
 
  1. В соответствии с примером в Elixir School я пытаюсь вставить и прочитать таблицу ETS
  :ets.insert(:kv, {"doomspork", "Sean", ["Elixir", "Ruby", "Java"]})
    IO.inspect(:ets.lookup(:kv, "doomspork"))
 
  1. К сожалению, это дает мне ошибку, описанную здесь:
 [error] Task #PID<0.159.0> started from #PID<0.156.0> terminating
** (ArgumentError) argument error
    (stdlib 3.13.2) :ets.insert(:kv, {"doomspork", "Sean", ["Elixir", "Ruby", "Java"]})
    (redis 1.0.0) lib/server.ex:61: Server.write_line/2
    (redis 1.0.0) lib/server.ex:41: Server.serve/1
    (elixir 1.11.0) lib/task/supervised.ex:90: Task.Supervised.invoke_mfa/2
    (stdlib 3.13.2) proc_lib.erl:226: :proc_lib.init_p_do_apply/3
Function: #Function<0.28410304/0 in Server.loop_acceptor/1>
    Args: []
 

Я надеялся понять, почему возникает ошибка, или как получить предложения о том, как я мог бы приступить к отладке этой проблемы. Любая помощь будет высоко оценена.

Я перечислил весь файл .ex для справки. Общие отзывы о коде в целом также приветствуются — я относительно новичок в Elixir и хотел бы услышать ваши предложения, чтобы я мог улучшить.

 
defmodule Server do
  @moduledoc """
  Your implementation of a Redis server
  """
  use Application
  require Logger

  def start(_type, _args) do
    Supervisor.start_link(
      [{Task.Supervisor, name: Server.TaskSupervisor}, {Task, fn -> Server.listen() end}],
      strategy: :one_for_one
    )
  end

  def init(_) do
    :ets.new(:kv, [:set, :public, :named_table])
  end

  @doc """
  Listen for incoming connections
  """
  def listen() do
    {:ok, socket} = :gen_tcp.listen(6379, [:binary, active: false, reuseaddr: true])
    Logger.info("Accepting connections on 6379")
    loop_acceptor(socket)
  end

  defp loop_acceptor(socket) do
    {:ok, client} = :gen_tcp.accept(socket)
    {:ok, pid} = Task.Supervisor.start_child(Server.TaskSupervisor, fn -> serve(client) end)
    :ok = :gen_tcp.controlling_process(client, pid)
    loop_acceptor(socket)
  end

  defp serve(client) do
    client
    |> read_line()
    |> write_line(client)

    serve(client)
  end

  defp read_line(socket) do
    {:ok, data} = :gen_tcp.recv(socket, 0)
    data
  end

  defp echo(socket, command_arr) do
    echo_statement = Enum.at(command_arr, -2)
    IO.inspect(echo_statement)
    :gen_tcp.send(socket, " #{echo_statement}rn")
  end
  defp write_line(line, socket) do
    command_arr = String.split(line, "\r\n")
    command = Enum.at(command_arr, 2) |>String.downcase
    IO.inspect(command_arr)
    IO.inspect(command)
    :ets.insert(:kv, {"doomspork", "Sean", ["Elixir", "Ruby", "Java"]})
    IO.inspect(:ets.lookup(:kv, "doomspork"))
    case command do
      "ping" -> :gen_tcp.send(socket, " PONGrn")
      "set"-> :ets.insert(:kv, {"samplekey", "sampleresp"})
      "get"-> :ets.lookup(:kv, "samplekey")
      "echo"-> echo(socket,command_arr)
      _ -> :gen_tcp.send(socket, "Nah")
    end
  end
end
 

Ответ №1:

Вы пытаетесь получить доступ к ETS до того, как он был создан. init/1 обратный вызов вызывается внутри процесса после его запуска, но первая попытка доступа происходит из задачи, запущенной, start которая может произойти раньше. В вашем случае это приложение, у Application которого вообще нет init/1 обратного вызова.

Просто перейдите :ets.new(:kv, [:set, :public, :named_table]) к start/2 тому, что предшествует чему-либо еще.