metadata-api: support dnsmasq (#3115)

This commit is contained in:
3u13r 2024-05-24 15:14:16 +02:00 committed by GitHub
parent 37e46b98c2
commit 21c30ebb76
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
18 changed files with 304 additions and 270 deletions

View File

@ -13,8 +13,9 @@ go_library(
importpath = "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api", importpath = "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api",
visibility = ["//visibility:private"], visibility = ["//visibility:private"],
deps = [ deps = [
"//hack/qemu-metadata-api/dhcp/dnsmasq",
"//hack/qemu-metadata-api/dhcp/virtwrapper",
"//hack/qemu-metadata-api/server", "//hack/qemu-metadata-api/server",
"//hack/qemu-metadata-api/virtwrapper",
"//internal/logger", "//internal/logger",
"@org_libvirt_go_libvirt//:libvirt", "@org_libvirt_go_libvirt//:libvirt",
], ],

View File

@ -0,0 +1,8 @@
load("@io_bazel_rules_go//go:def.bzl", "go_library")
go_library(
name = "dhcp",
srcs = ["dhcp.go"],
importpath = "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp",
visibility = ["//visibility:public"],
)

View File

@ -4,9 +4,9 @@ Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only SPDX-License-Identifier: AGPL-3.0-only
*/ */
package virtwrapper package dhcp
// NetworkDHCPLease abstracts a libvirt DHCP lease. // NetworkDHCPLease abstracts a DHCP lease.
type NetworkDHCPLease struct { type NetworkDHCPLease struct {
IPaddr string IPaddr string
Hostname string Hostname string

View File

@ -0,0 +1,24 @@
load("@io_bazel_rules_go//go:def.bzl", "go_library")
load("//bazel/go:go_test.bzl", "go_test")
go_library(
name = "dnsmasq",
srcs = ["dnsmasq.go"],
importpath = "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp/dnsmasq",
visibility = ["//visibility:public"],
deps = [
"//hack/qemu-metadata-api/dhcp",
"@com_github_spf13_afero//:afero",
],
)
go_test(
name = "dnsmasq_test",
srcs = ["dnsmasq_test.go"],
embed = [":dnsmasq"],
deps = [
"@com_github_spf13_afero//:afero",
"@com_github_stretchr_testify//assert",
"@com_github_stretchr_testify//require",
],
)

View File

@ -0,0 +1,56 @@
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package dnsmasq
import (
"bufio"
"strings"
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp"
"github.com/spf13/afero"
)
// DNSMasq is a DHCP lease getter for dnsmasq.
type DNSMasq struct {
leasesFileName string
fs *afero.Afero
}
// New creates a new DNSMasq.
func New(leasesFileName string) *DNSMasq {
return &DNSMasq{
leasesFileName: leasesFileName,
fs: &afero.Afero{Fs: afero.NewOsFs()},
}
}
// GetDHCPLeases returns the underlying DHCP leases.
func (d *DNSMasq) GetDHCPLeases() ([]dhcp.NetworkDHCPLease, error) {
file, err := d.fs.Open(d.leasesFileName)
if err != nil {
return nil, err
}
defer file.Close()
// read file
var leases []dhcp.NetworkDHCPLease
scanner := bufio.NewScanner(file)
for scanner.Scan() {
line := scanner.Text()
// split by whitespace
fields := strings.Fields(line)
leases = append(leases, dhcp.NetworkDHCPLease{
IPaddr: fields[2],
Hostname: fields[3],
})
}
if err := scanner.Err(); err != nil {
return nil, err
}
return leases, nil
}

View File

@ -0,0 +1,40 @@
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package dnsmasq
import (
"testing"
"github.com/spf13/afero"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
func TestGetDHCPLeases(t *testing.T) {
require := require.New(t)
assert := assert.New(t)
fs := afero.NewMemMapFs()
leasesFileName := "dnsmasq.leases"
leasesFile, err := fs.Create(leasesFileName)
require.NoError(err)
_, err = leasesFile.WriteString("1716219737 52:54:af:a1:98:9f 10.42.2.1 worker0 ff:c2:72:f6:09:00:02:00:00:ab:11:18:fc:48:85:40:3f:bc:41\n")
require.NoError(err)
_, err = leasesFile.WriteString("1716219735 52:54:7f:8f:ba:91 10.42.1.1 controlplane0 ff:c2:72:f6:09:00:02:00:00:ab:11:21:7c:b5:14:ec:43:b7:43\n")
require.NoError(err)
leasesFile.Close()
d := DNSMasq{leasesFileName: leasesFileName, fs: &afero.Afero{Fs: fs}}
leases, err := d.GetDHCPLeases()
require.NoError(err)
assert.Len(leases, 2)
assert.Equal("10.42.2.1", leases[0].IPaddr)
assert.Equal("worker0", leases[0].Hostname)
assert.Equal("10.42.1.1", leases[1].IPaddr)
assert.Equal("controlplane0", leases[1].Hostname)
}

View File

@ -7,7 +7,10 @@ go_library(
"virtwrapper_cgo.go", "virtwrapper_cgo.go",
"virtwrapper_cross.go", "virtwrapper_cross.go",
], ],
importpath = "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/virtwrapper", importpath = "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp/virtwrapper",
visibility = ["//visibility:public"], visibility = ["//visibility:public"],
deps = ["@org_libvirt_go_libvirt//:libvirt"], deps = [
"//hack/qemu-metadata-api/dhcp",
"@org_libvirt_go_libvirt//:libvirt",
],
) )

View File

@ -0,0 +1,7 @@
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package virtwrapper

View File

@ -0,0 +1,60 @@
//go:build cgo
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package virtwrapper
import (
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp"
"libvirt.org/go/libvirt"
)
// Connect wraps a libvirt connection.
type Connect struct {
conn *libvirt.Connect
networkName string
}
// New creates a new libvirt Connct wrapper.
func New(conn *libvirt.Connect, networkName string) *Connect {
return &Connect{
conn: conn,
networkName: networkName,
}
}
// LookupNetworkByName looks up a network by name.
func (c *Connect) lookupNetworkByName(name string) (*libvirt.Network, error) {
net, err := c.conn.LookupNetworkByName(name)
if err != nil {
return nil, err
}
return net, nil
}
// GetDHCPLeases returns the underlying DHCP leases.
func (c *Connect) GetDHCPLeases() ([]dhcp.NetworkDHCPLease, error) {
net, err := c.lookupNetworkByName(c.networkName)
if err != nil {
return nil, err
}
defer net.Free()
leases, err := net.GetDHCPLeases()
if err != nil {
return nil, err
}
ret := make([]dhcp.NetworkDHCPLease, len(leases))
for i, l := range leases {
ret[i] = dhcp.NetworkDHCPLease{
IPaddr: l.IPaddr,
Hostname: l.Hostname,
}
}
return ret, nil
}

View File

@ -0,0 +1,24 @@
//go:build !cgo
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package virtwrapper
import (
"errors"
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp"
)
// Connect wraps a libvirt connection.
type Connect struct{}
// GetDHCPLeases returns the underlying DHCP leases.
// This function errors if CGO is disabled.
func (n *Connect) GetDHCPLeases() ([]dhcp.NetworkDHCPLease, error) {
return nil, errors.New("using virtwrapper requires building with CGO")
}

View File

@ -13,29 +13,38 @@ import (
"log/slog" "log/slog"
"os" "os"
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp/dnsmasq"
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp/virtwrapper"
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/server" "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/server"
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/virtwrapper"
"github.com/edgelesssys/constellation/v2/internal/logger" "github.com/edgelesssys/constellation/v2/internal/logger"
"libvirt.org/go/libvirt" "libvirt.org/go/libvirt"
) )
func main() { func main() {
bindPort := flag.String("port", "8080", "Port to bind to") bindPort := flag.String("port", "8080", "Port to bind to")
targetNetwork := flag.String("network", "constellation-network", "Name of the network in QEMU to use") targetNetwork := flag.String("network", "constellation-network", "Name of the network in libvirt")
libvirtURI := flag.String("libvirt-uri", "qemu:///system", "URI of the libvirt connection") libvirtURI := flag.String("libvirt-uri", "qemu:///system", "URI of the libvirt connection")
leasesFileName := flag.String("dnsmasq-leases", "", "Path to the dnsmasq leases file")
initSecretHash := flag.String("initsecrethash", "", "brcypt hash of the init secret") initSecretHash := flag.String("initsecrethash", "", "brcypt hash of the init secret")
flag.Parse() flag.Parse()
log := logger.NewJSONLogger(slog.LevelInfo) log := logger.NewJSONLogger(slog.LevelInfo)
conn, err := libvirt.NewConnect(*libvirtURI) var leaseGetter server.LeaseGetter
if err != nil { if *leasesFileName == "" {
log.With(slog.Any("error", err)).Error("Failed to connect to libvirt") conn, err := libvirt.NewConnect(*libvirtURI)
os.Exit(1) if err != nil {
log.With(slog.Any("error", err)).Error("Failed to connect to libvirt")
os.Exit(1)
}
defer conn.Close()
leaseGetter = virtwrapper.New(conn, *targetNetwork)
} else {
log.Info("Using dnsmasq leases file")
leaseGetter = dnsmasq.New(*leasesFileName)
} }
defer conn.Close()
serv := server.New(log, *targetNetwork, *initSecretHash, &virtwrapper.Connect{Conn: conn}) serv := server.New(log, *targetNetwork, *initSecretHash, leaseGetter)
if err := serv.ListenAndServe(*bindPort); err != nil { if err := serv.ListenAndServe(*bindPort); err != nil {
log.With(slog.Any("error", err)).Error("Failed to serve") log.With(slog.Any("error", err)).Error("Failed to serve")
os.Exit(1) os.Exit(1)

View File

@ -10,7 +10,7 @@ go_library(
], ],
visibility = ["//visibility:public"], visibility = ["//visibility:public"],
deps = [ deps = [
"//hack/qemu-metadata-api/virtwrapper", "//hack/qemu-metadata-api/dhcp",
"//internal/cloud/metadata", "//internal/cloud/metadata",
"//internal/role", "//internal/role",
], ],
@ -18,22 +18,17 @@ go_library(
go_test( go_test(
name = "server_test", name = "server_test",
srcs = [ srcs = ["server_test.go"],
"server_cgo_test.go",
"server_cross_test.go",
"server_test.go",
],
embed = [":server"], embed = [":server"],
# keep # keep
pure = "on", pure = "on",
# keep # keep
race = "off", race = "off",
deps = [ deps = [
"//hack/qemu-metadata-api/virtwrapper", "//hack/qemu-metadata-api/dhcp",
"//internal/cloud/metadata", "//internal/cloud/metadata",
"//internal/logger", "//internal/logger",
"@com_github_stretchr_testify//assert", "@com_github_stretchr_testify//assert",
"@com_github_stretchr_testify//require", "@com_github_stretchr_testify//require",
"@org_libvirt_go_libvirt//:libvirt",
], ],
) )

View File

@ -14,7 +14,7 @@ import (
"net/http" "net/http"
"strings" "strings"
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/virtwrapper" "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp"
"github.com/edgelesssys/constellation/v2/internal/cloud/metadata" "github.com/edgelesssys/constellation/v2/internal/cloud/metadata"
"github.com/edgelesssys/constellation/v2/internal/role" "github.com/edgelesssys/constellation/v2/internal/role"
) )
@ -22,16 +22,16 @@ import (
// Server that provides QEMU metadata. // Server that provides QEMU metadata.
type Server struct { type Server struct {
log *slog.Logger log *slog.Logger
virt virConnect dhcpLeaseGetter LeaseGetter
network string network string
initSecretHashVal []byte initSecretHashVal []byte
} }
// New creates a new Server. // New creates a new Server.
func New(log *slog.Logger, network, initSecretHash string, conn virConnect) *Server { func New(log *slog.Logger, network, initSecretHash string, getter LeaseGetter) *Server {
return &Server{ return &Server{
log: log, log: log,
virt: conn, dhcpLeaseGetter: getter,
network: network, network: network,
initSecretHashVal: []byte(initSecretHash), initSecretHashVal: []byte(initSecretHash),
} }
@ -139,15 +139,7 @@ func (s *Server) getEndpoint(w http.ResponseWriter, r *http.Request) {
log := s.log.With(slog.String("peer", r.RemoteAddr)) log := s.log.With(slog.String("peer", r.RemoteAddr))
log.Info("Serving GET request for /endpoint") log.Info("Serving GET request for /endpoint")
net, err := s.virt.LookupNetworkByName(s.network) leases, err := s.dhcpLeaseGetter.GetDHCPLeases()
if err != nil {
log.With(slog.Any("error", err)).Error("Failed to lookup network")
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
defer net.Free()
leases, err := net.GetDHCPLeases()
if err != nil { if err != nil {
log.With(slog.Any("error", err)).Error("Failed to get DHCP leases") log.With(slog.Any("error", err)).Error("Failed to get DHCP leases")
http.Error(w, err.Error(), http.StatusInternalServerError) http.Error(w, err.Error(), http.StatusInternalServerError)
@ -172,13 +164,7 @@ func (s *Server) getEndpoint(w http.ResponseWriter, r *http.Request) {
// listAll returns a list of all active peers. // listAll returns a list of all active peers.
func (s *Server) listAll() ([]metadata.InstanceMetadata, error) { func (s *Server) listAll() ([]metadata.InstanceMetadata, error) {
net, err := s.virt.LookupNetworkByName(s.network) leases, err := s.dhcpLeaseGetter.GetDHCPLeases()
if err != nil {
return nil, err
}
defer net.Free()
leases, err := net.GetDHCPLeases()
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -201,6 +187,7 @@ func (s *Server) listAll() ([]metadata.InstanceMetadata, error) {
return peers, nil return peers, nil
} }
type virConnect interface { // LeaseGetter is an interface for getting DHCP leases.
LookupNetworkByName(name string) (*virtwrapper.Network, error) type LeaseGetter interface {
GetDHCPLeases() ([]dhcp.NetworkDHCPLease, error)
} }

View File

@ -1,41 +0,0 @@
//go:build cgo
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package server
import (
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/virtwrapper"
"libvirt.org/go/libvirt"
)
type stubNetwork struct {
leases []libvirt.NetworkDHCPLease
getLeaseErr error
}
func newStubNetwork(leases []virtwrapper.NetworkDHCPLease, getLeaseErr error) stubNetwork {
libvirtLeases := make([]libvirt.NetworkDHCPLease, len(leases))
for i, l := range leases {
libvirtLeases[i] = libvirt.NetworkDHCPLease{
IPaddr: l.IPaddr,
Hostname: l.Hostname,
}
}
return stubNetwork{
leases: libvirtLeases,
getLeaseErr: getLeaseErr,
}
}
func (n stubNetwork) GetDHCPLeases() ([]libvirt.NetworkDHCPLease, error) {
return n.leases, n.getLeaseErr
}
func (n stubNetwork) Free() error {
return nil
}

View File

@ -1,31 +0,0 @@
//go:build !cgo
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package server
import "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/virtwrapper"
type stubNetwork struct {
leases []virtwrapper.NetworkDHCPLease
getLeaseErr error
}
func newStubNetwork(leases []virtwrapper.NetworkDHCPLease, getLeaseErr error) stubNetwork {
return stubNetwork{
leases: leases,
getLeaseErr: getLeaseErr,
}
}
func (n stubNetwork) GetDHCPLeases() ([]virtwrapper.NetworkDHCPLease, error) {
return n.leases, n.getLeaseErr
}
func (n stubNetwork) Free() error {
return nil
}

View File

@ -9,13 +9,12 @@ package server
import ( import (
"context" "context"
"encoding/json" "encoding/json"
"errors"
"io" "io"
"net/http" "net/http"
"net/http/httptest" "net/http/httptest"
"testing" "testing"
"github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/virtwrapper" "github.com/edgelesssys/constellation/v2/hack/qemu-metadata-api/dhcp"
"github.com/edgelesssys/constellation/v2/internal/cloud/metadata" "github.com/edgelesssys/constellation/v2/internal/cloud/metadata"
"github.com/edgelesssys/constellation/v2/internal/logger" "github.com/edgelesssys/constellation/v2/internal/logger"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
@ -23,15 +22,13 @@ import (
) )
func TestListAll(t *testing.T) { func TestListAll(t *testing.T) {
someErr := errors.New("error")
testCases := map[string]struct { testCases := map[string]struct {
wantErr bool wantErr bool
connect *stubConnect stubLeaseGetter *stubLeaseGetter
}{ }{
"success": { "success": {
connect: &stubConnect{ stubLeaseGetter: &stubLeaseGetter{
network: newStubNetwork([]virtwrapper.NetworkDHCPLease{ leases: []dhcp.NetworkDHCPLease{
{ {
IPaddr: "192.0.100.1", IPaddr: "192.0.100.1",
Hostname: "control-plane-0", Hostname: "control-plane-0",
@ -44,20 +41,12 @@ func TestListAll(t *testing.T) {
IPaddr: "192.0.200.1", IPaddr: "192.0.200.1",
Hostname: "worker-0", Hostname: "worker-0",
}, },
}, nil), },
}, },
}, },
"LookupNetworkByName error": {
connect: &stubConnect{
getNetworkErr: someErr,
},
wantErr: true,
},
"GetDHCPLeases error": { "GetDHCPLeases error": {
connect: &stubConnect{ stubLeaseGetter: &stubLeaseGetter{
network: stubNetwork{ getErr: assert.AnError,
getLeaseErr: someErr,
},
}, },
wantErr: true, wantErr: true,
}, },
@ -67,7 +56,7 @@ func TestListAll(t *testing.T) {
t.Run(name, func(t *testing.T) { t.Run(name, func(t *testing.T) {
assert := assert.New(t) assert := assert.New(t)
server := New(logger.NewTest(t), "test", "initSecretHash", tc.connect) server := New(logger.NewTest(t), "test", "initSecretHash", tc.stubLeaseGetter)
res, err := server.listAll() res, err := server.listAll()
@ -76,58 +65,56 @@ func TestListAll(t *testing.T) {
return return
} }
assert.NoError(err) assert.NoError(err)
assert.Len(tc.connect.network.leases, len(res)) assert.Len(tc.stubLeaseGetter.leases, len(res))
}) })
} }
} }
func TestListSelf(t *testing.T) { func TestListSelf(t *testing.T) {
someErr := errors.New("error")
testCases := map[string]struct { testCases := map[string]struct {
remoteAddr string remoteAddr string
connect *stubConnect stubLeaseGetter *stubLeaseGetter
wantErr bool wantErr bool
}{ }{
"success": { "success": {
remoteAddr: "192.0.100.1:1234", remoteAddr: "192.0.100.1:1234",
connect: &stubConnect{ stubLeaseGetter: &stubLeaseGetter{
network: newStubNetwork([]virtwrapper.NetworkDHCPLease{ leases: []dhcp.NetworkDHCPLease{
{ {
IPaddr: "192.0.100.1", IPaddr: "192.0.100.1",
Hostname: "control-plane-0", Hostname: "control-plane-0",
}, },
}, nil), },
}, },
}, },
"listAll error": { "listAll error": {
remoteAddr: "192.0.100.1:1234", remoteAddr: "192.0.100.1:1234",
connect: &stubConnect{ stubLeaseGetter: &stubLeaseGetter{
getNetworkErr: someErr, getErr: assert.AnError,
}, },
wantErr: true, wantErr: true,
}, },
"remoteAddr error": { "remoteAddr error": {
remoteAddr: "", remoteAddr: "",
connect: &stubConnect{ stubLeaseGetter: &stubLeaseGetter{
network: newStubNetwork([]virtwrapper.NetworkDHCPLease{ leases: []dhcp.NetworkDHCPLease{
{ {
IPaddr: "192.0.100.1", IPaddr: "192.0.100.1",
Hostname: "control-plane-0", Hostname: "control-plane-0",
}, },
}, nil), },
}, },
wantErr: true, wantErr: true,
}, },
"peer not found": { "peer not found": {
remoteAddr: "192.0.200.1:1234", remoteAddr: "192.0.200.1:1234",
connect: &stubConnect{ stubLeaseGetter: &stubLeaseGetter{
network: newStubNetwork([]virtwrapper.NetworkDHCPLease{ leases: []dhcp.NetworkDHCPLease{
{ {
IPaddr: "192.0.100.1", IPaddr: "192.0.100.1",
Hostname: "control-plane-0", Hostname: "control-plane-0",
}, },
}, nil), },
}, },
wantErr: true, wantErr: true,
}, },
@ -138,7 +125,7 @@ func TestListSelf(t *testing.T) {
assert := assert.New(t) assert := assert.New(t)
require := require.New(t) require := require.New(t)
server := New(logger.NewTest(t), "test", "initSecretHash", tc.connect) server := New(logger.NewTest(t), "test", "initSecretHash", tc.stubLeaseGetter)
req, err := http.NewRequestWithContext(context.Background(), http.MethodGet, "http://192.0.0.1/self", nil) req, err := http.NewRequestWithContext(context.Background(), http.MethodGet, "http://192.0.0.1/self", nil)
require.NoError(err) require.NoError(err)
@ -157,22 +144,22 @@ func TestListSelf(t *testing.T) {
var metadata metadata.InstanceMetadata var metadata metadata.InstanceMetadata
require.NoError(json.Unmarshal(metadataRaw, &metadata)) require.NoError(json.Unmarshal(metadataRaw, &metadata))
assert.Equal(tc.connect.network.leases[0].Hostname, metadata.Name) assert.Equal(tc.stubLeaseGetter.leases[0].Hostname, metadata.Name)
assert.Equal(tc.connect.network.leases[0].IPaddr, metadata.VPCIP) assert.Equal(tc.stubLeaseGetter.leases[0].IPaddr, metadata.VPCIP)
}) })
} }
} }
func TestListPeers(t *testing.T) { func TestListPeers(t *testing.T) {
testCases := map[string]struct { testCases := map[string]struct {
remoteAddr string remoteAddr string
connect *stubConnect stubNetworkGetter *stubLeaseGetter
wantErr bool wantErr bool
}{ }{
"success": { "success": {
remoteAddr: "192.0.100.1:1234", remoteAddr: "192.0.100.1:1234",
connect: &stubConnect{ stubNetworkGetter: &stubLeaseGetter{
network: newStubNetwork([]virtwrapper.NetworkDHCPLease{ leases: []dhcp.NetworkDHCPLease{
{ {
IPaddr: "192.0.100.1", IPaddr: "192.0.100.1",
Hostname: "control-plane-0", Hostname: "control-plane-0",
@ -181,13 +168,13 @@ func TestListPeers(t *testing.T) {
IPaddr: "192.0.200.1", IPaddr: "192.0.200.1",
Hostname: "worker-0", Hostname: "worker-0",
}, },
}, nil), },
}, },
}, },
"listAll error": { "listAll error": {
remoteAddr: "192.0.100.1:1234", remoteAddr: "192.0.100.1:1234",
connect: &stubConnect{ stubNetworkGetter: &stubLeaseGetter{
getNetworkErr: errors.New("error"), getErr: assert.AnError,
}, },
wantErr: true, wantErr: true,
}, },
@ -198,7 +185,7 @@ func TestListPeers(t *testing.T) {
assert := assert.New(t) assert := assert.New(t)
require := require.New(t) require := require.New(t)
server := New(logger.NewTest(t), "test", "initSecretHash", tc.connect) server := New(logger.NewTest(t), "test", "initSecretHash", tc.stubNetworkGetter)
req, err := http.NewRequestWithContext(context.Background(), http.MethodGet, "http://192.0.0.1/peers", nil) req, err := http.NewRequestWithContext(context.Background(), http.MethodGet, "http://192.0.0.1/peers", nil)
require.NoError(err) require.NoError(err)
@ -217,22 +204,23 @@ func TestListPeers(t *testing.T) {
var metadata []metadata.InstanceMetadata var metadata []metadata.InstanceMetadata
require.NoError(json.Unmarshal(metadataRaw, &metadata)) require.NoError(json.Unmarshal(metadataRaw, &metadata))
assert.Len(metadata, len(tc.connect.network.leases)) assert.Len(metadata, len(tc.stubNetworkGetter.leases))
}) })
} }
} }
func TestInitSecretHash(t *testing.T) { func TestInitSecretHash(t *testing.T) {
defaultConnect := &stubConnect{ defaultConnect := &stubLeaseGetter{
network: newStubNetwork([]virtwrapper.NetworkDHCPLease{ leases: []dhcp.NetworkDHCPLease{
{ {
IPaddr: "192.0.100.1", IPaddr: "192.0.100.1",
Hostname: "control-plane-0", Hostname: "control-plane-0",
}, },
}, nil), },
} }
testCases := map[string]struct { testCases := map[string]struct {
connect *stubConnect connect *stubLeaseGetter
method string method string
wantHash string wantHash string
wantErr bool wantErr bool
@ -272,11 +260,11 @@ func TestInitSecretHash(t *testing.T) {
} }
} }
type stubConnect struct { type stubLeaseGetter struct {
network stubNetwork leases []dhcp.NetworkDHCPLease
getNetworkErr error getErr error
} }
func (c stubConnect) LookupNetworkByName(_ string) (*virtwrapper.Network, error) { func (c stubLeaseGetter) GetDHCPLeases() ([]dhcp.NetworkDHCPLease, error) {
return &virtwrapper.Network{Net: c.network}, c.getNetworkErr return c.leases, c.getErr
} }

View File

@ -1,56 +0,0 @@
//go:build cgo
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package virtwrapper
import "libvirt.org/go/libvirt"
// Connect wraps a libvirt connection.
type Connect struct {
Conn *libvirt.Connect
}
// LookupNetworkByName looks up a network by name.
func (c *Connect) LookupNetworkByName(name string) (*Network, error) {
net, err := c.Conn.LookupNetworkByName(name)
if err != nil {
return nil, err
}
return &Network{Net: net}, nil
}
// Network wraps a libvirt network.
type Network struct {
Net virNetwork
}
// GetDHCPLeases returns the underlying DHCP leases.
func (n *Network) GetDHCPLeases() ([]NetworkDHCPLease, error) {
leases, err := n.Net.GetDHCPLeases()
if err != nil {
return nil, err
}
ret := make([]NetworkDHCPLease, len(leases))
for i, l := range leases {
ret[i] = NetworkDHCPLease{
IPaddr: l.IPaddr,
Hostname: l.Hostname,
}
}
return ret, nil
}
// Free the network resource.
func (n *Network) Free() {
_ = n.Net.Free()
}
type virNetwork interface {
GetDHCPLeases() ([]libvirt.NetworkDHCPLease, error)
Free() error
}

View File

@ -1,40 +0,0 @@
//go:build !cgo
/*
Copyright (c) Edgeless Systems GmbH
SPDX-License-Identifier: AGPL-3.0-only
*/
package virtwrapper
import "errors"
// Connect wraps a libvirt connection.
type Connect struct{}
// LookupNetworkByName looks up a network by name.
// This function errors if CGO is disabled.
func (c *Connect) LookupNetworkByName(_ string) (*Network, error) {
return nil, errors.New("using virtwrapper requires building with CGO")
}
// Network wraps a libvirt network.
type Network struct {
Net Net
}
// GetDHCPLeases returns the underlying DHCP leases.
// This function errors if CGO is disabled.
func (n *Network) GetDHCPLeases() ([]NetworkDHCPLease, error) {
return n.Net.GetDHCPLeases()
}
// Free the network resource.
// This function does nothing if CGO is disabled.
func (n *Network) Free() {}
// Net is a libvirt Network.
type Net interface {
GetDHCPLeases() ([]NetworkDHCPLease, error)
}