From e5a270a172ad1fa5bf80666c0c57029b4f11ecc5 Mon Sep 17 00:00:00 2001 From: Scott Bennett Date: Mon, 30 Jul 2018 14:16:43 -0700 Subject: [PATCH] reuse port on certain OS's some OS's (like OS X) require the REUSE_PORT flag to be set in order to bind multiple listeners to the same UDP port --- lib/strategy/gossip.ex | 26 +++++++++++++++++++++++--- 1 file changed, 23 insertions(+), 3 deletions(-) diff --git a/lib/strategy/gossip.ex b/lib/strategy/gossip.ex index aa843c6..0e29e6f 100644 --- a/lib/strategy/gossip.ex +++ b/lib/strategy/gossip.ex @@ -51,6 +51,8 @@ defmodule Cluster.Strategy.Gossip do @default_port 45892 @default_addr {0, 0, 0, 0} @default_multicast_addr {230, 1, 1, 251} + @sol_socket 0xFFFF + @so_reuseport 0x0200 def start_link(args) do GenServer.start_link(__MODULE__, args) @@ -71,8 +73,8 @@ defmodule Cluster.Strategy.Gossip do |> Keyword.get(:multicast_addr, @default_multicast_addr) |> sanitize_ip() - {:ok, socket} = - :gen_udp.open(port, [ + options = + [ :binary, active: true, ip: ip, @@ -81,13 +83,31 @@ defmodule Cluster.Strategy.Gossip do multicast_ttl: ttl, multicast_loop: true, add_membership: {multicast_addr, {0, 0, 0, 0}} - ]) + ] ++ reuse_port() + + {:ok, socket} = :gen_udp.open(port, options) secret = Keyword.get(config, :secret, nil) state = %State{state | :meta => {multicast_addr, port, socket, secret}} {:ok, state, 0} end + defp reuse_port() do + case :os.type() do + {:unix, os_name} -> + cond do + os_name in [:darwin, :freebsd, :openbsd, :netbsd] -> + [{:raw, @sol_socket, @so_reuseport, <<1::native-32>>}] + + true -> + [] + end + + _ -> + [] + end + end + defp sanitize_ip(input) do case input do {_a, _b, _c, _d} = ip ->