/* 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/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 }