Files
ice/agent_udpmux_test.go
2025-09-22 17:39:54 +03:00

140 lines
3.4 KiB
Go

// SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
// SPDX-License-Identifier: MIT
//go:build !js
// +build !js
package ice
import (
"net"
"testing"
"time"
"github.com/pion/logging"
"github.com/pion/transport/v3/test"
"github.com/stretchr/testify/require"
)
// newMuxForAddr creates a UDPMuxDefault with the correct socket family for the given address.
// This fixes Windows dual-stack issues where IPv6 sockets don't receive IPv4 traffic by default.
func newMuxForAddr(t *testing.T, addr *net.UDPAddr, loggerFactory logging.LoggerFactory) *UDPMuxDefault {
t.Helper()
var (
network string
laddr *net.UDPAddr
)
switch {
case addr.IP == nil || addr.IP.IsUnspecified():
network = "udp4"
laddr = &net.UDPAddr{IP: net.IPv4zero, Port: addr.Port}
case addr.IP.To4() != nil:
network = "udp4"
laddr = &net.UDPAddr{IP: net.IPv4zero, Port: addr.Port}
default:
network = "udp6"
laddr = &net.UDPAddr{IP: net.IPv6unspecified, Port: addr.Port}
}
pc, err := net.ListenUDP(network, laddr)
require.NoError(t, err)
t.Cleanup(func() { _ = pc.Close() })
return NewUDPMuxDefault(UDPMuxParams{
Logger: loggerFactory.NewLogger("ice"),
UDPConn: pc,
})
}
// TestMuxAgent is an end to end test over UDP mux, ensuring two agents could connect over mux.
func TestMuxAgent(t *testing.T) {
defer test.CheckRoutines(t)()
defer test.TimeOut(time.Second * 30).Stop()
const muxPort = 7686
caseAddrs := map[string]*net.UDPAddr{
"unspecified": {Port: muxPort},
"ipv4Loopback": {IP: net.IPv4(127, 0, 0, 1), Port: muxPort},
}
for subTest, addr := range caseAddrs {
muxAddr := addr
t.Run(subTest, func(t *testing.T) {
loggerFactory := logging.NewDefaultLoggerFactory()
udpMux := newMuxForAddr(t, muxAddr, loggerFactory)
muxedA, err := NewAgent(&AgentConfig{
UDPMux: udpMux,
CandidateTypes: []CandidateType{CandidateTypeHost},
NetworkTypes: []NetworkType{
NetworkTypeUDP4,
},
IncludeLoopback: addr.IP.IsLoopback(),
})
require.NoError(t, err)
var muxedAClosed bool
defer func() {
if muxedAClosed {
return
}
require.NoError(t, muxedA.Close())
}()
agent, err := NewAgent(&AgentConfig{
CandidateTypes: []CandidateType{CandidateTypeHost},
NetworkTypes: supportedNetworkTypes(),
})
require.NoError(t, err)
var aClosed bool
defer func() {
if aClosed {
return
}
require.NoError(t, agent.Close())
}()
conn, muxedConn := connect(t, agent, muxedA)
pair := muxedA.getSelectedPair()
require.NotNil(t, pair)
require.Equal(t, muxPort, pair.Local.Port())
// Send a packet to Mux
data := []byte("hello world")
_, err = conn.Write(data)
require.NoError(t, err)
buf := make([]byte, 1024)
n, err := muxedConn.Read(buf)
require.NoError(t, err)
require.Equal(t, data, buf[:n])
// Send a packet from Mux
_, err = muxedConn.Write(data)
require.NoError(t, err)
n, err = conn.Read(buf)
require.NoError(t, err)
require.Equal(t, data, buf[:n])
// Close it down
require.NoError(t, conn.Close())
aClosed = true
require.NoError(t, muxedConn.Close())
muxedAClosed = true
require.NoError(t, udpMux.Close())
// Expect error when reading from closed mux
_, err = muxedConn.Read(data)
require.Error(t, err)
// Expect error when writing to closed mux
_, err = muxedConn.Write(data)
require.Error(t, err)
})
}
}