Files
netlink/netns_test.go
Vishvananda Abrams 1e35ba25dc test: Improve test reliability with proper cleanup and isolation
Refactors test setup and teardown logic to use `t.Cleanup` instead
of `defer`. This ensures that cleanup functions are correctly scoped
to each subtest's lifecycle, improving test isolation and reliability.

The `setUpNetlinkTest` helper function is also improved to correctly
save and restore the original network namespace, ensuring that tests
do not leak state.

To support this, a `Close()` method that returns an error is added to
the `Handle` struct, allowing for proper cleanup of underlying netlink
sockets. The test helpers are updated to use this new method,
preventing resource leaks between tests.

Additionally, a bug in the `netns` tests is fixed where a large
namespace ID could overflow a 32-bit integer, causing spurious
failures on some systems.
2025-08-27 09:24:17 -07:00

81 lines
2.1 KiB
Go

//go:build linux
// +build linux
package netlink
import (
"os"
"runtime"
"syscall"
"testing"
"github.com/vishvananda/netns"
)
// TestNetNsIdByFd tests setting and getting the network namespace ID
// by file descriptor. It opens a namespace fd, sets it to a random id,
// then retrieves the ID.
// This does not do any namespace switching.
func TestNetNsIdByFd(t *testing.T) {
skipUnlessRoot(t)
// create a network namespace
ns, err := netns.New()
CheckErrorFail(t, err)
// set its ID
// In an attempt to avoid namespace id collisions, set this to something
// insanely high. When the kernel assigns IDs, it does so starting from 0
// So, just use our pid shifted up 16 bits
wantID := (os.Getpid() << 16) & 0x7FFFFFFF
h, err := NewHandle()
CheckErrorFail(t, err)
err = h.SetNetNsIdByFd(int(ns), wantID)
CheckErrorFail(t, err)
// Get the ID back, make sure it matches
haveID, _ := h.GetNetNsIdByFd(int(ns))
if haveID != wantID {
t.Errorf("GetNetNsIdByFd returned %d, want %d", haveID, wantID)
}
ns.Close()
}
// TestNetNsIdByPid tests manipulating namespace IDs by pid (really, task / thread id)
// Does the same as TestNetNsIdByFd, but we need to change namespaces so we
// actually have a pid in that namespace
func TestNetNsIdByPid(t *testing.T) {
skipUnlessRoot(t)
runtime.LockOSThread() // we need a constant OS thread
origNs, _ := netns.Get()
// create and enter a new netns
ns, err := netns.New()
CheckErrorFail(t, err)
err = netns.Set(ns)
CheckErrorFail(t, err)
// make sure we go back to the original namespace when done
defer func() {
err := netns.Set(origNs)
if err != nil {
panic("failed to restore network ns, bailing")
}
runtime.UnlockOSThread()
}()
// As above, we'll pick a crazy large netnsid to avoid collisions
wantID := (syscall.Gettid() << 16) & 0x7FFFFFFF
h, err := NewHandle()
CheckErrorFail(t, err)
err = h.SetNetNsIdByPid(syscall.Gettid(), wantID)
CheckErrorFail(t, err)
//Get the ID and see if it worked
haveID, _ := h.GetNetNsIdByPid(syscall.Gettid())
if haveID != wantID {
t.Errorf("GetNetNsIdByPid returned %d, want %d", haveID, wantID)
}
}