mirror of
https://github.com/edgelesssys/constellation.git
synced 2025-01-25 14:56:18 -05:00
4151d365fb
Signed-off-by: Daniel Weiße <dw@edgeless.systems>
395 lines
8.9 KiB
Go
395 lines
8.9 KiB
Go
package server
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"io"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/edgelesssys/constellation/hack/qemu-metadata-api/virtwrapper"
|
|
"github.com/edgelesssys/constellation/internal/cloud/metadata"
|
|
"github.com/edgelesssys/constellation/internal/file"
|
|
"github.com/edgelesssys/constellation/internal/logger"
|
|
"github.com/spf13/afero"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
"libvirt.org/go/libvirt"
|
|
)
|
|
|
|
func TestListAll(t *testing.T) {
|
|
someErr := errors.New("error")
|
|
|
|
testCases := map[string]struct {
|
|
wantErr bool
|
|
connect *stubConnect
|
|
}{
|
|
"success": {
|
|
connect: &stubConnect{
|
|
network: stubNetwork{
|
|
leases: []libvirt.NetworkDHCPLease{
|
|
{
|
|
IPaddr: "192.0.100.1",
|
|
Hostname: "control-plane-0",
|
|
},
|
|
{
|
|
IPaddr: "192.0.100.2",
|
|
Hostname: "control-plane-1",
|
|
},
|
|
{
|
|
IPaddr: "192.0.200.1",
|
|
Hostname: "worker-0",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
"LookupNetworkByName error": {
|
|
connect: &stubConnect{
|
|
getNetworkErr: someErr,
|
|
},
|
|
wantErr: true,
|
|
},
|
|
"GetDHCPLeases error": {
|
|
connect: &stubConnect{
|
|
network: stubNetwork{
|
|
getLeaseErr: someErr,
|
|
},
|
|
},
|
|
wantErr: true,
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
assert := assert.New(t)
|
|
|
|
server := New(logger.NewTest(t), tc.connect, file.Handler{})
|
|
|
|
res, err := server.listAll()
|
|
|
|
if tc.wantErr {
|
|
assert.Error(err)
|
|
return
|
|
}
|
|
assert.NoError(err)
|
|
assert.Len(tc.connect.network.leases, len(res))
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestListSelf(t *testing.T) {
|
|
someErr := errors.New("error")
|
|
|
|
testCases := map[string]struct {
|
|
remoteAddr string
|
|
connect *stubConnect
|
|
wantErr bool
|
|
}{
|
|
"success": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: &stubConnect{
|
|
network: stubNetwork{
|
|
leases: []libvirt.NetworkDHCPLease{
|
|
{
|
|
IPaddr: "192.0.100.1",
|
|
Hostname: "control-plane-0",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
"listAll error": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: &stubConnect{
|
|
getNetworkErr: someErr,
|
|
},
|
|
wantErr: true,
|
|
},
|
|
"remoteAddr error": {
|
|
remoteAddr: "",
|
|
connect: &stubConnect{
|
|
network: stubNetwork{
|
|
leases: []libvirt.NetworkDHCPLease{
|
|
{
|
|
IPaddr: "192.0.100.1",
|
|
Hostname: "control-plane-0",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
wantErr: true,
|
|
},
|
|
"peer not found": {
|
|
remoteAddr: "192.0.200.1:1234",
|
|
connect: &stubConnect{
|
|
network: stubNetwork{
|
|
leases: []libvirt.NetworkDHCPLease{
|
|
{
|
|
IPaddr: "192.0.100.1",
|
|
Hostname: "control-plane-0",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
wantErr: true,
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
assert := assert.New(t)
|
|
require := require.New(t)
|
|
|
|
server := New(logger.NewTest(t), tc.connect, file.Handler{})
|
|
|
|
req, err := http.NewRequestWithContext(context.Background(), http.MethodGet, "http://192.0.0.1/self", nil)
|
|
require.NoError(err)
|
|
req.RemoteAddr = tc.remoteAddr
|
|
|
|
w := httptest.NewRecorder()
|
|
server.listSelf(w, req)
|
|
|
|
if tc.wantErr {
|
|
assert.NotEqual(http.StatusOK, w.Code)
|
|
return
|
|
}
|
|
assert.Equal(http.StatusOK, w.Code)
|
|
metadataRaw, err := io.ReadAll(w.Body)
|
|
require.NoError(err)
|
|
|
|
var metadata metadata.InstanceMetadata
|
|
require.NoError(json.Unmarshal(metadataRaw, &metadata))
|
|
assert.Equal(tc.connect.network.leases[0].Hostname, metadata.Name)
|
|
assert.Equal(tc.connect.network.leases[0].IPaddr, metadata.PublicIP)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestListPeers(t *testing.T) {
|
|
testCases := map[string]struct {
|
|
remoteAddr string
|
|
connect *stubConnect
|
|
wantErr bool
|
|
}{
|
|
"success": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: &stubConnect{
|
|
network: stubNetwork{
|
|
leases: []libvirt.NetworkDHCPLease{
|
|
{
|
|
IPaddr: "192.0.100.1",
|
|
Hostname: "control-plane-0",
|
|
},
|
|
{
|
|
IPaddr: "192.0.200.1",
|
|
Hostname: "worker-0",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
"listAll error": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: &stubConnect{
|
|
getNetworkErr: errors.New("error"),
|
|
},
|
|
wantErr: true,
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
assert := assert.New(t)
|
|
require := require.New(t)
|
|
|
|
server := New(logger.NewTest(t), tc.connect, file.Handler{})
|
|
|
|
req, err := http.NewRequestWithContext(context.Background(), http.MethodGet, "http://192.0.0.1/peers", nil)
|
|
require.NoError(err)
|
|
req.RemoteAddr = tc.remoteAddr
|
|
|
|
w := httptest.NewRecorder()
|
|
server.listPeers(w, req)
|
|
|
|
if tc.wantErr {
|
|
assert.NotEqual(http.StatusOK, w.Code)
|
|
return
|
|
}
|
|
assert.Equal(http.StatusOK, w.Code)
|
|
metadataRaw, err := io.ReadAll(w.Body)
|
|
require.NoError(err)
|
|
|
|
var metadata []metadata.InstanceMetadata
|
|
require.NoError(json.Unmarshal(metadataRaw, &metadata))
|
|
assert.Len(metadata, len(tc.connect.network.leases))
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestPostLog(t *testing.T) {
|
|
testCases := map[string]struct {
|
|
remoteAddr string
|
|
message io.Reader
|
|
method string
|
|
wantErr bool
|
|
}{
|
|
"success": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
method: http.MethodPost,
|
|
message: strings.NewReader("test message"),
|
|
},
|
|
"no body": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
method: http.MethodPost,
|
|
message: nil,
|
|
wantErr: true,
|
|
},
|
|
"incorrect method": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
method: http.MethodGet,
|
|
message: strings.NewReader("test message"),
|
|
wantErr: true,
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
assert := assert.New(t)
|
|
require := require.New(t)
|
|
|
|
server := New(logger.NewTest(t), &stubConnect{}, file.NewHandler(afero.NewMemMapFs()))
|
|
|
|
req, err := http.NewRequestWithContext(context.Background(), tc.method, "http://192.0.0.1/logs", tc.message)
|
|
require.NoError(err)
|
|
req.RemoteAddr = tc.remoteAddr
|
|
|
|
w := httptest.NewRecorder()
|
|
server.postLog(w, req)
|
|
|
|
if tc.wantErr {
|
|
assert.NotEqual(http.StatusOK, w.Code)
|
|
} else {
|
|
assert.Equal(http.StatusOK, w.Code)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestExportPCRs(t *testing.T) {
|
|
defaultConnect := &stubConnect{
|
|
network: stubNetwork{
|
|
leases: []libvirt.NetworkDHCPLease{
|
|
{
|
|
IPaddr: "192.0.100.1",
|
|
Hostname: "control-plane-0",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
testCases := map[string]struct {
|
|
remoteAddr string
|
|
connect *stubConnect
|
|
message string
|
|
method string
|
|
wantErr bool
|
|
}{
|
|
"success": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: defaultConnect,
|
|
method: http.MethodPost,
|
|
message: mustMarshal(t, map[uint32][]byte{0: []byte("AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA")}),
|
|
},
|
|
"incorrect method": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: defaultConnect,
|
|
message: mustMarshal(t, map[uint32][]byte{0: []byte("AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA")}),
|
|
method: http.MethodGet,
|
|
wantErr: true,
|
|
},
|
|
"listAll error": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: &stubConnect{
|
|
getNetworkErr: errors.New("error"),
|
|
},
|
|
message: mustMarshal(t, map[uint32][]byte{0: []byte("AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA")}),
|
|
method: http.MethodPost,
|
|
wantErr: true,
|
|
},
|
|
"invalid message": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: defaultConnect,
|
|
method: http.MethodPost,
|
|
message: "message",
|
|
wantErr: true,
|
|
},
|
|
"invalid remote address": {
|
|
remoteAddr: "localhost",
|
|
connect: defaultConnect,
|
|
method: http.MethodPost,
|
|
message: mustMarshal(t, map[uint32][]byte{0: []byte("AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA")}),
|
|
wantErr: true,
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
assert := assert.New(t)
|
|
require := require.New(t)
|
|
|
|
file := file.NewHandler(afero.NewMemMapFs())
|
|
server := New(logger.NewTest(t), tc.connect, file)
|
|
|
|
req, err := http.NewRequestWithContext(context.Background(), tc.method, "http://192.0.0.1/pcrs", strings.NewReader(tc.message))
|
|
require.NoError(err)
|
|
req.RemoteAddr = tc.remoteAddr
|
|
|
|
w := httptest.NewRecorder()
|
|
server.exportPCRs(w, req)
|
|
|
|
if tc.wantErr {
|
|
assert.NotEqual(http.StatusOK, w.Code)
|
|
return
|
|
}
|
|
|
|
assert.Equal(http.StatusOK, w.Code)
|
|
output, err := file.Read(exportedPCRsDir + tc.connect.network.leases[0].Hostname + "_pcrs.json")
|
|
require.NoError(err)
|
|
assert.JSONEq(tc.message, string(output))
|
|
})
|
|
}
|
|
}
|
|
|
|
func mustMarshal(t *testing.T, v interface{}) string {
|
|
t.Helper()
|
|
b, err := json.Marshal(v)
|
|
require.NoError(t, err)
|
|
return string(b)
|
|
}
|
|
|
|
type stubConnect struct {
|
|
network stubNetwork
|
|
getNetworkErr error
|
|
}
|
|
|
|
func (c stubConnect) LookupNetworkByName(name string) (*virtwrapper.Network, error) {
|
|
return &virtwrapper.Network{Net: c.network}, c.getNetworkErr
|
|
}
|
|
|
|
type stubNetwork struct {
|
|
leases []libvirt.NetworkDHCPLease
|
|
getLeaseErr error
|
|
}
|
|
|
|
func (n stubNetwork) GetDHCPLeases() ([]libvirt.NetworkDHCPLease, error) {
|
|
return n.leases, n.getLeaseErr
|
|
}
|
|
|
|
func (n stubNetwork) Free() error {
|
|
return nil
|
|
}
|