Files
onvif-go/discovery copy/discovery_test.go
T
2026-01-16 04:11:59 +00:00

455 lines
10 KiB
Go

package discovery
import (
"context"
"errors"
"net"
"testing"
"time"
)
func TestDevice_GetName(t *testing.T) {
tests := []struct {
name string
device *Device
want string
}{
{
name: "device with name in scopes",
device: &Device{
Scopes: []string{
"onvif://www.onvif.org/name/TestCamera",
"onvif://www.onvif.org/hardware/Model123",
},
},
want: "TestCamera",
},
{
name: "device without name in scopes",
device: &Device{
Scopes: []string{
"onvif://www.onvif.org/hardware/Model123",
},
},
want: "",
},
{
name: "device with no scopes",
device: &Device{},
want: "",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := tt.device.GetName(); got != tt.want {
t.Errorf("GetName() = %v, want %v", got, tt.want)
}
})
}
}
func TestDevice_GetDeviceEndpoint(t *testing.T) {
tests := []struct {
name string
device *Device
want string
}{
{
name: "device with valid XAddrs",
device: &Device{
XAddrs: []string{
"http://192.168.1.100:80/onvif/device_service",
"http://192.168.1.100:8080/onvif/device_service",
},
},
want: "http://192.168.1.100:80/onvif/device_service",
},
{
name: "device with no XAddrs",
device: &Device{},
want: "",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := tt.device.GetDeviceEndpoint(); got != tt.want {
t.Errorf("GetDeviceEndpoint() = %v, want %v", got, tt.want)
}
})
}
}
func TestDevice_GetLocation(t *testing.T) {
tests := []struct {
name string
device *Device
want string
}{
{
name: "device with location in scopes",
device: &Device{
Scopes: []string{
"onvif://www.onvif.org/location/Building1",
"onvif://www.onvif.org/hardware/Model123",
},
},
want: "Building1",
},
{
name: "device without location in scopes",
device: &Device{
Scopes: []string{
"onvif://www.onvif.org/hardware/Model123",
},
},
want: "",
},
{
name: "device with no scopes",
device: &Device{},
want: "",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := tt.device.GetLocation(); got != tt.want {
t.Errorf("GetLocation() = %v, want %v", got, tt.want)
}
})
}
}
func TestDiscover_WithTimeout(t *testing.T) {
// This test will timeout since there are likely no actual cameras on the test network
// It validates that the timeout mechanism works
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
defer cancel()
devices, err := Discover(ctx, 500*time.Millisecond)
// We expect either no error (empty devices list) or a timeout/context error
if err != nil && !errors.Is(err, context.DeadlineExceeded) {
t.Logf("Discover returned error: %v (this is expected in test environment)", err)
}
// Devices might be empty in test environment
t.Logf("Discovered %d devices", len(devices))
}
func TestDiscover_InvalidDuration(t *testing.T) {
ctx := context.Background()
// Test with zero duration
devices, err := Discover(ctx, 0)
if err != nil {
t.Logf("Discovery with 0 duration returned error: %v", err)
}
t.Logf("Discovered %d devices with 0 duration", len(devices))
}
func TestParseSpaceSeparated(t *testing.T) {
tests := []struct {
name string
input string
want []string
}{
{
name: "multiple values",
input: "value1 value2 value3",
want: []string{"value1", "value2", "value3"},
},
{
name: "empty string",
input: "",
want: []string{},
},
{
name: "single value",
input: "value1",
want: []string{"value1"},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got := parseSpaceSeparated(tt.input)
if len(got) != len(tt.want) {
t.Errorf("parseSpaceSeparated() = %v, want %v", got, tt.want)
}
})
}
}
func TestDevice_GetTypes(t *testing.T) {
device := &Device{
Types: []string{
"dn:NetworkVideoTransmitter",
"tds:Device",
},
}
types := device.Types
if len(types) != 2 {
t.Errorf("Expected 2 types, got %d", len(types))
}
}
func TestDevice_GetScopes(t *testing.T) {
scopes := []string{
"onvif://www.onvif.org/name/TestCamera",
"onvif://www.onvif.org/location/Building1",
"onvif://www.onvif.org/hardware/Model123",
}
device := &Device{
Scopes: scopes,
}
if len(device.Scopes) != 3 {
t.Errorf("Expected 3 scopes, got %d", len(device.Scopes))
}
// Test specific scope extraction
hasName := false
for _, scope := range device.Scopes {
if scope != "" && scope[:5] == "onvif" {
hasName = true
break
}
}
if !hasName {
t.Error("Expected to find onvif scope")
}
}
func BenchmarkDeviceGetName(b *testing.B) {
device := &Device{
Scopes: []string{
"onvif://www.onvif.org/name/TestCamera",
"onvif://www.onvif.org/hardware/Model123",
},
}
b.ResetTimer()
for i := 0; i < b.N; i++ {
_ = device.GetName()
}
}
func BenchmarkDeviceGetDeviceEndpoint(b *testing.B) {
device := &Device{
XAddrs: []string{
"http://192.168.1.100/onvif/device_service",
"http://192.168.1.100:8080/onvif/device_service",
},
}
b.ResetTimer()
for i := 0; i < b.N; i++ {
_ = device.GetDeviceEndpoint()
}
}
// Tests for network interface discovery features
func TestListNetworkInterfaces(t *testing.T) {
interfaces, err := ListNetworkInterfaces()
if err != nil {
t.Fatalf("ListNetworkInterfaces failed: %v", err)
}
if len(interfaces) == 0 {
t.Skip("No network interfaces available")
}
// Verify loopback interface exists (if available)
for _, iface := range interfaces {
if iface.Name == "lo" {
if len(iface.Addresses) == 0 {
t.Error("Loopback interface should have addresses")
}
break
}
}
// Loopback might not exist on all systems, but there should be at least one interface
t.Logf("Found %d network interface(s)", len(interfaces))
for _, iface := range interfaces {
t.Logf(" - %s: up=%v, multicast=%v, addresses=%v", iface.Name, iface.Up, iface.Multicast, iface.Addresses)
}
}
func TestResolveNetworkInterface(t *testing.T) {
// Determine the loopback interface name based on platform
loopbackName := "lo"
if _, err := net.InterfaceByName("lo"); err != nil {
// Loopback might be "lo0" on macOS
loopbackName = "lo0"
}
tests := []struct {
name string
ifaceSpec string
shouldErr bool
}{
{
name: "loopback by name",
ifaceSpec: loopbackName,
shouldErr: false,
},
{
name: "loopback by ip",
ifaceSpec: "127.0.0.1",
shouldErr: false,
},
{
name: "invalid interface",
ifaceSpec: "nonexistent-interface-12345xyz",
shouldErr: true,
},
{
name: "invalid ip",
ifaceSpec: "999.999.999.999",
shouldErr: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
iface, err := resolveNetworkInterface(tt.ifaceSpec)
if tt.shouldErr {
if err == nil {
t.Errorf("Expected error for interface %s, but got none", tt.ifaceSpec)
}
} else {
if err != nil {
t.Errorf("Unexpected error for interface %s: %v", tt.ifaceSpec, err)
}
if iface == nil {
t.Errorf("Expected interface for %s, but got nil", tt.ifaceSpec)
} else {
t.Logf("Resolved %s to interface: %s", tt.ifaceSpec, iface.Name)
}
}
})
}
}
func TestDiscoverWithOptions_DefaultOptions(t *testing.T) {
// Test with default options (should not error even if no cameras found)
ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
defer cancel()
devices, err := DiscoverWithOptions(ctx, 1*time.Second, &DiscoverOptions{})
if err != nil && !errors.Is(err, context.DeadlineExceeded) {
t.Logf("DiscoverWithOptions returned: %v (this is OK if no cameras on network)", err)
}
// Should return a slice (possibly empty)
if devices == nil {
t.Error("Expected devices slice, got nil")
}
t.Logf("Found %d devices with default options", len(devices))
}
func TestDiscoverWithOptions_NilOptions(t *testing.T) {
// Test with nil options (should work with nil)
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
defer cancel()
devices, err := DiscoverWithOptions(ctx, 500*time.Millisecond, nil)
if err != nil && !errors.Is(err, context.DeadlineExceeded) {
t.Logf("DiscoverWithOptions with nil returned: %v", err)
}
if devices == nil {
t.Error("Expected devices slice, got nil")
}
}
func TestDiscoverWithOptions_LoopbackInterface(t *testing.T) {
// Test with loopback interface for testing
// Try both common loopback names
loopbackName := ""
if _, err := net.InterfaceByName("lo"); err == nil {
loopbackName = "lo"
} else if _, err := net.InterfaceByName("lo0"); err == nil {
loopbackName = "lo0"
} else {
t.Skip("Loopback interface not available on this system")
}
opts := &DiscoverOptions{
NetworkInterface: loopbackName,
}
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
defer cancel()
devices, err := DiscoverWithOptions(ctx, 500*time.Millisecond, opts)
if err != nil && !errors.Is(err, context.DeadlineExceeded) {
t.Logf("DiscoverWithOptions with %s interface: %v (timeout is expected)", loopbackName, err)
}
if devices == nil {
t.Error("Expected devices slice, got nil")
}
t.Logf("Found %d devices on loopback interface", len(devices))
}
func TestDiscoverWithOptions_InvalidInterface(t *testing.T) {
opts := &DiscoverOptions{
NetworkInterface: "nonexistent-interface-xyz",
}
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
defer cancel()
_, err := DiscoverWithOptions(ctx, 500*time.Millisecond, opts)
if err == nil {
t.Error("Expected error for invalid interface, but got none")
}
t.Logf("Got expected error: %v", err)
}
func TestDiscover_BackwardCompatibility(t *testing.T) {
// Test that old Discover function still works (backward compatibility)
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
defer cancel()
devices, err := Discover(ctx, 500*time.Millisecond)
if err != nil && !errors.Is(err, context.DeadlineExceeded) {
t.Logf("Discover returned: %v", err)
}
if devices == nil {
t.Error("Expected devices slice, got nil")
}
t.Logf("Backward compat: found %d devices", len(devices))
}
func BenchmarkListNetworkInterfaces(b *testing.B) {
b.ResetTimer()
for i := 0; i < b.N; i++ {
_, _ = ListNetworkInterfaces()
}
}
func BenchmarkResolveNetworkInterface(b *testing.B) {
b.ResetTimer()
for i := 0; i < b.N; i++ {
_, _ = resolveNetworkInterface("127.0.0.1")
}
}