mirror of
https://github.com/edgelesssys/constellation.git
synced 2024-10-01 01:36:09 -04:00
f8001efbc0
* Merge enforced and expected measurements * Update measurement generation to new format * Write expected measurements hex encoded by default * Allow hex or base64 encoded expected measurements * Allow hex or base64 encoded clusterID * Allow security upgrades to warnOnly flag * Upload signed measurements in JSON format * Fetch measurements either from JSON or YAML * Use yaml.v3 instead of yaml.v2 * Error on invalid enforced selection * Add placeholder measurements to config * Update e2e test to new measurement format Signed-off-by: Daniel Weiße <dw@edgeless.systems>
396 lines
8.7 KiB
Go
396 lines
8.7 KiB
Go
/*
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
package server
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"io"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/virtwrapper"
|
|
"github.com/edgelesssys/constellation/v2/internal/attestation/measurements"
|
|
"github.com/edgelesssys/constellation/v2/internal/cloud/metadata"
|
|
"github.com/edgelesssys/constellation/v2/internal/logger"
|
|
"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), "test", tc.connect)
|
|
|
|
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), "test", tc.connect)
|
|
|
|
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.VPCIP)
|
|
})
|
|
}
|
|
}
|
|
|
|
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), "test", tc.connect)
|
|
|
|
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), "test", &stubConnect{})
|
|
|
|
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, measurements.M{0: measurements.WithAllBytes(0xAA, false)}),
|
|
},
|
|
"incorrect method": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: defaultConnect,
|
|
message: mustMarshal(t, measurements.M{0: measurements.WithAllBytes(0xAA, false)}),
|
|
method: http.MethodGet,
|
|
wantErr: true,
|
|
},
|
|
"listAll error": {
|
|
remoteAddr: "192.0.100.1:1234",
|
|
connect: &stubConnect{
|
|
getNetworkErr: errors.New("error"),
|
|
},
|
|
message: mustMarshal(t, measurements.M{0: measurements.WithAllBytes(0xAA, false)}),
|
|
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, measurements.M{0: measurements.WithAllBytes(0xAA, false)}),
|
|
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), "test", tc.connect)
|
|
|
|
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)
|
|
})
|
|
}
|
|
}
|
|
|
|
func mustMarshal(t *testing.T, v any) 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
|
|
}
|