上章说到咱们要引入syn https://github.com/ostinelli/syn/html
看过文档,它并无直接提供{via, Module, Name} 相关的方法。咱们须要封装一下。git
Name暂时能够用id,若是有须要再调整github
之后有回调需求的话,恰好也能够作在那个模块里。app
在player_server_manager项目里增长{:syn, "1.4.0"} 运行命令获取依赖socket
修改application 方法,增长依赖分布式
def application do [applications: [:logger, :syn], mod: {PlayerServerManager, []}] end
而后在项目里增长PlayerRegistry模块测试
defmodule PlayerRegistry do def register_name(name, pid) when is_pid(pid) do case :syn.register(name, pid) do :ok -> :yes _ -> :no end end def unregister_name(name) do :syn.unregister(name) :ok end def whereis_name(name) do :syn.find_by_key(name) end def send(name, msg) do case whereis_name(name) do :undefined -> exit({:badarg, {name, msg}}) pid -> Kernel.send pid, msg pid end end end
参照:global 模块实现。spa
修改PlayerServer.start_link 以下rest
def via_tuple(id), do: {:via, PlayerRegistry, id} def start_link(player) do GenServer.start_link(__MODULE__, %{player: player, socket: nil}, name: via_tuple(player.base_info.id)) end
PlayerServerManager.start 修改,增长syn的初始化,code
def start(_type, _args) do import Supervisor.Spec, warn: false :syn.init() children = [ # Define workers and child supervisors to be supervised worker(PlayerServer, [], restart: :temporary), ] # See http://elixir-lang.org/docs/stable/elixir/Supervisor.html # for other strategies and supported options opts = [strategy: :simple_one_for_one, name: PlayerServerManager.Supervisor] Supervisor.start_link(children, opts) end
PlayerServerManagerTest增长测试代码
test "player_registry", %{player: player} do assert {:ok, pid} = PlayerServerManager.start_player_server(player) assert PlayerServer.gem(PlayerServer.via_tuple(player.base_info.id)) == 0 end
测试成功,ok。目前看起来还好。分布式的注册以及冲突处理留待之后再作测试了。
写到这,发现via_tuple要能接受player和id2种参数更好,方便点。