2023-04-10 15:36:59 +00:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
|
2015-06-03 11:35:48 +00:00
|
|
|
package nomad
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net"
|
2015-07-24 00:30:07 +00:00
|
|
|
"reflect"
|
2015-06-03 11:35:48 +00:00
|
|
|
"testing"
|
|
|
|
|
2017-09-07 23:56:15 +00:00
|
|
|
version "github.com/hashicorp/go-version"
|
2022-03-15 12:42:43 +00:00
|
|
|
"github.com/hashicorp/nomad/ci"
|
2017-09-29 16:58:48 +00:00
|
|
|
"github.com/hashicorp/nomad/helper/uuid"
|
2015-06-03 11:35:48 +00:00
|
|
|
"github.com/hashicorp/serf/serf"
|
2019-06-03 18:30:27 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
2015-06-03 11:35:48 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestIsNomadServer(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2015-06-03 11:35:48 +00:00
|
|
|
m := serf.Member{
|
2017-09-07 23:56:15 +00:00
|
|
|
Name: "foo",
|
|
|
|
Addr: net.IP([]byte{127, 0, 0, 1}),
|
|
|
|
Status: serf.StatusAlive,
|
2015-06-03 11:35:48 +00:00
|
|
|
Tags: map[string]string{
|
2017-12-18 21:16:23 +00:00
|
|
|
"role": "nomad",
|
|
|
|
"region": "aws",
|
|
|
|
"dc": "east-aws",
|
|
|
|
"rpc_addr": "1.1.1.1",
|
|
|
|
"port": "10000",
|
2022-03-24 18:44:21 +00:00
|
|
|
"vsn": "1",
|
2017-12-18 21:16:23 +00:00
|
|
|
"raft_vsn": "2",
|
|
|
|
"build": "0.7.0+ent",
|
2018-09-20 00:13:37 +00:00
|
|
|
"nonvoter": "1",
|
2015-06-03 11:35:48 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
valid, parts := isNomadServer(m)
|
|
|
|
if !valid || parts.Region != "aws" ||
|
|
|
|
parts.Datacenter != "east-aws" || parts.Port != 10000 {
|
|
|
|
t.Fatalf("bad: %v %v", valid, parts)
|
|
|
|
}
|
|
|
|
if parts.Name != "foo" {
|
|
|
|
t.Fatalf("bad: %v", parts)
|
|
|
|
}
|
|
|
|
if parts.Bootstrap {
|
|
|
|
t.Fatalf("unexpected bootstrap")
|
|
|
|
}
|
|
|
|
if parts.Expect != 0 {
|
|
|
|
t.Fatalf("bad: %v", parts.Expect)
|
|
|
|
}
|
2017-09-07 23:56:15 +00:00
|
|
|
if parts.Status != serf.StatusAlive {
|
|
|
|
t.Fatalf("bad: %v", parts.Status)
|
|
|
|
}
|
2017-12-18 21:16:23 +00:00
|
|
|
if parts.RaftVersion != 2 {
|
|
|
|
t.Fatalf("bad: %v", parts.RaftVersion)
|
|
|
|
}
|
|
|
|
if parts.RPCAddr.String() != "1.1.1.1:10000" {
|
|
|
|
t.Fatalf("bad: %v", parts.RPCAddr.String())
|
|
|
|
}
|
2022-03-24 18:44:21 +00:00
|
|
|
require.Equal(t, 1, parts.MajorVersion)
|
2017-09-07 23:56:15 +00:00
|
|
|
if seg := parts.Build.Segments(); len(seg) != 3 {
|
|
|
|
t.Fatalf("bad: %v", parts.Build)
|
|
|
|
} else if seg[0] != 0 && seg[1] != 7 && seg[2] != 0 {
|
|
|
|
t.Fatalf("bad: %v", parts.Build)
|
|
|
|
}
|
2018-09-20 00:13:37 +00:00
|
|
|
if !parts.NonVoter {
|
|
|
|
t.Fatalf("should be nonvoter")
|
|
|
|
}
|
2015-06-03 11:35:48 +00:00
|
|
|
|
|
|
|
m.Tags["bootstrap"] = "1"
|
|
|
|
valid, parts = isNomadServer(m)
|
|
|
|
if !valid || !parts.Bootstrap {
|
|
|
|
t.Fatalf("expected bootstrap")
|
|
|
|
}
|
|
|
|
if parts.Addr.String() != "127.0.0.1:10000" {
|
|
|
|
t.Fatalf("bad addr: %v", parts.Addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
m.Tags["expect"] = "3"
|
|
|
|
delete(m.Tags, "bootstrap")
|
|
|
|
valid, parts = isNomadServer(m)
|
|
|
|
if !valid || parts.Expect != 3 {
|
|
|
|
t.Fatalf("bad: %v", parts.Expect)
|
|
|
|
}
|
2018-09-20 00:13:37 +00:00
|
|
|
|
|
|
|
delete(m.Tags, "nonvoter")
|
|
|
|
valid, parts = isNomadServer(m)
|
|
|
|
if !valid || parts.NonVoter {
|
|
|
|
t.Fatalf("should be a voter")
|
|
|
|
}
|
2015-06-03 11:35:48 +00:00
|
|
|
}
|
2015-06-05 22:14:08 +00:00
|
|
|
|
2019-03-05 21:41:41 +00:00
|
|
|
func TestServersMeetMinimumVersionExcludingFailed(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2017-09-07 23:56:15 +00:00
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
members []serf.Member
|
|
|
|
ver *version.Version
|
|
|
|
expected bool
|
|
|
|
}{
|
|
|
|
// One server, meets reqs
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
2019-03-05 21:41:41 +00:00
|
|
|
makeMember("0.7.5", serf.StatusAlive),
|
2017-09-07 23:56:15 +00:00
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("0.7.5")),
|
|
|
|
expected: true,
|
|
|
|
},
|
|
|
|
// One server in dev, meets reqs
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
2019-03-05 21:41:41 +00:00
|
|
|
makeMember("0.8.5-dev", serf.StatusAlive),
|
2017-09-07 23:56:15 +00:00
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("0.7.5")),
|
|
|
|
expected: true,
|
|
|
|
},
|
|
|
|
// One server with meta, meets reqs
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
2019-03-05 21:41:41 +00:00
|
|
|
makeMember("0.7.5+ent", serf.StatusAlive),
|
2017-09-07 23:56:15 +00:00
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("0.7.5")),
|
|
|
|
expected: true,
|
|
|
|
},
|
|
|
|
// One server, doesn't meet reqs
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
2019-03-05 21:41:41 +00:00
|
|
|
makeMember("0.7.5", serf.StatusAlive),
|
2017-09-07 23:56:15 +00:00
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("0.8.0")),
|
|
|
|
expected: false,
|
|
|
|
},
|
2019-03-05 21:41:41 +00:00
|
|
|
// Multiple servers, meets req version, includes failed that doesn't meet req
|
2017-09-07 23:56:15 +00:00
|
|
|
{
|
|
|
|
members: []serf.Member{
|
2019-03-05 21:41:41 +00:00
|
|
|
makeMember("0.7.5", serf.StatusAlive),
|
|
|
|
makeMember("0.8.0", serf.StatusAlive),
|
|
|
|
makeMember("0.7.0", serf.StatusFailed),
|
2017-09-07 23:56:15 +00:00
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("0.7.5")),
|
|
|
|
expected: true,
|
|
|
|
},
|
|
|
|
// Multiple servers, doesn't meet req version
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
2019-03-05 21:41:41 +00:00
|
|
|
makeMember("0.7.5", serf.StatusAlive),
|
|
|
|
makeMember("0.8.0", serf.StatusAlive),
|
2017-09-07 23:56:15 +00:00
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("0.8.0")),
|
|
|
|
expected: false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range cases {
|
2022-10-17 20:23:51 +00:00
|
|
|
result := ServersMeetMinimumVersion(tc.members, AllRegions, tc.ver, false)
|
2017-09-07 23:56:15 +00:00
|
|
|
if result != tc.expected {
|
|
|
|
t.Fatalf("bad: %v, %v, %v", result, tc.ver.String(), tc)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-05 21:41:41 +00:00
|
|
|
func TestServersMeetMinimumVersionIncludingFailed(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2019-03-05 21:41:41 +00:00
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
members []serf.Member
|
|
|
|
ver *version.Version
|
|
|
|
expected bool
|
|
|
|
}{
|
|
|
|
// Multiple servers, meets req version
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
|
|
|
makeMember("0.7.5", serf.StatusAlive),
|
|
|
|
makeMember("0.8.0", serf.StatusAlive),
|
|
|
|
makeMember("0.7.5", serf.StatusFailed),
|
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("0.7.5")),
|
|
|
|
expected: true,
|
|
|
|
},
|
|
|
|
// Multiple servers, doesn't meet req version
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
|
|
|
makeMember("0.7.5", serf.StatusAlive),
|
|
|
|
makeMember("0.8.0", serf.StatusAlive),
|
|
|
|
makeMember("0.7.0", serf.StatusFailed),
|
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("0.7.5")),
|
|
|
|
expected: false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range cases {
|
2022-10-17 20:23:51 +00:00
|
|
|
result := ServersMeetMinimumVersion(tc.members, AllRegions, tc.ver, true)
|
2019-03-05 21:41:41 +00:00
|
|
|
if result != tc.expected {
|
|
|
|
t.Fatalf("bad: %v, %v, %v", result, tc.ver.String(), tc)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-07 18:40:57 +00:00
|
|
|
func TestServersMeetMinimumVersionSuffix(t *testing.T) {
|
2023-08-25 13:48:06 +00:00
|
|
|
ci.Parallel(t)
|
2022-03-07 18:40:57 +00:00
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
members []serf.Member
|
|
|
|
ver *version.Version
|
|
|
|
expected bool
|
|
|
|
}{
|
|
|
|
// Multiple servers, meets req version
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
|
|
|
makeMember("1.3.0", serf.StatusAlive),
|
|
|
|
makeMember("1.2.6", serf.StatusAlive),
|
|
|
|
makeMember("1.2.6-dev", serf.StatusFailed),
|
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("1.2.6-dev")),
|
|
|
|
expected: true,
|
|
|
|
},
|
|
|
|
// Multiple servers, doesn't meet req version
|
|
|
|
{
|
|
|
|
members: []serf.Member{
|
|
|
|
makeMember("1.1.18", serf.StatusAlive),
|
|
|
|
makeMember("1.2.6-dev", serf.StatusAlive),
|
|
|
|
makeMember("1.0.11", serf.StatusFailed),
|
|
|
|
},
|
|
|
|
ver: version.Must(version.NewVersion("1.2.6-dev")),
|
|
|
|
expected: false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range cases {
|
2022-10-17 20:23:51 +00:00
|
|
|
result := ServersMeetMinimumVersion(tc.members, AllRegions, tc.ver, true)
|
2022-03-07 18:40:57 +00:00
|
|
|
if result != tc.expected {
|
|
|
|
t.Fatalf("bad: %v, %v, %v", result, tc.ver.String(), tc)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-05 21:41:41 +00:00
|
|
|
func makeMember(version string, status serf.MemberStatus) serf.Member {
|
|
|
|
return serf.Member{
|
|
|
|
Name: "foo",
|
|
|
|
Addr: net.IP([]byte{127, 0, 0, 1}),
|
|
|
|
Tags: map[string]string{
|
|
|
|
"role": "nomad",
|
|
|
|
"region": "aws",
|
|
|
|
"dc": "east-aws",
|
|
|
|
"port": "10000",
|
|
|
|
"build": version,
|
2022-03-24 18:44:21 +00:00
|
|
|
"vsn": "1",
|
2019-03-05 21:41:41 +00:00
|
|
|
},
|
|
|
|
Status: status,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-24 00:30:07 +00:00
|
|
|
func TestShuffleStrings(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2015-07-24 00:30:07 +00:00
|
|
|
// Generate input
|
|
|
|
inp := make([]string, 10)
|
|
|
|
for idx := range inp {
|
2017-09-29 16:58:48 +00:00
|
|
|
inp[idx] = uuid.Generate()
|
2015-07-24 00:30:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Copy the input
|
|
|
|
orig := make([]string, len(inp))
|
|
|
|
copy(orig, inp)
|
|
|
|
|
|
|
|
// Shuffle
|
|
|
|
shuffleStrings(inp)
|
|
|
|
|
|
|
|
// Ensure order is not the same
|
|
|
|
if reflect.DeepEqual(inp, orig) {
|
|
|
|
t.Fatalf("shuffle failed")
|
|
|
|
}
|
|
|
|
}
|
2015-08-05 00:13:40 +00:00
|
|
|
|
2019-06-03 18:30:27 +00:00
|
|
|
func Test_partitionAll(t *testing.T) {
|
|
|
|
xs := []string{"a", "b", "c", "d", "e", "f"}
|
|
|
|
// evenly divisible
|
|
|
|
require.Equal(t, [][]string{{"a", "b"}, {"c", "d"}, {"e", "f"}}, partitionAll(2, xs))
|
|
|
|
require.Equal(t, [][]string{{"a", "b", "c"}, {"d", "e", "f"}}, partitionAll(3, xs))
|
|
|
|
// whole thing fits int the last part
|
|
|
|
require.Equal(t, [][]string{{"a", "b", "c", "d", "e", "f"}}, partitionAll(7, xs))
|
|
|
|
// odd remainder
|
|
|
|
require.Equal(t, [][]string{{"a", "b", "c", "d"}, {"e", "f"}}, partitionAll(4, xs))
|
|
|
|
// zero size
|
|
|
|
require.Equal(t, [][]string{{"a", "b", "c", "d", "e", "f"}}, partitionAll(0, xs))
|
|
|
|
// one size
|
|
|
|
require.Equal(t, [][]string{{"a"}, {"b"}, {"c"}, {"d"}, {"e"}, {"f"}}, partitionAll(1, xs))
|
|
|
|
}
|
|
|
|
|
2015-08-05 00:13:40 +00:00
|
|
|
func TestMaxUint64(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2015-08-05 00:13:40 +00:00
|
|
|
if maxUint64(1, 2) != 2 {
|
|
|
|
t.Fatalf("bad")
|
|
|
|
}
|
|
|
|
if maxUint64(2, 2) != 2 {
|
|
|
|
t.Fatalf("bad")
|
|
|
|
}
|
|
|
|
if maxUint64(2, 1) != 2 {
|
|
|
|
t.Fatalf("bad")
|
|
|
|
}
|
|
|
|
}
|