Эликсир: правильный способ завести ребенка с помощью Dynamic Supervisor - PullRequest
0 голосов
/ 13 декабря 2018

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

Вот мой код.Первый - это модуль с основным методом

defmodule Simulator do
    def main(args) do
        {:ok, super_pid} = PersonManager.start_link(args)
        num_of_persons = 1
        start_persons(num_of_persons)
        IO.inspect PersonManager.count_children, label: "The Persons started in main method are"
    end

    def start_persons(num_of_persons) when num_of_persons >=1 do
        PersonManager.add_node(private_key, public_key, 0)
        start_persons(num_of_persons-1)    
    end
    # num_of_persons == 0 case handled
end   

Ниже приведен динамический супервизор

defmodule PersonManager do
  use DynamicSupervisor

  def start_link(args) do
    DynamicSupervisor.start_link(__MODULE__, :ok, name: __MODULE__)
  end

  def add_node(private_key, public_key, num_of_coins) do
    # The code flow does come here, used inspect to check
    child_spec = {Person, {private_key, public_key, num_of_coins}}
    DynamicSupervisor.start_child(__MODULE__, child_spec)
    #The code flow does come here, used inpect to check.   
  end
end

Ниже приводится GenServers, который рабочий должен создать

defmodule Person do
  use GenServer

  def init({private_key, public_key, address, num_of_coins}) do
    IO.puts "Starting a person"
    {:ok, %{private_key: private_key, public_key: public_key, num_of_coins: num_of_coins}}
  end

  def start_link({private_key, public_key, address, num_of_coins}) do
    IO.puts "Starting a person"
    GenServer.start_link(
      __MODULE__,
      {private_key, public_key, address, num_of_coins}
      name: {:global, "node:#{public_key}"},
    )
  end
end

Я ожидаю, что genserver будет работать, но я получаю The Persons started in main method are: %{active: 0, specs: 0, supervisors: 0, workers: 0}

Я не знаю, почему супервизор не может запустить genserver.Любая помощь будет хорошей.ТИА

1 Ответ

0 голосов
/ 14 декабря 2018

В вашем случае вам нужно будет сделать три вещи:

Правильно настроить дочернюю спецификацию перед ее запуском с DynamicSupervisor:

child_spec = {Person, [private_key, public_key, num_of_coins]}

Принудительно заставить ребенка начатьпотерпите неудачу, если не удалось:

case DynamicSupervisor.start_child do
  {:ok, _pid} -> {}
  {:error, {:already_started, _pid}} -> {}
  error -> raise error
end

И убедитесь, что дочерний процесс может обрабатывать точные аргументы, которые ему дает супервизор:

defmodule Person do
  use GenServer

  def start_link([], private_key, public_key, num_of_coins) do
    GenServer.start_link(
      __MODULE__,
      {private_key, public_key, num_of_coins},
      name: {:global, "node:#{public_key}"}
    )
  end

  def init({private_key, public_key, num_of_coins}) do
    {:ok, %{private_key: private_key, public_key: public_key, num_of_coins: num_of_coins}}
  end
end
...