Files
nomad/command/agent/secure_variable_endpoint_test.go
Charlie Voiselle ba74aadb90 Secure Variables: Variables - State store, FSM, RPC (#13098)
* Secure Variables: State Store
* Secure Variables: FSM
* Secure Variables: RPC
* Secure Variables: HTTP API

Co-authored-by: Tim Gross <tgross@hashicorp.com>
2022-07-11 13:34:04 -04:00

268 lines
7.6 KiB
Go

package agent
import (
"bytes"
"encoding/json"
"io"
"io/ioutil"
"net/http"
"net/http/httptest"
"testing"
"github.com/hashicorp/nomad/nomad/mock"
"github.com/hashicorp/nomad/nomad/structs"
"github.com/stretchr/testify/require"
)
var (
cb = func(c *Config) {
var ns int
ns = 0
c.LogLevel = "ERROR"
c.Server.NumSchedulers = &ns
}
)
func TestHTTP_SecureVariableList(t *testing.T) {
//ci.Parallel(t)
httpTest(t, cb, func(s *TestAgent) {
// Test the empty list case
req, err := http.NewRequest("GET", "/v1/vars", nil)
require.NoError(t, err)
respW := httptest.NewRecorder()
// Make the request
obj, err := s.Server.SecureVariablesListRequest(respW, req)
require.NoError(t, err)
// add vars and test a populated backend
svMap := mock.SecureVariables(4, 4)
svs := svMap.List()
svs[3].Path = svs[0].Path + "/child"
for _, sv := range svs {
require.NoError(t, rpcWriteSV(s, sv))
}
// Make the HTTP request
req, err = http.NewRequest("GET", "/v1/vars", nil)
require.NoError(t, err)
respW = httptest.NewRecorder()
// Make the request
obj, err = s.Server.SecureVariablesListRequest(respW, req)
require.NoError(t, err)
// Check for the index
require.NotZero(t, respW.HeaderMap.Get("X-Nomad-Index"))
require.Equal(t, "true", respW.HeaderMap.Get("X-Nomad-KnownLeader"))
require.NotZero(t, respW.HeaderMap.Get("X-Nomad-LastContact"))
// Check the output (the 3 we register )
require.Len(t, obj.([]*structs.SecureVariableStub), 4)
// test prefix query
req, err = http.NewRequest("GET", "/v1/vars?prefix="+svs[0].Path, nil)
require.NoError(t, err)
respW = httptest.NewRecorder()
// Make the request
obj, err = s.Server.SecureVariablesListRequest(respW, req)
require.NoError(t, err)
require.Len(t, obj.([]*structs.SecureVariableStub), 2)
})
}
func TestHTTP_SecureVariableQuery(t *testing.T) {
//ci.Parallel(t)
httpTest(t, cb, func(s *TestAgent) {
// Make a request for a non-existing variable
req, err := http.NewRequest("GET", "/v1/var/does/not/exist", nil)
require.NoError(t, err)
respW := httptest.NewRecorder()
obj, err := s.Server.SecureVariableSpecificRequest(respW, req)
require.EqualError(t, err, "Secure variable not found")
// Don't pass a path
req, err = http.NewRequest("GET", "/v1/var/", nil)
require.NoError(t, err)
respW = httptest.NewRecorder()
obj, err = s.Server.SecureVariableSpecificRequest(respW, req)
require.EqualError(t, err, "Missing secure variable path")
// Use an incorrect verb
req, err = http.NewRequest("LOLWUT", "/v1/var/foo", nil)
require.NoError(t, err)
respW = httptest.NewRecorder()
obj, err = s.Server.SecureVariableSpecificRequest(respW, req)
require.EqualError(t, err, ErrInvalidMethod)
// Use RPC to make a test variable
sv1 := mock.SecureVariable()
require.NoError(t, rpcWriteSV(s, sv1))
// Query a variable
req, err = http.NewRequest("GET", "/v1/var/"+sv1.Path, nil)
require.NoError(t, err)
respW = httptest.NewRecorder()
// Make the request
obj, err = s.Server.SecureVariableSpecificRequest(respW, req)
require.NoError(t, err)
// Check for the index
require.NotZero(t, respW.HeaderMap.Get("X-Nomad-Index"))
require.Equal(t, "true", respW.HeaderMap.Get("X-Nomad-KnownLeader"))
require.NotZero(t, respW.HeaderMap.Get("X-Nomad-LastContact"))
// Check the output
require.Equal(t, sv1.Path, obj.(*structs.SecureVariable).Path)
})
}
func TestHTTP_SecureVariableCreate(t *testing.T) {
//ci.Parallel(t)
httpTest(t, cb, func(s *TestAgent) {
sv1 := mock.SecureVariable()
sv1U := sv1.Copy()
require.NoError(t, rpcWriteSV(s, sv1))
// Make a change for update
sv1U.UnencryptedData["newness"] = "awwyeah"
// Make the HTTP request
buf := encodeReq(&sv1U)
req, err := http.NewRequest("PUT", "/v1/var/"+sv1.Path, buf)
require.NoError(t, err)
respW := httptest.NewRecorder()
// Make the request
obj, err := s.Server.SecureVariableSpecificRequest(respW, req)
require.NoError(t, err)
require.Nil(t, obj)
// Check for the index
require.NotZero(t, respW.HeaderMap.Get("X-Nomad-Index"))
// Check the variable was created
out, err := rpcReadSV(s, sv1.Namespace, sv1.Path)
require.NoError(t, err)
require.NotNil(t, out)
sv1.CreateIndex, sv1.ModifyIndex = out.CreateIndex, out.ModifyIndex
require.Equal(t, sv1.Path, out.Path)
require.NotEqual(t, sv1, out)
require.Contains(t, out.UnencryptedData, "newness")
// break the request body
badBuf := encodeBrokenReq(&sv1U)
req, err = http.NewRequest("PUT", "/v1/var/"+sv1.Path, badBuf)
require.NoError(t, err)
respW = httptest.NewRecorder()
// Make the request
obj, err = s.Server.SecureVariableSpecificRequest(respW, req)
require.EqualError(t, err, "unexpected EOF")
var cErr HTTPCodedError
require.ErrorAs(t, err, &cErr)
require.Equal(t, http.StatusBadRequest, cErr.Code())
})
}
func TestHTTP_SecureVariableUpdate(t *testing.T) {
//ci.Parallel(t)
httpTest(t, cb, func(s *TestAgent) {
// Make the HTTP request
sv1 := mock.SecureVariable()
sv1.EncryptedData = nil
buf := encodeReq(sv1)
req, err := http.NewRequest("PUT", "/v1/var/"+sv1.Path, buf)
require.NoError(t, err)
respW := httptest.NewRecorder()
// Make the request
obj, err := s.Server.SecureVariableSpecificRequest(respW, req)
require.NoError(t, err)
require.Nil(t, obj)
// Check for the index
require.NotZero(t, respW.HeaderMap.Get("X-Nomad-Index"))
// Check the variable was updated
out, err := rpcReadSV(s, sv1.Namespace, sv1.Path)
require.NoError(t, err)
require.NotNil(t, out)
sv1.CreateIndex, sv1.ModifyIndex = out.CreateIndex, out.ModifyIndex
sv1.CreateTime, sv1.ModifyTime = out.CreateTime, out.ModifyTime
require.Equal(t, sv1.Path, out.Path)
require.Equal(t, sv1, out)
})
}
func TestHTTP_SecureVariableDelete(t *testing.T) {
//ci.Parallel(t)
httpTest(t, cb, func(s *TestAgent) {
sv1 := mock.SecureVariable()
require.NoError(t, rpcWriteSV(s, sv1))
// Make the HTTP request
req, err := http.NewRequest("DELETE", "/v1/var/"+sv1.Path, nil)
require.NoError(t, err)
respW := httptest.NewRecorder()
// Make the request
obj, err := s.Server.SecureVariableSpecificRequest(respW, req)
require.NoError(t, err)
require.Nil(t, obj)
// Check for the index
require.NotZero(t, respW.HeaderMap.Get("X-Nomad-Index"))
// Check variable was deleted
sv, err := rpcReadSV(s, sv1.Namespace, sv1.Path)
require.NoError(t, err)
require.Nil(t, sv)
})
}
func encodeBrokenReq(obj interface{}) io.ReadCloser {
// var buf *bytes.Buffer
// enc := json.NewEncoder(buf)
// enc.Encode(obj)
b, _ := json.Marshal(obj)
b = b[0 : len(b)-5] // strip newline and final }
return ioutil.NopCloser(bytes.NewReader(b))
}
func rpcReadSV(s *TestAgent, ns, p string) (*structs.SecureVariable, error) {
checkArgs := structs.SecureVariablesReadRequest{Path: p, QueryOptions: structs.QueryOptions{Namespace: ns, Region: "global"}}
var checkResp structs.SecureVariablesReadResponse
err := s.Agent.RPC(structs.SecureVariablesReadRPCMethod, &checkArgs, &checkResp)
return checkResp.Data, err
}
func rpcWriteSV(s *TestAgent, sv *structs.SecureVariable) error {
args := structs.SecureVariablesUpsertRequest{
Data: []*structs.SecureVariable{sv},
WriteRequest: structs.WriteRequest{Namespace: sv.Namespace, Region: "global"},
}
var resp structs.SecureVariablesUpsertResponse
err := s.Agent.RPC(structs.SecureVariablesUpsertRPCMethod, &args, &resp)
if err != nil {
return err
}
nv, err := rpcReadSV(s, sv.Namespace, sv.Path)
if err != nil {
return err
}
sv.CreateIndex = nv.CreateIndex
sv.CreateTime = nv.CreateTime
sv.ModifyIndex = nv.ModifyIndex
sv.ModifyTime = nv.ModifyTime
return nil
}