2016-08-30 02:09:57 +00:00
|
|
|
package agent
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2016-11-15 02:54:37 +00:00
|
|
|
"fmt"
|
2016-08-30 02:09:57 +00:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/hashicorp/consul/consul/structs"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestOperator_OperatorRaftConfiguration(t *testing.T) {
|
|
|
|
httpTest(t, func(srv *HTTPServer) {
|
|
|
|
body := bytes.NewBuffer(nil)
|
|
|
|
req, err := http.NewRequest("GET", "/v1/operator/raft/configuration", body)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp := httptest.NewRecorder()
|
|
|
|
obj, err := srv.OperatorRaftConfiguration(resp, req)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if resp.Code != 200 {
|
|
|
|
t.Fatalf("bad code: %d", resp.Code)
|
|
|
|
}
|
|
|
|
out, ok := obj.(structs.RaftConfigurationResponse)
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("unexpected: %T", obj)
|
|
|
|
}
|
2016-08-30 18:30:56 +00:00
|
|
|
if len(out.Servers) != 1 ||
|
|
|
|
!out.Servers[0].Leader ||
|
|
|
|
!out.Servers[0].Voter {
|
2016-08-30 02:09:57 +00:00
|
|
|
t.Fatalf("bad: %v", out)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOperator_OperatorRaftPeer(t *testing.T) {
|
|
|
|
httpTest(t, func(srv *HTTPServer) {
|
|
|
|
body := bytes.NewBuffer(nil)
|
|
|
|
req, err := http.NewRequest("DELETE", "/v1/operator/raft/peer?address=nope", body)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we get this error, it proves we sent the address all the
|
|
|
|
// way through.
|
|
|
|
resp := httptest.NewRecorder()
|
|
|
|
_, err = srv.OperatorRaftPeer(resp, req)
|
|
|
|
if err == nil || !strings.Contains(err.Error(),
|
|
|
|
"address \"nope\" was not found in the Raft configuration") {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2016-11-15 02:54:37 +00:00
|
|
|
|
|
|
|
func TestOperator_KeyringInstall(t *testing.T) {
|
|
|
|
oldKey := "H3/9gBxcKKRf45CaI2DlRg=="
|
|
|
|
newKey := "z90lFx3sZZLtTOkutXcwYg=="
|
|
|
|
configFunc := func(c *Config) {
|
|
|
|
c.EncryptKey = oldKey
|
|
|
|
}
|
|
|
|
httpTestWithConfig(t, func(srv *HTTPServer) {
|
|
|
|
body := bytes.NewBufferString(fmt.Sprintf("{\"Key\":\"%s\"}", newKey))
|
2016-11-22 23:24:40 +00:00
|
|
|
req, err := http.NewRequest("POST", "/v1/operator/keyring", body)
|
2016-11-15 02:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp := httptest.NewRecorder()
|
2016-11-22 23:24:40 +00:00
|
|
|
_, err = srv.OperatorKeyringEndpoint(resp, req)
|
2016-11-15 02:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
listResponse, err := srv.agent.ListKeys("")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2016-11-22 23:24:40 +00:00
|
|
|
if len(listResponse.Responses) != 2 {
|
|
|
|
t.Fatalf("bad: %d", len(listResponse.Responses))
|
|
|
|
}
|
2016-11-15 02:54:37 +00:00
|
|
|
|
|
|
|
for _, response := range listResponse.Responses {
|
|
|
|
count, ok := response.Keys[newKey]
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("bad: %v", response.Keys)
|
|
|
|
}
|
|
|
|
if count != response.NumNodes {
|
|
|
|
t.Fatalf("bad: %d, %d", count, response.NumNodes)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}, configFunc)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOperator_KeyringList(t *testing.T) {
|
|
|
|
key := "H3/9gBxcKKRf45CaI2DlRg=="
|
|
|
|
configFunc := func(c *Config) {
|
|
|
|
c.EncryptKey = key
|
|
|
|
}
|
|
|
|
httpTestWithConfig(t, func(srv *HTTPServer) {
|
2016-11-22 23:24:40 +00:00
|
|
|
req, err := http.NewRequest("GET", "/v1/operator/keyring", nil)
|
2016-11-15 02:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp := httptest.NewRecorder()
|
2016-11-22 23:24:40 +00:00
|
|
|
r, err := srv.OperatorKeyringEndpoint(resp, req)
|
2016-11-15 02:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
responses, ok := r.([]*structs.KeyringResponse)
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("err: %v", !ok)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that we get both a LAN and WAN response, and that they both only
|
|
|
|
// contain the original key
|
|
|
|
if len(responses) != 2 {
|
|
|
|
t.Fatalf("bad: %d", len(responses))
|
|
|
|
}
|
2016-11-22 23:24:40 +00:00
|
|
|
|
|
|
|
// WAN
|
|
|
|
if len(responses[0].Keys) != 1 {
|
|
|
|
t.Fatalf("bad: %d", len(responses[0].Keys))
|
|
|
|
}
|
|
|
|
if !responses[0].WAN {
|
|
|
|
t.Fatalf("bad: %v", responses[0].WAN)
|
|
|
|
}
|
|
|
|
if _, ok := responses[0].Keys[key]; !ok {
|
|
|
|
t.Fatalf("bad: %v", ok)
|
|
|
|
}
|
|
|
|
|
|
|
|
// LAN
|
|
|
|
if len(responses[1].Keys) != 1 {
|
|
|
|
t.Fatalf("bad: %d", len(responses[1].Keys))
|
|
|
|
}
|
|
|
|
if responses[1].WAN {
|
|
|
|
t.Fatalf("bad: %v", responses[1].WAN)
|
|
|
|
}
|
|
|
|
if _, ok := responses[1].Keys[key]; !ok {
|
|
|
|
t.Fatalf("bad: %v", ok)
|
2016-11-15 02:54:37 +00:00
|
|
|
}
|
|
|
|
}, configFunc)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOperator_KeyringRemove(t *testing.T) {
|
|
|
|
key := "H3/9gBxcKKRf45CaI2DlRg=="
|
|
|
|
tempKey := "z90lFx3sZZLtTOkutXcwYg=="
|
|
|
|
configFunc := func(c *Config) {
|
|
|
|
c.EncryptKey = key
|
|
|
|
}
|
|
|
|
httpTestWithConfig(t, func(srv *HTTPServer) {
|
|
|
|
_, err := srv.agent.InstallKey(tempKey, "")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the temp key is installed
|
|
|
|
list, err := srv.agent.ListKeys("")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
responses := list.Responses
|
|
|
|
if len(responses) != 2 {
|
|
|
|
t.Fatalf("bad: %d", len(responses))
|
|
|
|
}
|
|
|
|
for _, response := range responses {
|
|
|
|
if len(response.Keys) != 2 {
|
|
|
|
t.Fatalf("bad: %d", len(response.Keys))
|
|
|
|
}
|
|
|
|
if _, ok := response.Keys[tempKey]; !ok {
|
|
|
|
t.Fatalf("bad: %v", ok)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
body := bytes.NewBufferString(fmt.Sprintf("{\"Key\":\"%s\"}", tempKey))
|
2016-11-22 23:24:40 +00:00
|
|
|
req, err := http.NewRequest("DELETE", "/v1/operator/keyring", body)
|
2016-11-15 02:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp := httptest.NewRecorder()
|
2016-11-22 23:24:40 +00:00
|
|
|
_, err = srv.OperatorKeyringEndpoint(resp, req)
|
2016-11-15 02:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the temp key has been removed
|
|
|
|
list, err = srv.agent.ListKeys("")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
responses = list.Responses
|
|
|
|
if len(responses) != 2 {
|
|
|
|
t.Fatalf("bad: %d", len(responses))
|
|
|
|
}
|
|
|
|
for _, response := range responses {
|
|
|
|
if len(response.Keys) != 1 {
|
|
|
|
t.Fatalf("bad: %d", len(response.Keys))
|
|
|
|
}
|
|
|
|
if _, ok := response.Keys[tempKey]; ok {
|
|
|
|
t.Fatalf("bad: %v", ok)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}, configFunc)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOperator_KeyringUse(t *testing.T) {
|
|
|
|
oldKey := "H3/9gBxcKKRf45CaI2DlRg=="
|
|
|
|
newKey := "z90lFx3sZZLtTOkutXcwYg=="
|
|
|
|
configFunc := func(c *Config) {
|
|
|
|
c.EncryptKey = oldKey
|
|
|
|
}
|
|
|
|
httpTestWithConfig(t, func(srv *HTTPServer) {
|
|
|
|
if _, err := srv.agent.InstallKey(newKey, ""); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
body := bytes.NewBufferString(fmt.Sprintf("{\"Key\":\"%s\"}", newKey))
|
2016-11-22 23:24:40 +00:00
|
|
|
req, err := http.NewRequest("PUT", "/v1/operator/keyring", body)
|
2016-11-15 02:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp := httptest.NewRecorder()
|
2016-11-22 23:24:40 +00:00
|
|
|
_, err = srv.OperatorKeyringEndpoint(resp, req)
|
2016-11-15 02:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := srv.agent.RemoveKey(oldKey, ""); err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure only the new key remains
|
|
|
|
list, err := srv.agent.ListKeys("")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
responses := list.Responses
|
|
|
|
if len(responses) != 2 {
|
|
|
|
t.Fatalf("bad: %d", len(responses))
|
|
|
|
}
|
|
|
|
for _, response := range responses {
|
|
|
|
if len(response.Keys) != 1 {
|
|
|
|
t.Fatalf("bad: %d", len(response.Keys))
|
|
|
|
}
|
|
|
|
if _, ok := response.Keys[newKey]; !ok {
|
|
|
|
t.Fatalf("bad: %v", ok)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}, configFunc)
|
|
|
|
}
|