9e3b5e0170
- Introduced CAMERA_TEST_REPORT.md and COMPREHENSIVE_TEST_SUMMARY.md to document testing results for the Bosch FLEXIDOME indoor 5100i IR camera. - Added detailed analysis of ONVIF Media Service operations and implementation status in MEDIA_OPERATIONS_ANALYSIS.md and MEDIA_WSDL_OPERATIONS_ANALYSIS.md. - Updated implementation status documentation to reflect the completion of all 79 operations in the ONVIF Media Service. - Enhanced existing comments and documentation across various files for better clarity and consistency.
455 lines
10 KiB
Go
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")
|
|
}
|
|
}
|