mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-03 20:27:35 +03:00
Move proxy/ to new tools/ folder
This commit is contained in:
parent
8ec11c0b53
commit
56cbce35b3
7 changed files with 4 additions and 4 deletions
348
integrationtests/tools/proxy/proxy_test.go
Normal file
348
integrationtests/tools/proxy/proxy_test.go
Normal file
|
@ -0,0 +1,348 @@
|
|||
package quicproxy
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"net"
|
||||
"strconv"
|
||||
"sync/atomic"
|
||||
"time"
|
||||
|
||||
"fmt"
|
||||
|
||||
"github.com/lucas-clemente/quic-go"
|
||||
"github.com/lucas-clemente/quic-go/protocol"
|
||||
. "github.com/onsi/ginkgo"
|
||||
. "github.com/onsi/gomega"
|
||||
)
|
||||
|
||||
type packetData []byte
|
||||
|
||||
var _ = Describe("QUIC Proxy", func() {
|
||||
var serverAddr string
|
||||
|
||||
makePacket := func(p protocol.PacketNumber, payload []byte) []byte {
|
||||
b := &bytes.Buffer{}
|
||||
hdr := quic.PublicHeader{
|
||||
PacketNumber: p,
|
||||
PacketNumberLen: protocol.PacketNumberLen6,
|
||||
ConnectionID: 1337,
|
||||
TruncateConnectionID: false,
|
||||
}
|
||||
hdr.Write(b, protocol.VersionWhatever, protocol.PerspectiveServer)
|
||||
raw := b.Bytes()
|
||||
raw = append(raw, payload...)
|
||||
return raw
|
||||
}
|
||||
|
||||
BeforeEach(func() {
|
||||
serverAddr = "localhost:7331"
|
||||
})
|
||||
|
||||
Context("Proxy setup and teardown", func() {
|
||||
It("sets up the UDPProxy", func() {
|
||||
proxy, err := NewQuicProxy("localhost:0", Opts{RemoteAddr: serverAddr})
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
Expect(proxy.clientDict).To(HaveLen(0))
|
||||
|
||||
// check that the proxy port is in use
|
||||
addr, err := net.ResolveUDPAddr("udp", "localhost:"+strconv.Itoa(proxy.LocalPort()))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
_, err = net.ListenUDP("udp", addr)
|
||||
Expect(err).To(MatchError(fmt.Sprintf("listen udp 127.0.0.1:%d: bind: address already in use", proxy.LocalPort())))
|
||||
Expect(proxy.Close()).To(Succeed()) // stopping is tested in the next test
|
||||
})
|
||||
|
||||
It("stops the UDPProxy", func() {
|
||||
proxy, err := NewQuicProxy("localhost:0", Opts{RemoteAddr: serverAddr})
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
port := proxy.LocalPort()
|
||||
err = proxy.Close()
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
|
||||
// check that the proxy port is not in use anymore
|
||||
addr, err := net.ResolveUDPAddr("udp", "localhost:"+strconv.Itoa(port))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
// sometimes it takes a while for the OS to free the port
|
||||
Eventually(func() error {
|
||||
ln, err := net.ListenUDP("udp", addr)
|
||||
defer ln.Close()
|
||||
return err
|
||||
}).ShouldNot(HaveOccurred())
|
||||
})
|
||||
|
||||
It("has the correct LocalAddr and LocalPort", func() {
|
||||
proxy, err := NewQuicProxy("localhost:0", Opts{RemoteAddr: serverAddr})
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
|
||||
Expect(proxy.LocalAddr().String()).To(Equal("127.0.0.1:" + strconv.Itoa(proxy.LocalPort())))
|
||||
Expect(proxy.LocalPort()).ToNot(BeZero())
|
||||
|
||||
Expect(proxy.Close()).To(Succeed())
|
||||
})
|
||||
})
|
||||
|
||||
Context("Proxy tests", func() {
|
||||
var (
|
||||
serverConn *net.UDPConn
|
||||
serverReceivedPackets []packetData
|
||||
serverNumPacketsSent int
|
||||
clientConn *net.UDPConn
|
||||
proxy *QuicProxy
|
||||
)
|
||||
|
||||
startProxy := func(opts Opts) {
|
||||
var err error
|
||||
proxy, err = NewQuicProxy("localhost:0", opts)
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
clientConn, err = net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
}
|
||||
|
||||
BeforeEach(func() {
|
||||
serverReceivedPackets = serverReceivedPackets[:0]
|
||||
serverNumPacketsSent = 0
|
||||
|
||||
// setup a dump UDP server on port 7331
|
||||
// in production this would be a QUIC server
|
||||
raddr, err := net.ResolveUDPAddr("udp", serverAddr)
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
serverConn, err = net.ListenUDP("udp", raddr)
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
|
||||
go func() {
|
||||
for {
|
||||
buf := make([]byte, protocol.MaxPacketSize)
|
||||
// the ReadFromUDP will error as soon as the UDP conn is closed
|
||||
n, addr, err2 := serverConn.ReadFromUDP(buf)
|
||||
if err2 != nil {
|
||||
return
|
||||
}
|
||||
data := buf[0:n]
|
||||
serverReceivedPackets = append(serverReceivedPackets, packetData(data))
|
||||
// echo the packet
|
||||
serverConn.WriteToUDP(data, addr)
|
||||
serverNumPacketsSent++
|
||||
}
|
||||
}()
|
||||
})
|
||||
|
||||
AfterEach(func() {
|
||||
err := proxy.Close()
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
err = serverConn.Close()
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
err = clientConn.Close()
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
time.Sleep(200 * time.Millisecond)
|
||||
})
|
||||
|
||||
Context("no packet drop", func() {
|
||||
It("relays packets from the client to the server", func() {
|
||||
startProxy(Opts{RemoteAddr: serverAddr})
|
||||
// send the first packet
|
||||
_, err := clientConn.Write(makePacket(1, []byte("foobar")))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
|
||||
Eventually(func() map[string]*connection { return proxy.clientDict }).Should(HaveLen(1))
|
||||
var conn *connection
|
||||
for _, conn = range proxy.clientDict {
|
||||
Expect(atomic.LoadUint64(&conn.incomingPacketCounter)).To(Equal(uint64(1)))
|
||||
}
|
||||
|
||||
// send the second packet
|
||||
_, err = clientConn.Write(makePacket(2, []byte("decafbad")))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
|
||||
Eventually(func() []packetData { return serverReceivedPackets }).Should(HaveLen(2))
|
||||
Expect(proxy.clientDict).To(HaveLen(1))
|
||||
Expect(string(serverReceivedPackets[0])).To(ContainSubstring("foobar"))
|
||||
Expect(string(serverReceivedPackets[1])).To(ContainSubstring("decafbad"))
|
||||
})
|
||||
|
||||
It("relays packets from the server to the client", func() {
|
||||
startProxy(Opts{RemoteAddr: serverAddr})
|
||||
// send the first packet
|
||||
_, err := clientConn.Write(makePacket(1, []byte("foobar")))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
|
||||
Eventually(func() map[string]*connection { return proxy.clientDict }).Should(HaveLen(1))
|
||||
var key string
|
||||
var conn *connection
|
||||
for key, conn = range proxy.clientDict {
|
||||
Eventually(func() uint64 { return atomic.LoadUint64(&conn.outgoingPacketCounter) }).Should(Equal(uint64(1)))
|
||||
}
|
||||
|
||||
// send the second packet
|
||||
_, err = clientConn.Write(makePacket(2, []byte("decafbad")))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
|
||||
Expect(proxy.clientDict).To(HaveLen(1))
|
||||
Eventually(func() uint64 { return proxy.clientDict[key].outgoingPacketCounter }).Should(BeEquivalentTo(2))
|
||||
|
||||
var clientReceivedPackets []packetData
|
||||
// receive the packets echoed by the server on client side
|
||||
go func() {
|
||||
for {
|
||||
buf := make([]byte, protocol.MaxPacketSize)
|
||||
// the ReadFromUDP will error as soon as the UDP conn is closed
|
||||
n, _, err2 := clientConn.ReadFromUDP(buf)
|
||||
if err2 != nil {
|
||||
return
|
||||
}
|
||||
data := buf[0:n]
|
||||
clientReceivedPackets = append(clientReceivedPackets, packetData(data))
|
||||
}
|
||||
}()
|
||||
|
||||
Eventually(func() []packetData { return serverReceivedPackets }).Should(HaveLen(2))
|
||||
Expect(serverReceivedPackets).To(HaveLen(2))
|
||||
Expect(serverNumPacketsSent).To(Equal(2))
|
||||
Eventually(func() []packetData { return clientReceivedPackets }).Should(HaveLen(2))
|
||||
Expect(string(clientReceivedPackets[0])).To(ContainSubstring("foobar"))
|
||||
Expect(string(clientReceivedPackets[1])).To(ContainSubstring("decafbad"))
|
||||
})
|
||||
})
|
||||
|
||||
Context("Drop Callbacks", func() {
|
||||
It("drops incoming packets", func() {
|
||||
opts := Opts{
|
||||
RemoteAddr: serverAddr,
|
||||
DropPacket: func(d Direction, p protocol.PacketNumber) bool {
|
||||
return d == DirectionIncoming && p%2 == 0
|
||||
},
|
||||
}
|
||||
startProxy(opts)
|
||||
|
||||
// send 6 packets
|
||||
for i := 1; i <= 6; i++ {
|
||||
_, err := clientConn.Write(makePacket(protocol.PacketNumber(i), []byte("foobar"+strconv.Itoa(i))))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
}
|
||||
Eventually(func() []packetData { return serverReceivedPackets }).Should(HaveLen(3))
|
||||
Consistently(func() []packetData { return serverReceivedPackets }).Should(HaveLen(3))
|
||||
})
|
||||
|
||||
It("drops outgoing packets", func() {
|
||||
opts := Opts{
|
||||
RemoteAddr: serverAddr,
|
||||
DropPacket: func(d Direction, p protocol.PacketNumber) bool {
|
||||
return d == DirectionOutgoing && p%2 == 0
|
||||
},
|
||||
}
|
||||
startProxy(opts)
|
||||
|
||||
var clientReceivedPackets []packetData
|
||||
// receive the packets echoed by the server on client side
|
||||
go func() {
|
||||
for {
|
||||
buf := make([]byte, protocol.MaxPacketSize)
|
||||
// the ReadFromUDP will error as soon as the UDP conn is closed
|
||||
n, _, err2 := clientConn.ReadFromUDP(buf)
|
||||
if err2 != nil {
|
||||
return
|
||||
}
|
||||
data := buf[0:n]
|
||||
clientReceivedPackets = append(clientReceivedPackets, packetData(data))
|
||||
}
|
||||
}()
|
||||
|
||||
// send 6 packets
|
||||
for i := 1; i <= 6; i++ {
|
||||
_, err := clientConn.Write(makePacket(protocol.PacketNumber(i), []byte("foobar"+strconv.Itoa(i))))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
}
|
||||
|
||||
Eventually(func() []packetData { return clientReceivedPackets }).Should(HaveLen(3))
|
||||
Consistently(func() []packetData { return clientReceivedPackets }).Should(HaveLen(3))
|
||||
})
|
||||
})
|
||||
|
||||
Context("Delay Callback", func() {
|
||||
expectDelay := func(startTime time.Time, rtt time.Duration, numRTTs int) {
|
||||
expectedReceiveTime := startTime.Add(time.Duration(numRTTs) * rtt)
|
||||
Expect(time.Now()).To(SatisfyAll(
|
||||
BeTemporally(">=", expectedReceiveTime),
|
||||
BeTemporally("<", expectedReceiveTime.Add(rtt/2)),
|
||||
))
|
||||
}
|
||||
|
||||
It("delays incoming packets", func() {
|
||||
delay := 300 * time.Millisecond
|
||||
opts := Opts{
|
||||
RemoteAddr: serverAddr,
|
||||
// delay packet 1 by 200 ms
|
||||
// delay packet 2 by 400 ms
|
||||
// ...
|
||||
DelayPacket: func(d Direction, p protocol.PacketNumber) time.Duration {
|
||||
if d == DirectionOutgoing {
|
||||
return 0
|
||||
}
|
||||
return time.Duration(p) * delay
|
||||
},
|
||||
}
|
||||
startProxy(opts)
|
||||
|
||||
// send 3 packets
|
||||
start := time.Now()
|
||||
for i := 1; i <= 3; i++ {
|
||||
_, err := clientConn.Write(makePacket(protocol.PacketNumber(i), []byte("foobar"+strconv.Itoa(i))))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
}
|
||||
Eventually(func() []packetData { return serverReceivedPackets }).Should(HaveLen(1))
|
||||
expectDelay(start, delay, 1)
|
||||
Eventually(func() []packetData { return serverReceivedPackets }).Should(HaveLen(2))
|
||||
expectDelay(start, delay, 2)
|
||||
Eventually(func() []packetData { return serverReceivedPackets }).Should(HaveLen(3))
|
||||
expectDelay(start, delay, 3)
|
||||
})
|
||||
|
||||
It("delays outgoing packets", func() {
|
||||
delay := 300 * time.Millisecond
|
||||
opts := Opts{
|
||||
RemoteAddr: serverAddr,
|
||||
// delay packet 1 by 200 ms
|
||||
// delay packet 2 by 400 ms
|
||||
// ...
|
||||
DelayPacket: func(d Direction, p protocol.PacketNumber) time.Duration {
|
||||
if d == DirectionIncoming {
|
||||
return 0
|
||||
}
|
||||
return time.Duration(p) * delay
|
||||
},
|
||||
}
|
||||
startProxy(opts)
|
||||
|
||||
var clientReceivedPackets []packetData
|
||||
// receive the packets echoed by the server on client side
|
||||
go func() {
|
||||
for {
|
||||
buf := make([]byte, protocol.MaxPacketSize)
|
||||
// the ReadFromUDP will error as soon as the UDP conn is closed
|
||||
n, _, err2 := clientConn.ReadFromUDP(buf)
|
||||
if err2 != nil {
|
||||
return
|
||||
}
|
||||
data := buf[0:n]
|
||||
clientReceivedPackets = append(clientReceivedPackets, packetData(data))
|
||||
}
|
||||
}()
|
||||
|
||||
// send 3 packets
|
||||
start := time.Now()
|
||||
for i := 1; i <= 3; i++ {
|
||||
_, err := clientConn.Write(makePacket(protocol.PacketNumber(i), []byte("foobar"+strconv.Itoa(i))))
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
}
|
||||
// the packets should have arrived immediately at the server
|
||||
Eventually(func() []packetData { return serverReceivedPackets }).Should(HaveLen(3))
|
||||
expectDelay(start, delay, 0)
|
||||
Eventually(func() []packetData { return clientReceivedPackets }).Should(HaveLen(1))
|
||||
expectDelay(start, delay, 1)
|
||||
Eventually(func() []packetData { return clientReceivedPackets }).Should(HaveLen(2))
|
||||
expectDelay(start, delay, 2)
|
||||
Eventually(func() []packetData { return clientReceivedPackets }).Should(HaveLen(3))
|
||||
expectDelay(start, delay, 3)
|
||||
})
|
||||
})
|
||||
})
|
||||
})
|
Loading…
Add table
Add a link
Reference in a new issue