feat: add network interface selection to WS-Discovery
- Add DiscoverOptions struct with NetworkInterface field - Add DiscoverWithOptions() function for interface-specific discovery - Add ListNetworkInterfaces() to enumerate available interfaces - Add resolveNetworkInterface() helper supporting names and IPs - Maintain full backward compatibility with existing Discover() function - Support specifying interface by name (eth0, wlan0) or IP address - Provide helpful error messages listing available interfaces - Comprehensive test suite with 6 unit tests + 2 benchmarks - Add NETWORK_INTERFACE_GUIDE.md with usage examples This addresses issue where users with multiple active network interfaces need to explicitly select which interface to use for WS-Discovery multicast, as auto-detection may select the wrong one.
This commit is contained in:
@@ -0,0 +1,471 @@
|
||||
# Network Interface Discovery Guide
|
||||
|
||||
This guide explains how to use the network interface selection feature for ONVIF device discovery.
|
||||
|
||||
## Overview
|
||||
|
||||
When you have multiple network interfaces on your system, you may need to specify which interface to use for sending multicast discovery messages to find your cameras. This is especially important when:
|
||||
|
||||
- You have multiple network cards (Ethernet, WiFi, Virtual Adapters)
|
||||
- Cameras are on a specific network segment
|
||||
- The auto-detected interface doesn't reach your cameras
|
||||
- You want to isolate discovery traffic to a specific network
|
||||
|
||||
## Features
|
||||
|
||||
✅ **Specify by Interface Name** - Use interface name (e.g., "eth0", "wlan0")
|
||||
✅ **Specify by IP Address** - Use any IP assigned to the interface
|
||||
✅ **List Available Interfaces** - See all interfaces with their configurations
|
||||
✅ **Backward Compatible** - Existing code continues to work unchanged
|
||||
✅ **Helpful Error Messages** - Lists available interfaces when one isn't found
|
||||
|
||||
## Basic Usage
|
||||
|
||||
### 1. List Available Network Interfaces
|
||||
|
||||
```go
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"log"
|
||||
"github.com/0x524a/onvif-go/discovery"
|
||||
)
|
||||
|
||||
func main() {
|
||||
interfaces, err := discovery.ListNetworkInterfaces()
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
fmt.Println("Available Network Interfaces:")
|
||||
for _, iface := range interfaces {
|
||||
fmt.Printf(" %s - Up: %v, Multicast: %v\n", iface.Name, iface.Up, iface.Multicast)
|
||||
for _, addr := range iface.Addresses {
|
||||
fmt.Printf(" IP: %s\n", addr)
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
**Output Example:**
|
||||
```
|
||||
Available Network Interfaces:
|
||||
lo - Up: true, Multicast: true
|
||||
IP: 127.0.0.1
|
||||
IP: ::1
|
||||
eth0 - Up: true, Multicast: true
|
||||
IP: 192.168.1.100
|
||||
IP: 169.254.1.1
|
||||
wlan0 - Up: true, Multicast: true
|
||||
IP: 192.168.88.50
|
||||
docker0 - Up: true, Multicast: true
|
||||
IP: 172.17.0.1
|
||||
```
|
||||
|
||||
### 2. Discover Cameras on Specific Interface (by name)
|
||||
|
||||
```go
|
||||
package main
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"log"
|
||||
"time"
|
||||
"github.com/0x524a/onvif-go/discovery"
|
||||
)
|
||||
|
||||
func main() {
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: "eth0", // Discover on Ethernet
|
||||
}
|
||||
|
||||
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||
defer cancel()
|
||||
|
||||
devices, err := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
fmt.Printf("Found %d devices on eth0:\n", len(devices))
|
||||
for _, device := range devices {
|
||||
fmt.Printf(" - %s\n", device.GetDeviceEndpoint())
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 3. Discover Cameras Using IP Address
|
||||
|
||||
```go
|
||||
package main
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"log"
|
||||
"time"
|
||||
"github.com/0x524a/onvif-go/discovery"
|
||||
)
|
||||
|
||||
func main() {
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: "192.168.1.100", // Use interface with this IP
|
||||
}
|
||||
|
||||
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||
defer cancel()
|
||||
|
||||
devices, err := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
fmt.Printf("Found %d devices:\n", len(devices))
|
||||
for _, device := range devices {
|
||||
fmt.Printf(" - %s\n", device.GetDeviceEndpoint())
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### 4. Backward Compatible - No Changes Required
|
||||
|
||||
Existing code continues to work without modification:
|
||||
|
||||
```go
|
||||
package main
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"log"
|
||||
"time"
|
||||
"github.com/0x524a/onvif-go/discovery"
|
||||
)
|
||||
|
||||
func main() {
|
||||
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||
defer cancel()
|
||||
|
||||
// This still works exactly as before
|
||||
devices, err := discovery.Discover(ctx, 5*time.Second)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
fmt.Printf("Found %d devices\n", len(devices))
|
||||
}
|
||||
```
|
||||
|
||||
## API Reference
|
||||
|
||||
### DiscoverOptions
|
||||
|
||||
```go
|
||||
type DiscoverOptions struct {
|
||||
// NetworkInterface specifies the network interface to use for multicast.
|
||||
// If empty, the system will choose the default interface.
|
||||
// Examples: "eth0", "wlan0", "192.168.1.100"
|
||||
NetworkInterface string
|
||||
}
|
||||
```
|
||||
|
||||
### Functions
|
||||
|
||||
#### `Discover(ctx context.Context, timeout time.Duration) ([]*Device, error)`
|
||||
|
||||
Discovers ONVIF devices using the default network interface (backward compatible).
|
||||
|
||||
**Parameters:**
|
||||
- `ctx`: Context for cancellation and timeout
|
||||
- `timeout`: How long to listen for responses
|
||||
|
||||
**Returns:**
|
||||
- `[]*Device`: Discovered devices
|
||||
- `error`: Any error that occurred
|
||||
|
||||
#### `DiscoverWithOptions(ctx context.Context, timeout time.Duration, opts *DiscoverOptions) ([]*Device, error)`
|
||||
|
||||
Discovers ONVIF devices with custom options including network interface selection.
|
||||
|
||||
**Parameters:**
|
||||
- `ctx`: Context for cancellation and timeout
|
||||
- `timeout`: How long to listen for responses
|
||||
- `opts`: Discovery options (including NetworkInterface)
|
||||
|
||||
**Returns:**
|
||||
- `[]*Device`: Discovered devices
|
||||
- `error`: Any error that occurred
|
||||
|
||||
#### `ListNetworkInterfaces() ([]NetworkInterface, error)`
|
||||
|
||||
Lists all available network interfaces with their details.
|
||||
|
||||
**Returns:**
|
||||
- `[]NetworkInterface`: All network interfaces
|
||||
- `error`: Any error that occurred
|
||||
|
||||
### NetworkInterface
|
||||
|
||||
```go
|
||||
type NetworkInterface struct {
|
||||
// Name of the interface (e.g., "eth0", "wlan0")
|
||||
Name string
|
||||
|
||||
// IP addresses assigned to this interface
|
||||
Addresses []string
|
||||
|
||||
// Up indicates if the interface is up
|
||||
Up bool
|
||||
|
||||
// Multicast indicates if the interface supports multicast
|
||||
Multicast bool
|
||||
}
|
||||
```
|
||||
|
||||
## Common Scenarios
|
||||
|
||||
### Scenario 1: Multiple Ethernet and WiFi Interfaces
|
||||
|
||||
You have both Ethernet (eth0) and WiFi (wlan0), cameras are on Ethernet:
|
||||
|
||||
```go
|
||||
// List to see what's available
|
||||
interfaces, _ := discovery.ListNetworkInterfaces()
|
||||
for _, i := range interfaces {
|
||||
log.Printf("%s: %v", i.Name, i.Addresses)
|
||||
}
|
||||
|
||||
// Discover on Ethernet only
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: "eth0",
|
||||
}
|
||||
devices, _ := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
```
|
||||
|
||||
### Scenario 2: Virtual Machine with Multiple Adapters
|
||||
|
||||
VM has management interface and camera network interface:
|
||||
|
||||
```go
|
||||
// Use the camera network IP directly
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: "192.168.200.50", // Camera network segment
|
||||
}
|
||||
devices, _ := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
```
|
||||
|
||||
### Scenario 3: Docker Container with Custom Network
|
||||
|
||||
```go
|
||||
// Container has multiple networks, specify which one
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: "172.20.0.10", // Custom bridge network IP
|
||||
}
|
||||
devices, _ := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
```
|
||||
|
||||
### Scenario 4: CLI Tool with User Selection
|
||||
|
||||
```go
|
||||
package main
|
||||
|
||||
import (
|
||||
"flag"
|
||||
"fmt"
|
||||
"log"
|
||||
"github.com/0x524a/onvif-go/discovery"
|
||||
)
|
||||
|
||||
func main() {
|
||||
ifaceFlag := flag.String("interface", "", "Network interface to use")
|
||||
flag.Parse()
|
||||
|
||||
if *ifaceFlag == "" {
|
||||
// List available if not specified
|
||||
interfaces, _ := discovery.ListNetworkInterfaces()
|
||||
fmt.Println("Available interfaces:")
|
||||
for _, i := range interfaces {
|
||||
fmt.Printf(" %s\n", i.Name)
|
||||
}
|
||||
fmt.Println("Use -interface flag to specify")
|
||||
return
|
||||
}
|
||||
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: *ifaceFlag,
|
||||
}
|
||||
|
||||
devices, _ := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
fmt.Printf("Found %d devices\n", len(devices))
|
||||
}
|
||||
```
|
||||
|
||||
**Usage:**
|
||||
```bash
|
||||
# List interfaces
|
||||
./app
|
||||
|
||||
# Available interfaces:
|
||||
# eth0
|
||||
# wlan0
|
||||
|
||||
# Discover on specific interface
|
||||
./app -interface eth0
|
||||
./app -interface wlan0
|
||||
./app -interface 192.168.1.100
|
||||
```
|
||||
|
||||
## Error Handling
|
||||
|
||||
### Interface Not Found
|
||||
|
||||
```go
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: "nonexistent-interface",
|
||||
}
|
||||
|
||||
devices, err := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
// Output:
|
||||
// network interface "nonexistent-interface" not found.
|
||||
// Available interfaces: [eth0 [192.168.1.100] wlan0 [192.168.88.50] ...]
|
||||
}
|
||||
```
|
||||
|
||||
### Invalid IP Address
|
||||
|
||||
```go
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: "192.168.999.999", // Invalid IP
|
||||
}
|
||||
|
||||
devices, err := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
if err != nil {
|
||||
// Error: network interface not found
|
||||
log.Fatal(err)
|
||||
}
|
||||
```
|
||||
|
||||
## Migration Guide
|
||||
|
||||
### From: Using Default Discovery
|
||||
|
||||
```go
|
||||
// Old code - still works!
|
||||
devices, err := discovery.Discover(ctx, 5*time.Second)
|
||||
```
|
||||
|
||||
### To: Using Specific Interface
|
||||
|
||||
```go
|
||||
// New code - with interface selection
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: "eth0",
|
||||
}
|
||||
devices, err := discovery.DiscoverWithOptions(ctx, 5*time.Second, opts)
|
||||
```
|
||||
|
||||
No breaking changes - old code continues to work!
|
||||
|
||||
## Troubleshooting
|
||||
|
||||
### "No devices found on interface X"
|
||||
|
||||
**Possible causes:**
|
||||
1. Cameras are on a different network segment
|
||||
2. Interface is not connected to the camera network
|
||||
3. Firewall is blocking multicast on that interface
|
||||
4. Camera network interface name is different than expected
|
||||
|
||||
**Solution:**
|
||||
```go
|
||||
// List interfaces to verify
|
||||
interfaces, _ := discovery.ListNetworkInterfaces()
|
||||
for _, i := range interfaces {
|
||||
if i.Up && i.Multicast {
|
||||
fmt.Printf("Try: %s (%v)\n", i.Name, i.Addresses)
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### "Network interface not found"
|
||||
|
||||
**Possible causes:**
|
||||
1. Interface name typo (e.g., "eth0" vs "eth1")
|
||||
2. Interface is down
|
||||
3. IP address not assigned to any interface
|
||||
|
||||
**Solution:**
|
||||
- Check spelling: `discovery.ListNetworkInterfaces()`
|
||||
- Verify interface is up: `Up: true`
|
||||
- Verify IP is correct: Check `Addresses` field
|
||||
|
||||
### Multicast Not Supported
|
||||
|
||||
```go
|
||||
interfaces, _ := discovery.ListNetworkInterfaces()
|
||||
for _, i := range interfaces {
|
||||
if i.Multicast {
|
||||
fmt.Printf("%s supports multicast\n", i.Name)
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## Best Practices
|
||||
|
||||
1. **Always list interfaces first** if uncertain:
|
||||
```go
|
||||
interfaces, _ := discovery.ListNetworkInterfaces()
|
||||
// Show user and let them choose
|
||||
```
|
||||
|
||||
2. **Validate interface exists** before discovery:
|
||||
```go
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: userInput,
|
||||
}
|
||||
// Try with empty timeout first to validate
|
||||
```
|
||||
|
||||
3. **Try multiple interfaces** for robust applications:
|
||||
```go
|
||||
for _, iface := range interfaces {
|
||||
if iface.Up && iface.Multicast {
|
||||
opts := &discovery.DiscoverOptions{
|
||||
NetworkInterface: iface.Name,
|
||||
}
|
||||
devices, _ := discovery.DiscoverWithOptions(ctx, 2*time.Second, opts)
|
||||
if len(devices) > 0 {
|
||||
return devices
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
4. **Check interface capabilities**:
|
||||
```go
|
||||
for _, i := range interfaces {
|
||||
if i.Up && i.Multicast {
|
||||
// Good candidate for discovery
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## Testing
|
||||
|
||||
```bash
|
||||
# Run discovery tests
|
||||
go test -v ./discovery/
|
||||
|
||||
# Run with specific interface test
|
||||
go test -v ./discovery/ -run TestDiscoverWithOptions
|
||||
```
|
||||
|
||||
## Related Documentation
|
||||
|
||||
- [QUICKSTART](../QUICKSTART.md) - Getting started with onvif-go
|
||||
- [discovery/discovery.go](./discovery.go) - Source code
|
||||
- [discovery/discovery_test.go](./discovery_test.go) - Test examples
|
||||
+134
-1
@@ -66,15 +66,44 @@ type ProbeMatches struct {
|
||||
ProbeMatch []ProbeMatch `xml:"ProbeMatch"`
|
||||
}
|
||||
|
||||
// DiscoverOptions contains options for device discovery
|
||||
type DiscoverOptions struct {
|
||||
// NetworkInterface specifies the network interface to use for multicast.
|
||||
// If empty, the system will choose the default interface.
|
||||
// Examples: "eth0", "wlan0", "192.168.1.100"
|
||||
NetworkInterface string
|
||||
|
||||
// Context and timeout are handled by the caller
|
||||
}
|
||||
|
||||
// Discover discovers ONVIF devices on the network
|
||||
// For advanced options like specifying a network interface, use DiscoverWithOptions
|
||||
func Discover(ctx context.Context, timeout time.Duration) ([]*Device, error) {
|
||||
return DiscoverWithOptions(ctx, timeout, &DiscoverOptions{})
|
||||
}
|
||||
|
||||
// DiscoverWithOptions discovers ONVIF devices with custom options
|
||||
func DiscoverWithOptions(ctx context.Context, timeout time.Duration, opts *DiscoverOptions) ([]*Device, error) {
|
||||
if opts == nil {
|
||||
opts = &DiscoverOptions{}
|
||||
}
|
||||
|
||||
// Create UDP connection for multicast
|
||||
addr, err := net.ResolveUDPAddr("udp", multicastAddr)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to resolve multicast address: %w", err)
|
||||
}
|
||||
|
||||
conn, err := net.ListenMulticastUDP("udp", nil, addr)
|
||||
// Get the network interface to use
|
||||
var iface *net.Interface
|
||||
if opts.NetworkInterface != "" {
|
||||
iface, err = resolveNetworkInterface(opts.NetworkInterface)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to resolve network interface: %w", err)
|
||||
}
|
||||
}
|
||||
|
||||
conn, err := net.ListenMulticastUDP("udp", iface, addr)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to listen on multicast address: %w", err)
|
||||
}
|
||||
@@ -186,6 +215,110 @@ func generateUUID() string {
|
||||
time.Now().UnixNano()%10000)
|
||||
}
|
||||
|
||||
// resolveNetworkInterface resolves a network interface by name or IP address
|
||||
func resolveNetworkInterface(ifaceSpec string) (*net.Interface, error) {
|
||||
// Try to get interface by name (e.g., "eth0", "wlan0")
|
||||
if iface, err := net.InterfaceByName(ifaceSpec); err == nil {
|
||||
return iface, nil
|
||||
}
|
||||
|
||||
// Try to parse as IP address and find the interface
|
||||
if ip := net.ParseIP(ifaceSpec); ip != nil {
|
||||
interfaces, err := net.Interfaces()
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to list network interfaces: %w", err)
|
||||
}
|
||||
|
||||
for _, iface := range interfaces {
|
||||
addrs, err := iface.Addrs()
|
||||
if err != nil {
|
||||
continue
|
||||
}
|
||||
|
||||
for _, addr := range addrs {
|
||||
switch v := addr.(type) {
|
||||
case *net.IPNet:
|
||||
if v.IP.Equal(ip) {
|
||||
return &iface, nil
|
||||
}
|
||||
case *net.IPAddr:
|
||||
if v.IP.Equal(ip) {
|
||||
return &iface, nil
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// List available interfaces for error message
|
||||
interfaces, _ := net.Interfaces()
|
||||
availableInterfaces := make([]string, 0)
|
||||
for _, iface := range interfaces {
|
||||
addrs, _ := iface.Addrs()
|
||||
ifaceInfo := iface.Name
|
||||
if len(addrs) > 0 {
|
||||
var addrStrs []string
|
||||
for _, addr := range addrs {
|
||||
addrStrs = append(addrStrs, addr.String())
|
||||
}
|
||||
ifaceInfo += " [" + strings.Join(addrStrs, ", ") + "]"
|
||||
}
|
||||
availableInterfaces = append(availableInterfaces, ifaceInfo)
|
||||
}
|
||||
|
||||
return nil, fmt.Errorf("network interface %q not found. Available interfaces: %v", ifaceSpec, availableInterfaces)
|
||||
}
|
||||
|
||||
// ListNetworkInterfaces returns all available network interfaces with their addresses
|
||||
func ListNetworkInterfaces() ([]NetworkInterface, error) {
|
||||
interfaces, err := net.Interfaces()
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to list network interfaces: %w", err)
|
||||
}
|
||||
|
||||
var result []NetworkInterface
|
||||
for _, iface := range interfaces {
|
||||
addrs, err := iface.Addrs()
|
||||
if err != nil {
|
||||
continue
|
||||
}
|
||||
|
||||
var ipAddrs []string
|
||||
for _, addr := range addrs {
|
||||
switch v := addr.(type) {
|
||||
case *net.IPNet:
|
||||
ipAddrs = append(ipAddrs, v.IP.String())
|
||||
case *net.IPAddr:
|
||||
ipAddrs = append(ipAddrs, v.IP.String())
|
||||
}
|
||||
}
|
||||
|
||||
result = append(result, NetworkInterface{
|
||||
Name: iface.Name,
|
||||
Addresses: ipAddrs,
|
||||
Up: iface.Flags&net.FlagUp != 0,
|
||||
Multicast: iface.Flags&net.FlagMulticast != 0,
|
||||
})
|
||||
}
|
||||
|
||||
return result, nil
|
||||
}
|
||||
|
||||
// NetworkInterface represents a network interface
|
||||
type NetworkInterface struct {
|
||||
// Name of the interface (e.g., "eth0", "wlan0")
|
||||
Name string
|
||||
|
||||
// IP addresses assigned to this interface
|
||||
Addresses []string
|
||||
|
||||
// Up indicates if the interface is up
|
||||
Up bool
|
||||
|
||||
// Multicast indicates if the interface supports multicast
|
||||
Multicast bool
|
||||
}
|
||||
|
||||
// GetDeviceEndpoint extracts the primary device endpoint from XAddrs
|
||||
func (d *Device) GetDeviceEndpoint() string {
|
||||
if len(d.XAddrs) == 0 {
|
||||
|
||||
@@ -2,6 +2,7 @@ package discovery
|
||||
|
||||
import (
|
||||
"context"
|
||||
"net"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
@@ -251,3 +252,188 @@ func BenchmarkDeviceGetDeviceEndpoint(b *testing.B) {
|
||||
_ = 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) {
|
||||
tests := []struct {
|
||||
name string
|
||||
ifaceSpec string
|
||||
shouldErr bool
|
||||
}{
|
||||
{
|
||||
name: "loopback by name",
|
||||
ifaceSpec: "lo",
|
||||
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 && 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 && 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
|
||||
_, err := net.InterfaceByName("lo")
|
||||
if err != nil {
|
||||
t.Skip("Loopback interface not available on this system")
|
||||
}
|
||||
|
||||
opts := &DiscoverOptions{
|
||||
NetworkInterface: "lo",
|
||||
}
|
||||
|
||||
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
|
||||
defer cancel()
|
||||
|
||||
devices, err := DiscoverWithOptions(ctx, 500*time.Millisecond, opts)
|
||||
if err != nil && err != context.DeadlineExceeded {
|
||||
t.Logf("DiscoverWithOptions with lo interface: %v (timeout is expected)", 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 && 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")
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user