mirror of
https://github.com/apernet/hysteria.git
synced 2025-04-03 20:47:38 +03:00
176 lines
5.2 KiB
Go
176 lines
5.2 KiB
Go
package server
|
|
|
|
import (
|
|
"errors"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/mock"
|
|
"go.uber.org/goleak"
|
|
|
|
"github.com/apernet/hysteria/core/v2/internal/protocol"
|
|
)
|
|
|
|
func TestUDPSessionManager(t *testing.T) {
|
|
io := newMockUDPIO(t)
|
|
eventLogger := newMockUDPEventLogger(t)
|
|
sm := newUDPSessionManager(io, eventLogger, 2*time.Second)
|
|
|
|
msgCh := make(chan *protocol.UDPMessage, 4)
|
|
io.EXPECT().ReceiveMessage().RunAndReturn(func() (*protocol.UDPMessage, error) {
|
|
m := <-msgCh
|
|
if m == nil {
|
|
return nil, errors.New("closed")
|
|
}
|
|
return m, nil
|
|
})
|
|
io.EXPECT().Hook(mock.Anything, mock.Anything).Return(nil, nil)
|
|
|
|
go sm.Run()
|
|
|
|
udpReadFunc := func(addr string, ch chan []byte, b []byte) (int, string, error) {
|
|
bs := <-ch
|
|
if bs == nil {
|
|
return 0, "", errors.New("closed")
|
|
}
|
|
n := copy(b, bs)
|
|
return n, addr, nil
|
|
}
|
|
|
|
// Test normal session creation & timeout
|
|
msg1 := &protocol.UDPMessage{
|
|
SessionID: 1234,
|
|
PacketID: 0,
|
|
FragID: 0,
|
|
FragCount: 1,
|
|
Addr: "address1.com:9000",
|
|
Data: []byte("hello"),
|
|
}
|
|
eventLogger.EXPECT().New(msg1.SessionID, msg1.Addr).Return().Once()
|
|
udpConn1 := newMockUDPConn(t)
|
|
udpConn1Ch := make(chan []byte, 1)
|
|
io.EXPECT().UDP(msg1.Addr).Return(udpConn1, nil).Once()
|
|
udpConn1.EXPECT().WriteTo(msg1.Data, msg1.Addr).Return(5, nil).Once()
|
|
udpConn1.EXPECT().ReadFrom(mock.Anything).RunAndReturn(func(b []byte) (int, string, error) {
|
|
return udpReadFunc(msg1.Addr, udpConn1Ch, b)
|
|
})
|
|
io.EXPECT().SendMessage(mock.Anything, &protocol.UDPMessage{
|
|
SessionID: msg1.SessionID,
|
|
PacketID: 0,
|
|
FragID: 0,
|
|
FragCount: 1,
|
|
Addr: msg1.Addr,
|
|
Data: []byte("hi back"),
|
|
}).Return(nil).Once()
|
|
msgCh <- msg1
|
|
udpConn1Ch <- []byte("hi back")
|
|
|
|
msg2 := &protocol.UDPMessage{
|
|
SessionID: 5678,
|
|
PacketID: 0,
|
|
FragID: 0,
|
|
FragCount: 1,
|
|
Addr: "address2.net:12450",
|
|
Data: []byte("how are you"),
|
|
}
|
|
eventLogger.EXPECT().New(msg2.SessionID, msg2.Addr).Return().Once()
|
|
udpConn2 := newMockUDPConn(t)
|
|
udpConn2Ch := make(chan []byte, 1)
|
|
io.EXPECT().UDP(msg2.Addr).Return(udpConn2, nil).Once()
|
|
udpConn2.EXPECT().WriteTo(msg2.Data, msg2.Addr).Return(11, nil).Once()
|
|
udpConn2.EXPECT().ReadFrom(mock.Anything).RunAndReturn(func(b []byte) (int, string, error) {
|
|
return udpReadFunc(msg2.Addr, udpConn2Ch, b)
|
|
})
|
|
io.EXPECT().SendMessage(mock.Anything, &protocol.UDPMessage{
|
|
SessionID: msg2.SessionID,
|
|
PacketID: 0,
|
|
FragID: 0,
|
|
FragCount: 1,
|
|
Addr: msg2.Addr,
|
|
Data: []byte("im fine"),
|
|
}).Return(nil).Once()
|
|
msgCh <- msg2
|
|
udpConn2Ch <- []byte("im fine")
|
|
|
|
msg3 := &protocol.UDPMessage{
|
|
SessionID: 1234,
|
|
PacketID: 0,
|
|
FragID: 0,
|
|
FragCount: 1,
|
|
Addr: "address1.com:9000",
|
|
Data: []byte("who are you?"),
|
|
}
|
|
udpConn1.EXPECT().WriteTo(msg3.Data, msg3.Addr).Return(12, nil).Once()
|
|
io.EXPECT().SendMessage(mock.Anything, &protocol.UDPMessage{
|
|
SessionID: msg3.SessionID,
|
|
PacketID: 0,
|
|
FragID: 0,
|
|
FragCount: 1,
|
|
Addr: msg3.Addr,
|
|
Data: []byte("im your father"),
|
|
}).Return(nil).Once()
|
|
msgCh <- msg3
|
|
udpConn1Ch <- []byte("im your father")
|
|
|
|
// Make sure timeout works (connections closed & close events emitted)
|
|
udpConn1.EXPECT().Close().RunAndReturn(func() error {
|
|
close(udpConn1Ch)
|
|
return nil
|
|
}).Once()
|
|
udpConn2.EXPECT().Close().RunAndReturn(func() error {
|
|
close(udpConn2Ch)
|
|
return nil
|
|
}).Once()
|
|
eventLogger.EXPECT().Close(msg1.SessionID, nil).Once()
|
|
eventLogger.EXPECT().Close(msg2.SessionID, nil).Once()
|
|
|
|
time.Sleep(3 * time.Second) // Wait for timeout
|
|
mock.AssertExpectationsForObjects(t, io, eventLogger, udpConn1, udpConn2)
|
|
|
|
// Test UDP connection close error propagation
|
|
errUDPClosed := errors.New("UDP connection closed")
|
|
msg4 := &protocol.UDPMessage{
|
|
SessionID: 666,
|
|
PacketID: 0,
|
|
FragID: 0,
|
|
FragCount: 1,
|
|
Addr: "oh-no.com:27015",
|
|
Data: []byte("dont say bye"),
|
|
}
|
|
eventLogger.EXPECT().New(msg4.SessionID, msg4.Addr).Return().Once()
|
|
udpConn4 := newMockUDPConn(t)
|
|
io.EXPECT().UDP(msg4.Addr).Return(udpConn4, nil).Once()
|
|
udpConn4.EXPECT().WriteTo(msg4.Data, msg4.Addr).Return(12, nil).Once()
|
|
udpConn4.EXPECT().ReadFrom(mock.Anything).Return(0, "", errUDPClosed).Once()
|
|
udpConn4.EXPECT().Close().Return(nil).Once()
|
|
eventLogger.EXPECT().Close(msg4.SessionID, errUDPClosed).Once()
|
|
msgCh <- msg4
|
|
|
|
time.Sleep(1 * time.Second)
|
|
mock.AssertExpectationsForObjects(t, io, eventLogger, udpConn4)
|
|
|
|
// Test UDP connection creation error propagation
|
|
errUDPIO := errors.New("UDP IO error")
|
|
msg5 := &protocol.UDPMessage{
|
|
SessionID: 777,
|
|
PacketID: 0,
|
|
FragID: 0,
|
|
FragCount: 1,
|
|
Addr: "callmemaybe.com:15353",
|
|
Data: []byte("babe i miss you"),
|
|
}
|
|
eventLogger.EXPECT().New(msg5.SessionID, msg5.Addr).Return().Once()
|
|
io.EXPECT().UDP(msg5.Addr).Return(nil, errUDPIO).Once()
|
|
eventLogger.EXPECT().Close(msg5.SessionID, errUDPIO).Once()
|
|
msgCh <- msg5
|
|
|
|
time.Sleep(1 * time.Second)
|
|
mock.AssertExpectationsForObjects(t, io, eventLogger)
|
|
|
|
// Leak checks
|
|
close(msgCh) // This will return error from ReceiveMessage(), should stop the session manager
|
|
time.Sleep(1 * time.Second) // Wait one more second just to be sure
|
|
assert.Zero(t, sm.Count(), "session count should be 0")
|
|
goleak.VerifyNone(t)
|
|
}
|