mirror of https://github.com/hashicorp/consul
parent
d62743c565
commit
1333fa57a1
|
@ -5,6 +5,7 @@ import (
|
||||||
|
|
||||||
"github.com/hashicorp/consul/agent/metadata"
|
"github.com/hashicorp/consul/agent/metadata"
|
||||||
"github.com/hashicorp/consul/types"
|
"github.com/hashicorp/consul/types"
|
||||||
|
"github.com/hashicorp/go-version"
|
||||||
"github.com/hashicorp/serf/serf"
|
"github.com/hashicorp/serf/serf"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -18,6 +19,10 @@ type lanMergeDelegate struct {
|
||||||
segment string
|
segment string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// uniqueIDMinVersion is the lowest version where we insist that nodes
|
||||||
|
// have a unique ID.
|
||||||
|
var uniqueIDMinVersion = version.Must(version.NewVersion("0.8.5"))
|
||||||
|
|
||||||
func (md *lanMergeDelegate) NotifyMerge(members []*serf.Member) error {
|
func (md *lanMergeDelegate) NotifyMerge(members []*serf.Member) error {
|
||||||
nodeMap := make(map[types.NodeID]string)
|
nodeMap := make(map[types.NodeID]string)
|
||||||
for _, m := range members {
|
for _, m := range members {
|
||||||
|
@ -37,7 +42,16 @@ func (md *lanMergeDelegate) NotifyMerge(members []*serf.Member) error {
|
||||||
return fmt.Errorf("Member '%s' has conflicting node ID '%s' with member '%s'",
|
return fmt.Errorf("Member '%s' has conflicting node ID '%s' with member '%s'",
|
||||||
m.Name, nodeID, other)
|
m.Name, nodeID, other)
|
||||||
}
|
}
|
||||||
nodeMap[nodeID] = m.Name
|
|
||||||
|
// Only map nodes with a version that's newer than when
|
||||||
|
// we made host-based IDs opt-in, which helps prevent
|
||||||
|
// chaos when upgrading older clusters. See #3070 for
|
||||||
|
// more details.
|
||||||
|
if ver, err := metadata.Build(m); err == nil {
|
||||||
|
if ver.Compare(uniqueIDMinVersion) >= 0 {
|
||||||
|
nodeMap[nodeID] = m.Name
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ok, dc := isConsulNode(*m)
|
ok, dc := isConsulNode(*m)
|
||||||
|
|
|
@ -8,7 +8,7 @@ import (
|
||||||
"github.com/hashicorp/serf/serf"
|
"github.com/hashicorp/serf/serf"
|
||||||
)
|
)
|
||||||
|
|
||||||
func makeNode(dc, name, id string, server bool) *serf.Member {
|
func makeNode(dc, name, id string, server bool, build string) *serf.Member {
|
||||||
var role string
|
var role string
|
||||||
if server {
|
if server {
|
||||||
role = "consul"
|
role = "consul"
|
||||||
|
@ -23,7 +23,7 @@ func makeNode(dc, name, id string, server bool) *serf.Member {
|
||||||
"dc": dc,
|
"dc": dc,
|
||||||
"id": id,
|
"id": id,
|
||||||
"port": "8300",
|
"port": "8300",
|
||||||
"build": "0.7.5",
|
"build": build,
|
||||||
"vsn": "2",
|
"vsn": "2",
|
||||||
"vsn_max": "3",
|
"vsn_max": "3",
|
||||||
"vsn_min": "2",
|
"vsn_min": "2",
|
||||||
|
@ -43,7 +43,8 @@ func TestMerge_LAN(t *testing.T) {
|
||||||
makeNode("dc2",
|
makeNode("dc2",
|
||||||
"node1",
|
"node1",
|
||||||
"96430788-246f-4379-94ce-257f7429e340",
|
"96430788-246f-4379-94ce-257f7429e340",
|
||||||
false),
|
false,
|
||||||
|
"0.7.5"),
|
||||||
},
|
},
|
||||||
expect: "wrong datacenter",
|
expect: "wrong datacenter",
|
||||||
},
|
},
|
||||||
|
@ -53,7 +54,8 @@ func TestMerge_LAN(t *testing.T) {
|
||||||
makeNode("dc2",
|
makeNode("dc2",
|
||||||
"node1",
|
"node1",
|
||||||
"96430788-246f-4379-94ce-257f7429e340",
|
"96430788-246f-4379-94ce-257f7429e340",
|
||||||
true),
|
true,
|
||||||
|
"0.7.5"),
|
||||||
},
|
},
|
||||||
expect: "wrong datacenter",
|
expect: "wrong datacenter",
|
||||||
},
|
},
|
||||||
|
@ -63,7 +65,8 @@ func TestMerge_LAN(t *testing.T) {
|
||||||
makeNode("dc1",
|
makeNode("dc1",
|
||||||
"node1",
|
"node1",
|
||||||
"ee954a2f-80de-4b34-8780-97b942a50a99",
|
"ee954a2f-80de-4b34-8780-97b942a50a99",
|
||||||
true),
|
true,
|
||||||
|
"0.7.5"),
|
||||||
},
|
},
|
||||||
expect: "with this agent's ID",
|
expect: "with this agent's ID",
|
||||||
},
|
},
|
||||||
|
@ -73,11 +76,30 @@ func TestMerge_LAN(t *testing.T) {
|
||||||
makeNode("dc1",
|
makeNode("dc1",
|
||||||
"node1",
|
"node1",
|
||||||
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
||||||
true),
|
true,
|
||||||
|
"0.8.5"),
|
||||||
makeNode("dc1",
|
makeNode("dc1",
|
||||||
"node2",
|
"node2",
|
||||||
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
||||||
true),
|
true,
|
||||||
|
"0.9.0"),
|
||||||
|
},
|
||||||
|
expect: "with member",
|
||||||
|
},
|
||||||
|
// Cluster with existing conflicting node IDs, but version is
|
||||||
|
// old enough to skip the check.
|
||||||
|
{
|
||||||
|
members: []*serf.Member{
|
||||||
|
makeNode("dc1",
|
||||||
|
"node1",
|
||||||
|
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
||||||
|
true,
|
||||||
|
"0.8.5"),
|
||||||
|
makeNode("dc1",
|
||||||
|
"node2",
|
||||||
|
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
||||||
|
true,
|
||||||
|
"0.8.4"),
|
||||||
},
|
},
|
||||||
expect: "with member",
|
expect: "with member",
|
||||||
},
|
},
|
||||||
|
@ -87,11 +109,13 @@ func TestMerge_LAN(t *testing.T) {
|
||||||
makeNode("dc1",
|
makeNode("dc1",
|
||||||
"node1",
|
"node1",
|
||||||
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
||||||
true),
|
true,
|
||||||
|
"0.8.5"),
|
||||||
makeNode("dc1",
|
makeNode("dc1",
|
||||||
"node2",
|
"node2",
|
||||||
"cda916bc-a357-4a19-b886-59419fcee50c",
|
"cda916bc-a357-4a19-b886-59419fcee50c",
|
||||||
true),
|
true,
|
||||||
|
"0.8.5"),
|
||||||
},
|
},
|
||||||
expect: "",
|
expect: "",
|
||||||
},
|
},
|
||||||
|
@ -128,7 +152,8 @@ func TestMerge_WAN(t *testing.T) {
|
||||||
makeNode("dc2",
|
makeNode("dc2",
|
||||||
"node1",
|
"node1",
|
||||||
"96430788-246f-4379-94ce-257f7429e340",
|
"96430788-246f-4379-94ce-257f7429e340",
|
||||||
false),
|
false,
|
||||||
|
"0.7.5"),
|
||||||
},
|
},
|
||||||
expect: "not a server",
|
expect: "not a server",
|
||||||
},
|
},
|
||||||
|
@ -138,11 +163,13 @@ func TestMerge_WAN(t *testing.T) {
|
||||||
makeNode("dc2",
|
makeNode("dc2",
|
||||||
"node1",
|
"node1",
|
||||||
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
"6185913b-98d7-4441-bd8f-f7f7d854a4af",
|
||||||
true),
|
true,
|
||||||
|
"0.7.5"),
|
||||||
makeNode("dc3",
|
makeNode("dc3",
|
||||||
"node2",
|
"node2",
|
||||||
"cda916bc-a357-4a19-b886-59419fcee50c",
|
"cda916bc-a357-4a19-b886-59419fcee50c",
|
||||||
true),
|
true,
|
||||||
|
"0.7.5"),
|
||||||
},
|
},
|
||||||
expect: "",
|
expect: "",
|
||||||
},
|
},
|
||||||
|
|
|
@ -0,0 +1,12 @@
|
||||||
|
package metadata
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/hashicorp/go-version"
|
||||||
|
"github.com/hashicorp/serf/serf"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Build extracts the Consul version info for a member.
|
||||||
|
func Build(m *serf.Member) (*version.Version, error) {
|
||||||
|
str := versionFormat.FindString(m.Tags["build"])
|
||||||
|
return version.NewVersion(str)
|
||||||
|
}
|
|
@ -0,0 +1,75 @@
|
||||||
|
package metadata
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/hashicorp/go-version"
|
||||||
|
"github.com/hashicorp/serf/serf"
|
||||||
|
"github.com/pascaldekloe/goe/verify"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestBuild(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
desc string
|
||||||
|
m *serf.Member
|
||||||
|
ver *version.Version
|
||||||
|
err bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
"no version",
|
||||||
|
&serf.Member{},
|
||||||
|
nil,
|
||||||
|
true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"bad version",
|
||||||
|
&serf.Member{
|
||||||
|
Tags: map[string]string{
|
||||||
|
"build": "nope",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
nil,
|
||||||
|
true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"good version",
|
||||||
|
&serf.Member{
|
||||||
|
Tags: map[string]string{
|
||||||
|
"build": "0.8.5",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
version.Must(version.NewVersion("0.8.5")),
|
||||||
|
false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"rc version",
|
||||||
|
&serf.Member{
|
||||||
|
Tags: map[string]string{
|
||||||
|
"build": "0.9.3rc1:d62743c",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
version.Must(version.NewVersion("0.9.3")),
|
||||||
|
false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"ent version",
|
||||||
|
&serf.Member{
|
||||||
|
Tags: map[string]string{
|
||||||
|
"build": "0.9.3+ent:d62743c",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
version.Must(version.NewVersion("0.9.3")),
|
||||||
|
false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.desc, func(t *testing.T) {
|
||||||
|
ver, err := Build(tt.m)
|
||||||
|
gotErr := err != nil
|
||||||
|
if wantErr := tt.err; gotErr != wantErr {
|
||||||
|
t.Fatalf("got %v want %v", gotErr, wantErr)
|
||||||
|
}
|
||||||
|
verify.Values(t, "", ver, tt.ver)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,8 +1,3 @@
|
||||||
// Package agent provides a logical endpoint for Consul agents in the
|
|
||||||
// network. agent data originates from Serf gossip and is primarily used to
|
|
||||||
// communicate Consul server information. Gossiped information that ends up
|
|
||||||
// in Server contains the necessary metadata required for servers.Manager to
|
|
||||||
// select which server an RPC request should be routed to.
|
|
||||||
package metadata
|
package metadata
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
@ -116,7 +111,7 @@ func IsConsulServer(m serf.Member) (bool, *Server) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
build_version, err := version.NewVersion(versionFormat.FindString(m.Tags["build"]))
|
build_version, err := Build(&m)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue