2018-01-15 22:48:53 +00:00
|
|
|
package nomad
|
|
|
|
|
|
|
|
import (
|
2018-02-07 19:04:30 +00:00
|
|
|
"net"
|
2019-09-04 12:45:25 +00:00
|
|
|
"strings"
|
2018-01-15 22:48:53 +00:00
|
|
|
"testing"
|
|
|
|
|
2018-02-06 21:03:09 +00:00
|
|
|
"github.com/hashicorp/nomad/client"
|
|
|
|
"github.com/hashicorp/nomad/client/config"
|
2018-01-15 22:48:53 +00:00
|
|
|
"github.com/hashicorp/nomad/helper/uuid"
|
2018-01-27 01:05:38 +00:00
|
|
|
"github.com/hashicorp/nomad/nomad/structs"
|
2018-01-15 22:48:53 +00:00
|
|
|
"github.com/hashicorp/nomad/testutil"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
2018-02-07 19:04:30 +00:00
|
|
|
type namedConnWrapper struct {
|
|
|
|
net.Conn
|
|
|
|
name string
|
|
|
|
}
|
|
|
|
|
|
|
|
type namedAddr string
|
|
|
|
|
|
|
|
func (n namedAddr) String() string { return string(n) }
|
|
|
|
func (n namedAddr) Network() string { return string(n) }
|
|
|
|
|
|
|
|
func (n namedConnWrapper) LocalAddr() net.Addr {
|
|
|
|
return namedAddr(n.name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestServer_removeNodeConn_differentAddrs(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
|
|
|
|
s1, cleanupS1 := TestServer(t, nil)
|
|
|
|
defer cleanupS1()
|
2018-02-07 19:04:30 +00:00
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
p1, p2 := net.Pipe()
|
|
|
|
w1 := namedConnWrapper{
|
|
|
|
Conn: p1,
|
|
|
|
name: "a",
|
|
|
|
}
|
|
|
|
w2 := namedConnWrapper{
|
|
|
|
Conn: p2,
|
|
|
|
name: "b",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the connections
|
|
|
|
nodeID := uuid.Generate()
|
|
|
|
ctx1 := &RPCContext{
|
|
|
|
Conn: w1,
|
|
|
|
NodeID: nodeID,
|
|
|
|
}
|
|
|
|
ctx2 := &RPCContext{
|
|
|
|
Conn: w2,
|
|
|
|
NodeID: nodeID,
|
|
|
|
}
|
|
|
|
|
|
|
|
s1.addNodeConn(ctx1)
|
|
|
|
s1.addNodeConn(ctx2)
|
|
|
|
require.Len(s1.connectedNodes(), 1)
|
2018-04-26 20:28:24 +00:00
|
|
|
require.Len(s1.nodeConns[nodeID], 2)
|
2018-02-07 19:04:30 +00:00
|
|
|
|
2018-02-14 21:05:33 +00:00
|
|
|
// Check that the value is the second conn.
|
|
|
|
state, ok := s1.getNodeConn(nodeID)
|
|
|
|
require.True(ok)
|
|
|
|
require.Equal(state.Ctx.Conn.LocalAddr().String(), w2.name)
|
|
|
|
|
2018-02-07 19:04:30 +00:00
|
|
|
// Delete the first
|
|
|
|
s1.removeNodeConn(ctx1)
|
|
|
|
require.Len(s1.connectedNodes(), 1)
|
2018-04-26 20:28:24 +00:00
|
|
|
require.Len(s1.nodeConns[nodeID], 1)
|
2018-02-07 19:04:30 +00:00
|
|
|
|
2018-02-14 21:05:33 +00:00
|
|
|
// Check that the value is the second conn.
|
|
|
|
state, ok = s1.getNodeConn(nodeID)
|
|
|
|
require.True(ok)
|
|
|
|
require.Equal(state.Ctx.Conn.LocalAddr().String(), w2.name)
|
|
|
|
|
2018-02-07 19:04:30 +00:00
|
|
|
// Delete the second
|
|
|
|
s1.removeNodeConn(ctx2)
|
|
|
|
require.Len(s1.connectedNodes(), 0)
|
2018-04-27 17:16:03 +00:00
|
|
|
|
|
|
|
_, ok = s1.getNodeConn(nodeID)
|
|
|
|
require.False(ok)
|
2018-02-07 19:04:30 +00:00
|
|
|
}
|
|
|
|
|
2018-01-15 22:48:53 +00:00
|
|
|
func TestServerWithNodeConn_NoPath(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
|
2020-03-02 15:29:24 +00:00
|
|
|
s1, cleanupS1 := TestServer(t, func(c *Config) {
|
|
|
|
c.BootstrapExpect = 2
|
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS1()
|
|
|
|
s2, cleanupS2 := TestServer(t, func(c *Config) {
|
2020-03-02 15:29:24 +00:00
|
|
|
c.BootstrapExpect = 2
|
2018-01-15 22:48:53 +00:00
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS2()
|
2018-01-15 22:48:53 +00:00
|
|
|
TestJoin(t, s1, s2)
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
testutil.WaitForLeader(t, s2.RPC)
|
|
|
|
|
|
|
|
nodeID := uuid.Generate()
|
2018-01-30 06:01:42 +00:00
|
|
|
srv, err := s1.serverWithNodeConn(nodeID, s1.Region())
|
2018-01-15 22:48:53 +00:00
|
|
|
require.Nil(srv)
|
2018-01-27 01:05:38 +00:00
|
|
|
require.EqualError(err, structs.ErrNoNodeConn.Error())
|
2018-01-15 22:48:53 +00:00
|
|
|
}
|
|
|
|
|
2018-01-30 06:01:42 +00:00
|
|
|
func TestServerWithNodeConn_NoPath_Region(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
|
|
|
|
s1, cleanupS1 := TestServer(t, nil)
|
|
|
|
defer cleanupS1()
|
2018-01-30 06:01:42 +00:00
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
nodeID := uuid.Generate()
|
|
|
|
srv, err := s1.serverWithNodeConn(nodeID, "fake-region")
|
|
|
|
require.Nil(srv)
|
|
|
|
require.EqualError(err, structs.ErrNoRegionPath.Error())
|
|
|
|
}
|
|
|
|
|
2018-01-15 22:48:53 +00:00
|
|
|
func TestServerWithNodeConn_Path(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
|
2020-03-02 15:29:24 +00:00
|
|
|
s1, cleanupS1 := TestServer(t, func(c *Config) {
|
|
|
|
c.BootstrapExpect = 2
|
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS1()
|
|
|
|
s2, cleanupS2 := TestServer(t, func(c *Config) {
|
2020-03-02 15:29:24 +00:00
|
|
|
c.BootstrapExpect = 2
|
2018-01-15 22:48:53 +00:00
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS2()
|
2018-01-15 22:48:53 +00:00
|
|
|
TestJoin(t, s1, s2)
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
testutil.WaitForLeader(t, s2.RPC)
|
|
|
|
|
|
|
|
// Create a fake connection for the node on server 2
|
|
|
|
nodeID := uuid.Generate()
|
|
|
|
s2.addNodeConn(&RPCContext{
|
|
|
|
NodeID: nodeID,
|
|
|
|
})
|
|
|
|
|
2018-01-30 06:01:42 +00:00
|
|
|
srv, err := s1.serverWithNodeConn(nodeID, s1.Region())
|
|
|
|
require.NotNil(srv)
|
|
|
|
require.Equal(srv.Addr.String(), s2.config.RPCAddr.String())
|
|
|
|
require.Nil(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestServerWithNodeConn_Path_Region(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
|
|
|
|
s1, cleanupS1 := TestServer(t, nil)
|
|
|
|
defer cleanupS1()
|
|
|
|
s2, cleanupS2 := TestServer(t, func(c *Config) {
|
2018-01-30 06:01:42 +00:00
|
|
|
c.Region = "two"
|
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS2()
|
2018-01-30 06:01:42 +00:00
|
|
|
TestJoin(t, s1, s2)
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
testutil.WaitForLeader(t, s2.RPC)
|
|
|
|
|
|
|
|
// Create a fake connection for the node on server 2
|
|
|
|
nodeID := uuid.Generate()
|
|
|
|
s2.addNodeConn(&RPCContext{
|
|
|
|
NodeID: nodeID,
|
|
|
|
})
|
|
|
|
|
|
|
|
srv, err := s1.serverWithNodeConn(nodeID, s2.Region())
|
2018-01-15 22:48:53 +00:00
|
|
|
require.NotNil(srv)
|
|
|
|
require.Equal(srv.Addr.String(), s2.config.RPCAddr.String())
|
|
|
|
require.Nil(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestServerWithNodeConn_Path_Newest(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
|
2020-03-02 15:29:24 +00:00
|
|
|
s1, cleanupS1 := TestServer(t, func(c *Config) {
|
|
|
|
c.BootstrapExpect = 3
|
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS1()
|
|
|
|
s2, cleanupS2 := TestServer(t, func(c *Config) {
|
2020-03-02 15:29:24 +00:00
|
|
|
c.BootstrapExpect = 3
|
2018-01-15 22:48:53 +00:00
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS2()
|
|
|
|
s3, cleanupS3 := TestServer(t, func(c *Config) {
|
2020-03-02 15:29:24 +00:00
|
|
|
c.BootstrapExpect = 3
|
2018-01-15 22:48:53 +00:00
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS3()
|
2018-01-15 22:48:53 +00:00
|
|
|
TestJoin(t, s1, s2, s3)
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
testutil.WaitForLeader(t, s2.RPC)
|
|
|
|
testutil.WaitForLeader(t, s3.RPC)
|
|
|
|
|
|
|
|
// Create a fake connection for the node on server 2 and 3
|
|
|
|
nodeID := uuid.Generate()
|
|
|
|
s2.addNodeConn(&RPCContext{
|
|
|
|
NodeID: nodeID,
|
|
|
|
})
|
|
|
|
s3.addNodeConn(&RPCContext{
|
|
|
|
NodeID: nodeID,
|
|
|
|
})
|
|
|
|
|
2018-01-30 06:01:42 +00:00
|
|
|
srv, err := s1.serverWithNodeConn(nodeID, s1.Region())
|
2018-01-15 22:48:53 +00:00
|
|
|
require.NotNil(srv)
|
|
|
|
require.Equal(srv.Addr.String(), s3.config.RPCAddr.String())
|
|
|
|
require.Nil(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestServerWithNodeConn_PathAndErr(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
|
2020-03-02 15:29:24 +00:00
|
|
|
s1, cleanupS1 := TestServer(t, func(c *Config) {
|
|
|
|
c.BootstrapExpect = 3
|
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS1()
|
|
|
|
s2, cleanupS2 := TestServer(t, func(c *Config) {
|
2020-03-02 15:29:24 +00:00
|
|
|
c.BootstrapExpect = 3
|
2018-01-15 22:48:53 +00:00
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS2()
|
|
|
|
s3, cleanupS3 := TestServer(t, func(c *Config) {
|
2020-03-02 15:29:24 +00:00
|
|
|
c.BootstrapExpect = 3
|
2018-01-15 22:48:53 +00:00
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS3()
|
2018-01-15 22:48:53 +00:00
|
|
|
TestJoin(t, s1, s2, s3)
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
testutil.WaitForLeader(t, s2.RPC)
|
|
|
|
testutil.WaitForLeader(t, s3.RPC)
|
|
|
|
|
|
|
|
// Create a fake connection for the node on server 2
|
|
|
|
nodeID := uuid.Generate()
|
|
|
|
s2.addNodeConn(&RPCContext{
|
|
|
|
NodeID: nodeID,
|
|
|
|
})
|
|
|
|
|
|
|
|
// Shutdown the RPC layer for server 3
|
|
|
|
s3.rpcListener.Close()
|
|
|
|
|
2018-01-30 06:01:42 +00:00
|
|
|
srv, err := s1.serverWithNodeConn(nodeID, s1.Region())
|
2018-01-15 22:48:53 +00:00
|
|
|
require.NotNil(srv)
|
|
|
|
require.Equal(srv.Addr.String(), s2.config.RPCAddr.String())
|
|
|
|
require.Nil(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestServerWithNodeConn_NoPathAndErr(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
|
2020-03-02 15:29:24 +00:00
|
|
|
s1, cleanupS1 := TestServer(t, func(c *Config) {
|
|
|
|
c.BootstrapExpect = 3
|
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS1()
|
|
|
|
s2, cleanupS2 := TestServer(t, func(c *Config) {
|
2020-03-02 15:29:24 +00:00
|
|
|
c.BootstrapExpect = 3
|
2018-01-15 22:48:53 +00:00
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS2()
|
|
|
|
s3, cleanupS3 := TestServer(t, func(c *Config) {
|
2020-03-02 15:29:24 +00:00
|
|
|
c.BootstrapExpect = 3
|
2018-01-15 22:48:53 +00:00
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupS3()
|
2018-01-15 22:48:53 +00:00
|
|
|
TestJoin(t, s1, s2, s3)
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
testutil.WaitForLeader(t, s2.RPC)
|
|
|
|
testutil.WaitForLeader(t, s3.RPC)
|
|
|
|
|
|
|
|
// Shutdown the RPC layer for server 3
|
|
|
|
s3.rpcListener.Close()
|
|
|
|
|
2018-01-30 06:01:42 +00:00
|
|
|
srv, err := s1.serverWithNodeConn(uuid.Generate(), s1.Region())
|
2018-01-15 22:48:53 +00:00
|
|
|
require.Nil(srv)
|
|
|
|
require.NotNil(err)
|
2019-09-04 12:45:25 +00:00
|
|
|
|
|
|
|
// the exact error seems to be dependent on timing and raft protocol version
|
|
|
|
if !strings.Contains(err.Error(), "failed querying") && !strings.Contains(err.Error(), "No path to node") {
|
|
|
|
require.Contains(err.Error(), "failed querying")
|
|
|
|
}
|
2018-01-15 22:48:53 +00:00
|
|
|
}
|
2018-02-06 21:03:09 +00:00
|
|
|
|
|
|
|
func TestNodeStreamingRpc_badEndpoint(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
require := require.New(t)
|
2019-12-04 00:15:11 +00:00
|
|
|
s1, cleanupS1 := TestServer(t, nil)
|
|
|
|
defer cleanupS1()
|
2018-02-06 21:03:09 +00:00
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
2019-12-04 00:15:11 +00:00
|
|
|
c, cleanupC := client.TestClient(t, func(c *config.Config) {
|
2018-02-06 21:03:09 +00:00
|
|
|
c.Servers = []string{s1.config.RPCAddr.String()}
|
|
|
|
})
|
2019-12-04 00:15:11 +00:00
|
|
|
defer cleanupC()
|
2018-02-06 21:03:09 +00:00
|
|
|
|
|
|
|
// Wait for the client to connect
|
|
|
|
testutil.WaitForResult(func() (bool, error) {
|
|
|
|
nodes := s1.connectedNodes()
|
|
|
|
return len(nodes) == 1, nil
|
|
|
|
}, func(err error) {
|
|
|
|
t.Fatalf("should have a clients")
|
|
|
|
})
|
|
|
|
|
|
|
|
state, ok := s1.getNodeConn(c.NodeID())
|
|
|
|
require.True(ok)
|
|
|
|
|
|
|
|
conn, err := NodeStreamingRpc(state.Session, "Bogus")
|
|
|
|
require.Nil(conn)
|
|
|
|
require.NotNil(err)
|
2018-02-15 23:29:31 +00:00
|
|
|
require.Contains(err.Error(), "Bogus")
|
|
|
|
require.True(structs.IsErrUnknownMethod(err))
|
2018-02-06 21:03:09 +00:00
|
|
|
}
|