mirror of
https://github.com/apernet/hysteria.git
synced 2025-04-04 21:17:47 +03:00
109 lines
2.1 KiB
Go
109 lines
2.1 KiB
Go
package client
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
io2 "io"
|
|
"testing"
|
|
"time"
|
|
|
|
coreErrs "github.com/apernet/hysteria/core/errors"
|
|
"github.com/apernet/hysteria/core/internal/protocol"
|
|
"go.uber.org/goleak"
|
|
)
|
|
|
|
type udpEchoIO struct {
|
|
MsgCh chan *protocol.UDPMessage
|
|
CloseCh chan struct{}
|
|
}
|
|
|
|
func (io *udpEchoIO) ReceiveMessage() (*protocol.UDPMessage, error) {
|
|
select {
|
|
case m := <-io.MsgCh:
|
|
return m, nil
|
|
case <-io.CloseCh:
|
|
return nil, errors.New("closed")
|
|
}
|
|
}
|
|
|
|
func (io *udpEchoIO) SendMessage(buf []byte, msg *protocol.UDPMessage) error {
|
|
nMsg := *msg
|
|
nMsg.Data = make([]byte, len(msg.Data))
|
|
copy(nMsg.Data, msg.Data)
|
|
io.MsgCh <- &nMsg
|
|
return nil
|
|
}
|
|
|
|
func (io *udpEchoIO) Close() {
|
|
close(io.CloseCh)
|
|
}
|
|
|
|
func TestUDPSessionManager(t *testing.T) {
|
|
io := &udpEchoIO{
|
|
MsgCh: make(chan *protocol.UDPMessage, 10),
|
|
CloseCh: make(chan struct{}),
|
|
}
|
|
sm := newUDPSessionManager(io)
|
|
|
|
rChan := make(chan error, 5)
|
|
|
|
for i := 0; i < 5; i++ {
|
|
go func(id int) {
|
|
conn, err := sm.NewUDP()
|
|
if err != nil {
|
|
rChan <- err
|
|
return
|
|
}
|
|
defer conn.Close()
|
|
|
|
addr := fmt.Sprintf("wow.com:%d", id)
|
|
for j := 0; j < 2; j++ {
|
|
s := fmt.Sprintf("hello %d %d", id, j)
|
|
err = conn.Send([]byte(s), addr)
|
|
if err != nil {
|
|
rChan <- err
|
|
return
|
|
}
|
|
bs, addr, err := conn.Receive()
|
|
if err != nil {
|
|
rChan <- err
|
|
return
|
|
}
|
|
if string(bs) != s || addr != addr {
|
|
rChan <- fmt.Errorf("unexpected message: %s %s", bs, addr)
|
|
return
|
|
}
|
|
}
|
|
rChan <- nil // Success
|
|
}(i)
|
|
}
|
|
|
|
// Check the results
|
|
for i := 0; i < 5; i++ {
|
|
err := <-rChan
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
// Leak checks
|
|
// Create another UDP session
|
|
conn, err := sm.NewUDP()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
io.Close()
|
|
time.Sleep(1 * time.Second) // Give some time for the goroutines to exit
|
|
_, _, err = conn.Receive()
|
|
if err != io2.EOF {
|
|
t.Fatal("expected EOF after closing io")
|
|
}
|
|
_, err = sm.NewUDP()
|
|
if !errors.Is(err, coreErrs.ClosedError{}) {
|
|
t.Fatal("expected ClosedError after closing io")
|
|
}
|
|
if sm.Count() != 0 {
|
|
t.Error("session count should be 0")
|
|
}
|
|
goleak.VerifyNone(t)
|
|
}
|