package consul import ( "fmt" "github.com/hashicorp/consul/consul/agent" "github.com/hashicorp/consul/types" "github.com/hashicorp/serf/serf" ) // lanMergeDelegate is used to handle a cluster merge on the LAN gossip // ring. We check that the peers are in the same datacenter and abort the // merge if there is a mis-match. type lanMergeDelegate struct { dc string nodeID types.NodeID nodeName string } func (md *lanMergeDelegate) NotifyMerge(members []*serf.Member) error { nodeMap := make(map[types.NodeID]string) for _, m := range members { if rawID, ok := m.Tags["id"]; ok && rawID != "" { nodeID := types.NodeID(rawID) // See if there's another node that conflicts with us. if (nodeID == md.nodeID) && (m.Name != md.nodeName) { return fmt.Errorf("Member '%s' has conflicting node ID '%s' with this agent's ID", m.Name, nodeID) } // See if there are any two nodes that conflict with each // other. This lets us only do joins into a hygienic // cluster now that node IDs are critical for operation. if other, ok := nodeMap[nodeID]; ok { return fmt.Errorf("Member '%s' has conflicting node ID '%s' with member '%s'", m.Name, nodeID, other) } nodeMap[nodeID] = m.Name } ok, dc := isConsulNode(*m) if ok { if dc != md.dc { return fmt.Errorf("Member '%s' part of wrong datacenter '%s'", m.Name, dc) } continue } ok, parts := agent.IsConsulServer(*m) if ok && parts.Datacenter != md.dc { return fmt.Errorf("Member '%s' part of wrong datacenter '%s'", m.Name, parts.Datacenter) } } return nil } // wanMergeDelegate is used to handle a cluster merge on the WAN gossip // ring. We check that the peers are server nodes and abort the merge // otherwise. type wanMergeDelegate struct { } func (md *wanMergeDelegate) NotifyMerge(members []*serf.Member) error { for _, m := range members { ok, _ := agent.IsConsulServer(*m) if !ok { return fmt.Errorf("Member '%s' is not a server", m.Name) } } return nil }