diff --git a/command/agent/agent.go b/command/agent/agent.go
index f43bcac696..827ccd2c40 100644
--- a/command/agent/agent.go
+++ b/command/agent/agent.go
@@ -376,6 +376,14 @@ func (a *Agent) consulConfig() (*consul.Config, error) {
if a.config.ReconnectTimeoutWan != 0 {
base.SerfWANConfig.ReconnectTimeout = a.config.ReconnectTimeoutWan
}
+ if a.config.EncryptVerifyIncoming != nil {
+ base.SerfWANConfig.MemberlistConfig.GossipVerifyIncoming = *a.config.EncryptVerifyIncoming
+ base.SerfLANConfig.MemberlistConfig.GossipVerifyIncoming = *a.config.EncryptVerifyIncoming
+ }
+ if a.config.EncryptVerifyOutgoing != nil {
+ base.SerfWANConfig.MemberlistConfig.GossipVerifyOutgoing = *a.config.EncryptVerifyOutgoing
+ base.SerfLANConfig.MemberlistConfig.GossipVerifyOutgoing = *a.config.EncryptVerifyOutgoing
+ }
if a.config.AdvertiseAddrs.RPC != nil {
base.RPCAdvertise = a.config.AdvertiseAddrs.RPC
}
diff --git a/command/agent/config.go b/command/agent/config.go
index 16069919d8..05fa770e73 100644
--- a/command/agent/config.go
+++ b/command/agent/config.go
@@ -365,6 +365,12 @@ type Config struct {
// Encryption key to use for the Serf communication
EncryptKey string `mapstructure:"encrypt" json:"-"`
+ // EncryptVerifyIncoming and EncryptVerifyOutgoing are used to enforce
+ // incoming/outgoing gossip encryption and can be used to upshift to
+ // encrypted gossip on a running cluster.
+ EncryptVerifyIncoming *bool `mapstructure:"encrypt_verify_incoming"`
+ EncryptVerifyOutgoing *bool `mapstructure:"encrypt_verify_outgoing"`
+
// LogLevel is the level of the logs to putout
LogLevel string `mapstructure:"log_level"`
@@ -864,6 +870,9 @@ func DefaultConfig() *Config {
RetryIntervalWan: 30 * time.Second,
TLSMinVersion: "tls10",
+
+ EncryptVerifyIncoming: Bool(true),
+ EncryptVerifyOutgoing: Bool(true),
}
}
@@ -1477,6 +1486,12 @@ func MergeConfig(a, b *Config) *Config {
if b.EncryptKey != "" {
result.EncryptKey = b.EncryptKey
}
+ if b.EncryptVerifyIncoming != nil {
+ result.EncryptVerifyIncoming = b.EncryptVerifyIncoming
+ }
+ if b.EncryptVerifyOutgoing != nil {
+ result.EncryptVerifyOutgoing = b.EncryptVerifyOutgoing
+ }
if b.LogLevel != "" {
result.LogLevel = b.LogLevel
}
diff --git a/command/agent/config_test.go b/command/agent/config_test.go
index d0413524b7..2d64264f07 100644
--- a/command/agent/config_test.go
+++ b/command/agent/config_test.go
@@ -123,6 +123,18 @@ func TestDecodeConfig(t *testing.T) {
t.Fatalf("bad: %#v", config)
}
+ input = `{"encrypt_verify_incoming":true, "encrypt_verify_outgoing":true}`
+ config, err = DecodeConfig(bytes.NewReader([]byte(input)))
+ if err != nil {
+ t.Fatalf("err: %s", err)
+ }
+ if config.EncryptVerifyIncoming == nil || !*config.EncryptVerifyIncoming {
+ t.Fatalf("bad: %#v", config)
+ }
+ if config.EncryptVerifyOutgoing == nil || !*config.EncryptVerifyOutgoing {
+ t.Fatalf("bad: %#v", config)
+ }
+
// DNS setup
input = `{"ports": {"dns": 8500}, "recursors": ["8.8.8.8","8.8.4.4"], "recursor":"127.0.0.1", "domain": "foobar"}`
config, err = DecodeConfig(bytes.NewReader([]byte(input)))
diff --git a/vendor/github.com/hashicorp/memberlist/config.go b/vendor/github.com/hashicorp/memberlist/config.go
index 2f43d14cb1..5cad4ed545 100644
--- a/vendor/github.com/hashicorp/memberlist/config.go
+++ b/vendor/github.com/hashicorp/memberlist/config.go
@@ -141,6 +141,16 @@ type Config struct {
GossipNodes int
GossipToTheDeadTime time.Duration
+ // GossipVerifyIncoming controls whether to enforce encryption for incoming
+ // gossip. It is used for upshifting from unencrypted to encrypted gossip on
+ // a running cluster.
+ GossipVerifyIncoming bool
+
+ // GossipVerifyOutgoing controls whether to enforce encryption for outgoing
+ // gossip. It is used for upshifting from unencrypted to encrypted gossip on
+ // a running cluster.
+ GossipVerifyOutgoing bool
+
// EnableCompression is used to control message compression. This can
// be used to reduce bandwidth usage at the cost of slightly more CPU
// utilization. This is only available starting at protocol version 1.
@@ -233,9 +243,11 @@ func DefaultLANConfig() *Config {
DisableTcpPings: false, // TCP pings are safe, even with mixed versions
AwarenessMaxMultiplier: 8, // Probe interval backs off to 8 seconds
- GossipNodes: 3, // Gossip to 3 nodes
- GossipInterval: 200 * time.Millisecond, // Gossip more rapidly
- GossipToTheDeadTime: 30 * time.Second, // Same as push/pull
+ GossipNodes: 3, // Gossip to 3 nodes
+ GossipInterval: 200 * time.Millisecond, // Gossip more rapidly
+ GossipToTheDeadTime: 30 * time.Second, // Same as push/pull
+ GossipVerifyIncoming: true,
+ GossipVerifyOutgoing: true,
EnableCompression: true, // Enable compression by default
diff --git a/vendor/github.com/hashicorp/memberlist/memberlist.go b/vendor/github.com/hashicorp/memberlist/memberlist.go
index 2aba22322d..e4b0d7347d 100644
--- a/vendor/github.com/hashicorp/memberlist/memberlist.go
+++ b/vendor/github.com/hashicorp/memberlist/memberlist.go
@@ -334,7 +334,7 @@ func (m *Memberlist) setAlive() error {
addr, port, err := m.transport.FinalAdvertiseAddr(
m.config.AdvertiseAddr, m.config.AdvertisePort)
if err != nil {
- return fmt.Errorf("Failed to get final advertise address: %v")
+ return fmt.Errorf("Failed to get final advertise address: %v", err)
}
// Check if this is a public address without encryption
diff --git a/vendor/github.com/hashicorp/memberlist/net.go b/vendor/github.com/hashicorp/memberlist/net.go
index e0036d01d6..65a60159d1 100644
--- a/vendor/github.com/hashicorp/memberlist/net.go
+++ b/vendor/github.com/hashicorp/memberlist/net.go
@@ -283,8 +283,13 @@ func (m *Memberlist) ingestPacket(buf []byte, from net.Addr, timestamp time.Time
// Decrypt the payload
plain, err := decryptPayload(m.config.Keyring.GetKeys(), buf, nil)
if err != nil {
- m.logger.Printf("[ERR] memberlist: Decrypt packet failed: %v %s", err, LogAddress(from))
- return
+ if !m.config.GossipVerifyIncoming {
+ // Treat the message as plaintext
+ plain = buf
+ } else {
+ m.logger.Printf("[ERR] memberlist: Decrypt packet failed: %v %s", err, LogAddress(from))
+ return
+ }
}
// Continue processing the plaintext buffer
@@ -557,7 +562,7 @@ func (m *Memberlist) encodeAndSendMsg(addr string, msgType messageType, msg inte
func (m *Memberlist) sendMsg(addr string, msg []byte) error {
// Check if we can piggy back any messages
bytesAvail := m.config.UDPBufferSize - len(msg) - compoundHeaderOverhead
- if m.config.EncryptionEnabled() {
+ if m.config.EncryptionEnabled() && m.config.GossipVerifyOutgoing {
bytesAvail -= encryptOverhead(m.encryptionVersion())
}
extra := m.getBroadcasts(compoundOverhead, bytesAvail)
@@ -621,7 +626,7 @@ func (m *Memberlist) rawSendMsgPacket(addr string, node *Node, msg []byte) error
}
// Check if we have encryption enabled
- if m.config.EncryptionEnabled() {
+ if m.config.EncryptionEnabled() && m.config.GossipVerifyOutgoing {
// Encrypt the payload
var buf bytes.Buffer
primaryKey := m.config.Keyring.GetPrimaryKey()
@@ -652,7 +657,7 @@ func (m *Memberlist) rawSendMsgStream(conn net.Conn, sendBuf []byte) error {
}
// Check if encryption is enabled
- if m.config.EncryptionEnabled() {
+ if m.config.EncryptionEnabled() && m.config.GossipVerifyOutgoing {
crypt, err := m.encryptLocalState(sendBuf)
if err != nil {
m.logger.Printf("[ERROR] memberlist: Failed to encrypt local state: %v", err)
@@ -876,7 +881,7 @@ func (m *Memberlist) readStream(conn net.Conn) (messageType, io.Reader, *codec.D
// Reset message type and bufConn
msgType = messageType(plain[0])
bufConn = bytes.NewReader(plain[1:])
- } else if m.config.EncryptionEnabled() {
+ } else if m.config.EncryptionEnabled() && m.config.GossipVerifyIncoming {
return 0, nil, nil,
fmt.Errorf("Encryption is configured but remote state is not encrypted")
}
diff --git a/vendor/github.com/hashicorp/memberlist/state.go b/vendor/github.com/hashicorp/memberlist/state.go
index 71bf6f34d2..8513361b1a 100644
--- a/vendor/github.com/hashicorp/memberlist/state.go
+++ b/vendor/github.com/hashicorp/memberlist/state.go
@@ -40,6 +40,11 @@ func (n *Node) Address() string {
return joinHostPort(n.Addr.String(), n.Port)
}
+// String returns the node name
+func (n *Node) String() string {
+ return n.Name
+}
+
// NodeState is used to manage our state view of another node
type nodeState struct {
Node
diff --git a/vendor/github.com/hashicorp/memberlist/tag.sh b/vendor/github.com/hashicorp/memberlist/tag.sh
new file mode 100755
index 0000000000..cd16623a70
--- /dev/null
+++ b/vendor/github.com/hashicorp/memberlist/tag.sh
@@ -0,0 +1,16 @@
+#!/usr/bin/env bash
+set -e
+
+# The version must be supplied from the environment. Do not include the
+# leading "v".
+if [ -z $VERSION ]; then
+ echo "Please specify a version."
+ exit 1
+fi
+
+# Generate the tag.
+echo "==> Tagging version $VERSION..."
+git commit --allow-empty -a --gpg-sign=348FFC4C -m "Release v$VERSION"
+git tag -a -m "Version $VERSION" -s -u 348FFC4C "v${VERSION}" master
+
+exit 0
diff --git a/vendor/vendor.json b/vendor/vendor.json
index 770afdf7a6..8f2c0ce3a8 100644
--- a/vendor/vendor.json
+++ b/vendor/vendor.json
@@ -642,10 +642,10 @@
"revisionTime": "2015-06-09T07:04:31Z"
},
{
- "checksumSHA1": "JJsKjmgNTUTaEHEEAQgb9jCGGiM=",
+ "checksumSHA1": "AoIvQFHycqypYK57ZjiWzlQmdwk=",
"path": "github.com/hashicorp/memberlist",
- "revision": "6cc6075ba9fba1915fa0416f00d2b4efa9dc2262",
- "revisionTime": "2017-03-17T22:24:04Z"
+ "revision": "16fe34d996eba2b68f6f46f26c51c617c6bc1bf0",
+ "revisionTime": "2017-05-26T19:17:51Z"
},
{
"checksumSHA1": "qnlqWJYV81ENr61SZk9c65R1mDo=",
diff --git a/website/source/docs/agent/encryption.html.md b/website/source/docs/agent/encryption.html.md
index 6f182ee320..83e2db0a07 100644
--- a/website/source/docs/agent/encryption.html.md
+++ b/website/source/docs/agent/encryption.html.md
@@ -52,6 +52,24 @@ $ consul agent -data-dir=/tmp/consul -config-file=encrypt.json
All nodes within a Consul cluster must share the same encryption key in
order to send and receive cluster information.
+## Configuring Gossip Encryption on an existing cluster
+
+As of version 0.8.4, Consul supports upshifting to encrypted gossip on a running cluster
+through the following process.
+
+1. Generate an encryption key using [`consul keygen`](/docs/commands/keygen.html)
+2. Set the [`encrypt`](/docs/agent/options.html#_encrypt) key in the agent configuration and set
+[`encrypt_verify_incoming`](/docs/agent/options.html#encrypt_verify_incoming) and
+[`encrypt_verify_outgoing`](/docs/agent/options.html#encrypt_verify_outgoing) to `false`, doing a
+rolling update of the cluster with these new values. After this step, the agents will be able to
+decrypt gossip but will not yet be sending encrypted traffic.
+3. Remove the [`encrypt_verify_outgoing`](/docs/agent/options.html#encrypt_verify_outgoing) setting
+to change it back to false (the default) and perform another rolling update of the cluster. The
+agents will now be sending encrypted gossip but will still allow incoming unencrypted traffic.
+4. Remove the [`encrypt_verify_incoming`](/docs/agent/options.html#encrypt_verify_incoming) setting
+to change it back to false (the default) and perform a final rolling update of the cluster. All the
+agents will now be strictly enforcing encrypted gossip.
+
## RPC Encryption with TLS
Consul supports using TLS to verify the authenticity of servers and clients. To enable this,
diff --git a/website/source/docs/agent/options.html.md b/website/source/docs/agent/options.html.md
index 2961c2bd57..48416f9d82 100644
--- a/website/source/docs/agent/options.html.md
+++ b/website/source/docs/agent/options.html.md
@@ -708,6 +708,18 @@ Consul will not enable TLS for the HTTP API unless the `https` port has been ass
* `encrypt` Equivalent to the
[`-encrypt` command-line flag](#_encrypt).
+* `encrypt_verify_incoming` -
+ This is an optional parameter that can be used to disable enforcing encryption for incoming gossip in order
+ to upshift from unencrypted to encrypted gossip on a running cluster. See [this section]
+ (/docs/agent/encryption.html#configuring-gossip-encryption-on-an-existing-cluster) for more information.
+ Defaults to true.
+
+* `encrypt_verify_outgoing` -
+ This is an optional parameter that can be used to disable enforcing encryption for outgoing gossip in order
+ to upshift from unencrypted to encrypted gossip on a running cluster. See [this section]
+ (/docs/agent/encryption.html#configuring-gossip-encryption-on-an-existing-cluster) for more information.
+ Defaults to true.
+
* `key_file` This provides a the file path to a
PEM-encoded private key. The key is used with the certificate to verify the agent's authenticity.
This must be provided along with [`cert_file`](#cert_file).