mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
proxy: rename to Proxy, refactor initialization (#4921)
* proxy: rename to Proxy, refactor initialization * improve documentation
This commit is contained in:
parent
f5145eb627
commit
79bae396b4
18 changed files with 274 additions and 303 deletions
|
@ -4,7 +4,6 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
|
||||||
"net"
|
"net"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"testing"
|
"testing"
|
||||||
|
@ -28,9 +27,10 @@ func TestConnectionCloseRetransmission(t *testing.T) {
|
||||||
|
|
||||||
var drop atomic.Bool
|
var drop atomic.Bool
|
||||||
dropped := make(chan []byte, 100)
|
dropped := make(chan []byte, 100)
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := &quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", server.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DelayPacket: func(quicproxy.Direction, []byte) time.Duration {
|
ServerAddr: server.Addr().(*net.UDPAddr),
|
||||||
|
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration {
|
||||||
return 5 * time.Millisecond // 10ms RTT
|
return 5 * time.Millisecond // 10ms RTT
|
||||||
},
|
},
|
||||||
DropPacket: func(dir quicproxy.Direction, b []byte) bool {
|
DropPacket: func(dir quicproxy.Direction, b []byte) bool {
|
||||||
|
@ -40,8 +40,8 @@ func TestConnectionCloseRetransmission(t *testing.T) {
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
||||||
|
|
|
@ -3,7 +3,6 @@ package self_test
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
|
||||||
mrand "math/rand/v2"
|
mrand "math/rand/v2"
|
||||||
"net"
|
"net"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
|
@ -138,8 +137,9 @@ func TestDatagramLoss(t *testing.T) {
|
||||||
defer server.Close()
|
defer server.Close()
|
||||||
|
|
||||||
var droppedIncoming, droppedOutgoing, total atomic.Int32
|
var droppedIncoming, droppedOutgoing, total atomic.Int32
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := &quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", server.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: server.Addr().(*net.UDPAddr),
|
||||||
DropPacket: func(dir quicproxy.Direction, packet []byte) bool {
|
DropPacket: func(dir quicproxy.Direction, packet []byte) bool {
|
||||||
if wire.IsLongHeaderPacket(packet[0]) { // don't drop Long Header packets
|
if wire.IsLongHeaderPacket(packet[0]) { // don't drop Long Header packets
|
||||||
return false
|
return false
|
||||||
|
@ -159,9 +159,9 @@ func TestDatagramLoss(t *testing.T) {
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
},
|
},
|
||||||
DelayPacket: func(dir quicproxy.Direction, packet []byte) time.Duration { return rtt / 2 },
|
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration { return rtt / 2 },
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), scaleDuration(numDatagrams*time.Millisecond))
|
ctx, cancel := context.WithTimeout(context.Background(), scaleDuration(numDatagrams*time.Millisecond))
|
||||||
|
|
|
@ -33,9 +33,10 @@ func TestDropTests(t *testing.T) {
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
var numDroppedPackets atomic.Int32
|
var numDroppedPackets atomic.Int32
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := &quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", ln.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DelayPacket: func(dir quicproxy.Direction, _ []byte) time.Duration { return rtt / 2 },
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
|
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
||||||
DropPacket: func(d quicproxy.Direction, b []byte) bool {
|
DropPacket: func(d quicproxy.Direction, b []byte) bool {
|
||||||
if !d.Is(direction) {
|
if !d.Is(direction) {
|
||||||
return false
|
return false
|
||||||
|
@ -49,8 +50,8 @@ func TestDropTests(t *testing.T) {
|
||||||
}
|
}
|
||||||
return drop
|
return drop
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
||||||
|
|
|
@ -2,7 +2,6 @@ package self_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"testing"
|
"testing"
|
||||||
|
@ -20,11 +19,12 @@ func TestEarlyData(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := &quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", ln.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
connChan := make(chan quic.EarlyConnection)
|
connChan := make(chan quic.EarlyConnection)
|
||||||
|
|
|
@ -43,12 +43,13 @@ func startDropTestListenerAndProxy(t *testing.T, rtt, timeout time.Duration, dro
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
t.Cleanup(func() { ln.Close() })
|
t.Cleanup(func() { ln.Close() })
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", ln.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DropPacket: dropCallback,
|
DropPacket: dropCallback,
|
||||||
DelayPacket: func(dir quicproxy.Direction, packet []byte) time.Duration { return rtt / 2 },
|
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
t.Cleanup(func() { proxy.Close() })
|
t.Cleanup(func() { proxy.Close() })
|
||||||
return ln, proxy.LocalAddr()
|
return ln, proxy.LocalAddr()
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,11 +17,12 @@ import (
|
||||||
func handshakeWithRTT(t *testing.T, serverAddr net.Addr, tlsConf *tls.Config, quicConf *quic.Config, rtt time.Duration) quic.Connection {
|
func handshakeWithRTT(t *testing.T, serverAddr net.Addr, tlsConf *tls.Config, quicConf *quic.Config, rtt time.Duration) quic.Connection {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: serverAddr.String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration { return rtt / 2 },
|
ServerAddr: serverAddr.(*net.UDPAddr),
|
||||||
})
|
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
||||||
require.NoError(t, err)
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
t.Cleanup(func() { proxy.Close() })
|
t.Cleanup(func() { proxy.Close() })
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 10*rtt)
|
ctx, cancel := context.WithTimeout(context.Background(), 10*rtt)
|
||||||
|
|
|
@ -359,11 +359,12 @@ func TestHandshakingConnectionsClosedOnServerShutdown(t *testing.T) {
|
||||||
ln, err := tr.Listen(tlsConf, getQuicConfig(nil))
|
ln, err := tr.Listen(tlsConf, getQuicConfig(nil))
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", ln.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
errChan := make(chan error, 1)
|
errChan := make(chan error, 1)
|
||||||
|
@ -549,14 +550,12 @@ func TestInvalidToken(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer server.Close()
|
defer server.Close()
|
||||||
|
|
||||||
serverPort := server.Addr().(*net.UDPAddr).Port
|
proxy := quicproxy.Proxy{
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
Conn: newUPDConnLocalhost(t),
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", serverPort),
|
ServerAddr: server.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(quicproxy.Direction, []byte) time.Duration {
|
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
||||||
return rtt / 2
|
}
|
||||||
},
|
require.NoError(t, proxy.Start())
|
||||||
})
|
|
||||||
require.NoError(t, err)
|
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
||||||
|
|
|
@ -848,16 +848,17 @@ func TestHTTP0RTT(t *testing.T) {
|
||||||
port := startHTTPServer(t, mux)
|
port := startHTTPServer(t, mux)
|
||||||
|
|
||||||
var num0RTTPackets atomic.Uint32
|
var num0RTTPackets atomic.Uint32
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: &net.UDPAddr{IP: net.IPv4(127, 0, 0, 1), Port: port},
|
||||||
DelayPacket: func(_ quicproxy.Direction, data []byte) time.Duration {
|
DelayPacket: func(_ quicproxy.Direction, data []byte) time.Duration {
|
||||||
if contains0RTTPacket(data) {
|
if contains0RTTPacket(data) {
|
||||||
num0RTTPackets.Add(1)
|
num0RTTPackets.Add(1)
|
||||||
}
|
}
|
||||||
return scaleDuration(25 * time.Millisecond)
|
return scaleDuration(25 * time.Millisecond)
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
tlsConf := getTLSClientConfigWithoutServerName()
|
tlsConf := getTLSClientConfigWithoutServerName()
|
||||||
|
|
|
@ -3,7 +3,6 @@ package self_test
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"math"
|
"math"
|
||||||
"net"
|
"net"
|
||||||
|
@ -204,13 +203,13 @@ func runMITMTest(t *testing.T, serverTr, clientTr *quic.Transport, rtt time.Dura
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
serverPort := ln.Addr().(*net.UDPAddr).Port
|
proxy := quicproxy.Proxy{
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
Conn: newUPDConnLocalhost(t),
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", serverPort),
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(dir quicproxy.Direction, b []byte) time.Duration { return rtt / 2 },
|
DelayPacket: func(_ quicproxy.Direction, b []byte) time.Duration { return rtt / 2 },
|
||||||
DropPacket: dropCb,
|
DropPacket: dropCb,
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), scaleDuration(time.Second))
|
ctx, cancel := context.WithTimeout(context.Background(), scaleDuration(time.Second))
|
||||||
|
@ -413,12 +412,12 @@ func runMITMTestSuccessful(t *testing.T, serverTransport, clientTransport *quic.
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
serverPort := ln.Addr().(*net.UDPAddr).Port
|
proxy := quicproxy.Proxy{
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
Conn: newUPDConnLocalhost(t),
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", serverPort),
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: delayCb,
|
DelayPacket: delayCb,
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), scaleDuration(50*time.Millisecond))
|
ctx, cancel := context.WithTimeout(context.Background(), scaleDuration(50*time.Millisecond))
|
||||||
|
|
|
@ -79,10 +79,10 @@ func TestPathMTUDiscovery(t *testing.T) {
|
||||||
var mx sync.Mutex
|
var mx sync.Mutex
|
||||||
var maxPacketSizeServer int
|
var maxPacketSizeServer int
|
||||||
var clientPacketSizes []int
|
var clientPacketSizes []int
|
||||||
serverPort := ln.Addr().(*net.UDPAddr).Port
|
proxy := &quicproxy.Proxy{
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
Conn: newUPDConnLocalhost(t),
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", serverPort),
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration { return rtt / 2 },
|
||||||
DropPacket: func(dir quicproxy.Direction, packet []byte) bool {
|
DropPacket: func(dir quicproxy.Direction, packet []byte) bool {
|
||||||
if len(packet) > mtu {
|
if len(packet) > mtu {
|
||||||
return true
|
return true
|
||||||
|
@ -99,8 +99,8 @@ func TestPathMTUDiscovery(t *testing.T) {
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
// Make sure to use v4-only socket here.
|
// Make sure to use v4-only socket here.
|
||||||
|
|
|
@ -4,6 +4,7 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"net"
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
@ -33,13 +34,14 @@ func TestACKBundling(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer server.Close()
|
defer server.Close()
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: server.Addr().String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DelayPacket: func(dir quicproxy.Direction, _ []byte) time.Duration {
|
ServerAddr: server.Addr().(*net.UDPAddr),
|
||||||
|
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration {
|
||||||
return 5 * time.Millisecond
|
return 5 * time.Millisecond
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
clientCounter, clientTracer := newPacketTracer()
|
clientCounter, clientTracer := newPacketTracer()
|
||||||
|
@ -161,13 +163,14 @@ func testConnAndStreamDataBlocked(t *testing.T, limitStream, limitConn bool) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: ln.Addr().String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DelayPacket: func(dir quicproxy.Direction, _ []byte) time.Duration {
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
|
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration {
|
||||||
return rtt / 2
|
return rtt / 2
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
counter, tracer := newPacketTracer()
|
counter, tracer := newPacketTracer()
|
||||||
|
|
|
@ -65,11 +65,12 @@ func TestDownloadWithFixedRTT(t *testing.T) {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", addr.(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
ServerAddr: &net.UDPAddr{IP: net.IPv4(127, 0, 0, 1), Port: addr.(*net.UDPAddr).Port},
|
||||||
})
|
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration { return rtt / 2 },
|
||||||
require.NoError(t, err)
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
t.Cleanup(func() { proxy.Close() })
|
t.Cleanup(func() { proxy.Close() })
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
||||||
|
@ -109,13 +110,14 @@ func TestDownloadWithReordering(t *testing.T) {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", addr.(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DelayPacket: func(quicproxy.Direction, []byte) time.Duration {
|
ServerAddr: &net.UDPAddr{IP: net.IPv4(127, 0, 0, 1), Port: addr.(*net.UDPAddr).Port},
|
||||||
|
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration {
|
||||||
return randomDuration(rtt/2, rtt*3/2) / 2
|
return randomDuration(rtt/2, rtt*3/2) / 2
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
t.Cleanup(func() { proxy.Close() })
|
t.Cleanup(func() { proxy.Close() })
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
||||||
|
|
|
@ -3,7 +3,6 @@ package self_test
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"crypto/rand"
|
"crypto/rand"
|
||||||
"fmt"
|
|
||||||
"net"
|
"net"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"testing"
|
"testing"
|
||||||
|
@ -37,7 +36,6 @@ func testStatelessReset(t *testing.T, connIDLen int) {
|
||||||
defer tr.Close()
|
defer tr.Close()
|
||||||
ln, err := tr.Listen(getTLSConfig(), getQuicConfig(nil))
|
ln, err := tr.Listen(getTLSConfig(), getQuicConfig(nil))
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
serverPort := ln.Addr().(*net.UDPAddr).Port
|
|
||||||
|
|
||||||
serverErr := make(chan error, 1)
|
serverErr := make(chan error, 1)
|
||||||
go func() {
|
go func() {
|
||||||
|
@ -60,12 +58,12 @@ func testStatelessReset(t *testing.T, connIDLen int) {
|
||||||
}()
|
}()
|
||||||
|
|
||||||
var drop atomic.Bool
|
var drop atomic.Bool
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", serverPort),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DropPacket: func(quicproxy.Direction, []byte) bool {
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
return drop.Load()
|
DropPacket: func(_ quicproxy.Direction, _ []byte) bool { return drop.Load() },
|
||||||
},
|
}
|
||||||
})
|
require.NoError(t, proxy.Start())
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
|
|
|
@ -111,13 +111,12 @@ func TestIdleTimeout(t *testing.T) {
|
||||||
defer server.Close()
|
defer server.Close()
|
||||||
|
|
||||||
var drop atomic.Bool
|
var drop atomic.Bool
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", server.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DropPacket: func(quicproxy.Direction, []byte) bool {
|
ServerAddr: server.Addr().(*net.UDPAddr),
|
||||||
return drop.Load()
|
DropPacket: func(_ quicproxy.Direction, _ []byte) bool { return drop.Load() },
|
||||||
},
|
}
|
||||||
})
|
require.NoError(t, proxy.Start())
|
||||||
require.NoError(t, err)
|
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
conn, err := quic.Dial(
|
conn, err := quic.Dial(
|
||||||
|
@ -179,11 +178,12 @@ func TestKeepAlive(t *testing.T) {
|
||||||
defer server.Close()
|
defer server.Close()
|
||||||
|
|
||||||
var drop atomic.Bool
|
var drop atomic.Bool
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", server.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DropPacket: func(quicproxy.Direction, []byte) bool { return drop.Load() },
|
ServerAddr: server.Addr().(*net.UDPAddr),
|
||||||
})
|
DropPacket: func(_ quicproxy.Direction, _ []byte) bool { return drop.Load() },
|
||||||
require.NoError(t, err)
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
||||||
|
@ -320,11 +320,12 @@ func TestTimeoutAfterSendingPacket(t *testing.T) {
|
||||||
defer server.Close()
|
defer server.Close()
|
||||||
|
|
||||||
var drop atomic.Bool
|
var drop atomic.Bool
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", server.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
DropPacket: func(d quicproxy.Direction, _ []byte) bool { return d == quicproxy.DirectionOutgoing && drop.Load() },
|
ServerAddr: server.Addr().(*net.UDPAddr),
|
||||||
})
|
DropPacket: func(_ quicproxy.Direction, _ []byte) bool { return drop.Load() },
|
||||||
require.NoError(t, err)
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
||||||
|
|
|
@ -21,19 +21,20 @@ import (
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
func runCountingProxyAndCount0RTTPackets(t *testing.T, serverPort int, rtt time.Duration) (*quicproxy.QuicProxy, *atomic.Uint32) {
|
func runCountingProxyAndCount0RTTPackets(t *testing.T, serverPort int, rtt time.Duration) (*quicproxy.Proxy, *atomic.Uint32) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
var num0RTTPackets atomic.Uint32
|
var num0RTTPackets atomic.Uint32
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := &quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", serverPort),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: &net.UDPAddr{IP: net.IPv4(127, 0, 0, 1), Port: serverPort},
|
||||||
DelayPacket: func(_ quicproxy.Direction, data []byte) time.Duration {
|
DelayPacket: func(_ quicproxy.Direction, data []byte) time.Duration {
|
||||||
if contains0RTTPacket(data) {
|
if contains0RTTPacket(data) {
|
||||||
num0RTTPackets.Add(1)
|
num0RTTPackets.Add(1)
|
||||||
}
|
}
|
||||||
return rtt / 2
|
return rtt / 2
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
t.Cleanup(func() { proxy.Close() })
|
t.Cleanup(func() { proxy.Close() })
|
||||||
return proxy, &num0RTTPackets
|
return proxy, &num0RTTPackets
|
||||||
}
|
}
|
||||||
|
@ -51,11 +52,12 @@ func dialAndReceiveTicket(
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := &quicproxy.Proxy{
|
||||||
RemoteAddr: ln.Addr().String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration { return rtt / 2 },
|
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration { return rtt / 2 },
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
clientTLSConf = getTLSClientConfig()
|
clientTLSConf = getTLSClientConfig()
|
||||||
|
@ -361,8 +363,9 @@ func Test0RTTDataLoss(t *testing.T) {
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
var num0RTTPackets, numDropped atomic.Uint32
|
var num0RTTPackets, numDropped atomic.Uint32
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", ln.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(_ quicproxy.Direction, data []byte) time.Duration { return rtt / 2 },
|
DelayPacket: func(_ quicproxy.Direction, data []byte) time.Duration { return rtt / 2 },
|
||||||
DropPacket: func(_ quicproxy.Direction, data []byte) bool {
|
DropPacket: func(_ quicproxy.Direction, data []byte) bool {
|
||||||
if !wire.IsLongHeaderPacket(data[0]) {
|
if !wire.IsLongHeaderPacket(data[0]) {
|
||||||
|
@ -380,8 +383,8 @@ func Test0RTTDataLoss(t *testing.T) {
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
transfer0RTTData(t, ln, proxy.LocalAddr(), clientConf, nil, PRData)
|
transfer0RTTData(t, ln, proxy.LocalAddr(), clientConf, nil, PRData)
|
||||||
|
@ -430,8 +433,9 @@ func Test0RTTRetransmitOnRetry(t *testing.T) {
|
||||||
}
|
}
|
||||||
var mutex sync.Mutex
|
var mutex sync.Mutex
|
||||||
var connIDToCounter []*connIDCounter
|
var connIDToCounter []*connIDCounter
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: fmt.Sprintf("localhost:%d", ln.Addr().(*net.UDPAddr).Port),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(dir quicproxy.Direction, data []byte) time.Duration {
|
DelayPacket: func(dir quicproxy.Direction, data []byte) time.Duration {
|
||||||
connID, err := wire.ParseConnectionID(data, 0)
|
connID, err := wire.ParseConnectionID(data, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -455,8 +459,8 @@ func Test0RTTRetransmitOnRetry(t *testing.T) {
|
||||||
}
|
}
|
||||||
return 2 * time.Millisecond
|
return 2 * time.Millisecond
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
transfer0RTTData(t, ln, proxy.LocalAddr(), clientConf, nil, GeneratePRData(5000)) // ~5 packets
|
transfer0RTTData(t, ln, proxy.LocalAddr(), clientConf, nil, GeneratePRData(5000)) // ~5 packets
|
||||||
|
@ -905,8 +909,9 @@ func Test0RTTPacketQueueing(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
proxy := quicproxy.Proxy{
|
||||||
RemoteAddr: ln.Addr().String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
DelayPacket: func(dir quicproxy.Direction, data []byte) time.Duration {
|
DelayPacket: func(dir quicproxy.Direction, data []byte) time.Duration {
|
||||||
// delay the client's Initial by 1 RTT
|
// delay the client's Initial by 1 RTT
|
||||||
if dir == quicproxy.DirectionIncoming && wire.IsLongHeaderPacket(data[0]) && data[0]&0x30>>4 == 0 {
|
if dir == quicproxy.DirectionIncoming && wire.IsLongHeaderPacket(data[0]) && data[0]&0x30>>4 == 0 {
|
||||||
|
@ -914,8 +919,8 @@ func Test0RTTPacketQueueing(t *testing.T) {
|
||||||
}
|
}
|
||||||
return rtt / 2
|
return rtt / 2
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, proxy.Start())
|
||||||
defer proxy.Close()
|
defer proxy.Close()
|
||||||
|
|
||||||
data := GeneratePRData(5000) // ~5 packets
|
data := GeneratePRData(5000) // ~5 packets
|
||||||
|
|
|
@ -113,100 +113,52 @@ func (d Direction) Is(dir Direction) bool {
|
||||||
// DropCallback is a callback that determines which packet gets dropped.
|
// DropCallback is a callback that determines which packet gets dropped.
|
||||||
type DropCallback func(dir Direction, packet []byte) bool
|
type DropCallback func(dir Direction, packet []byte) bool
|
||||||
|
|
||||||
// NoDropper doesn't drop packets.
|
|
||||||
var NoDropper DropCallback = func(Direction, []byte) bool {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// DelayCallback is a callback that determines how much delay to apply to a packet.
|
// DelayCallback is a callback that determines how much delay to apply to a packet.
|
||||||
type DelayCallback func(dir Direction, packet []byte) time.Duration
|
type DelayCallback func(dir Direction, packet []byte) time.Duration
|
||||||
|
|
||||||
// NoDelay doesn't apply a delay.
|
// Proxy is a QUIC proxy that can drop and delay packets.
|
||||||
var NoDelay DelayCallback = func(Direction, []byte) time.Duration {
|
type Proxy struct {
|
||||||
return 0
|
// Conn is the UDP socket that the proxy listens on for incoming packets
|
||||||
}
|
// from clients.
|
||||||
|
Conn *net.UDPConn
|
||||||
|
|
||||||
// Opts are proxy options.
|
// ServerAddr is the address of the server that the proxy forwards packets to.
|
||||||
type Opts struct {
|
ServerAddr *net.UDPAddr
|
||||||
// The address this proxy proxies packets to.
|
|
||||||
RemoteAddr string
|
// DropPacket is a callback that determines which packet gets dropped.
|
||||||
// DropPacket determines whether a packet gets dropped.
|
|
||||||
DropPacket DropCallback
|
DropPacket DropCallback
|
||||||
// DelayPacket determines how long a packet gets delayed. This allows
|
|
||||||
// simulating a connection with non-zero RTTs.
|
|
||||||
// Note that the RTT is the sum of the delay for the incoming and the outgoing packet.
|
|
||||||
DelayPacket DelayCallback
|
|
||||||
}
|
|
||||||
|
|
||||||
// QuicProxy is a QUIC proxy that can drop and delay packets.
|
// DelayPacket is a callback that determines how much delay to apply to a packet.
|
||||||
type QuicProxy struct {
|
DelayPacket DelayCallback
|
||||||
mutex sync.Mutex
|
|
||||||
|
|
||||||
closeChan chan struct{}
|
closeChan chan struct{}
|
||||||
|
logger utils.Logger
|
||||||
|
|
||||||
conn *net.UDPConn
|
// mapping from client addresses (as host:port) to connection
|
||||||
serverAddr *net.UDPAddr
|
mutex sync.Mutex
|
||||||
|
|
||||||
dropPacket DropCallback
|
|
||||||
delayPacket DelayCallback
|
|
||||||
|
|
||||||
// Mapping from client addresses (as host:port) to connection
|
|
||||||
clientDict map[string]*connection
|
clientDict map[string]*connection
|
||||||
|
|
||||||
logger utils.Logger
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewQuicProxy creates a new UDP proxy
|
// NewQuicProxy creates a new UDP proxy
|
||||||
func NewQuicProxy(local string, opts *Opts) (*QuicProxy, error) {
|
func (p *Proxy) Start() error {
|
||||||
if opts == nil {
|
p.clientDict = make(map[string]*connection)
|
||||||
opts = &Opts{}
|
p.closeChan = make(chan struct{})
|
||||||
|
p.logger = utils.DefaultLogger.WithPrefix("proxy")
|
||||||
|
|
||||||
|
if err := p.Conn.SetReadBuffer(protocol.DesiredReceiveBufferSize); err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
laddr, err := net.ResolveUDPAddr("udp", local)
|
if err := p.Conn.SetWriteBuffer(protocol.DesiredSendBufferSize); err != nil {
|
||||||
if err != nil {
|
return err
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
conn, err := net.ListenUDP("udp", laddr)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err := conn.SetReadBuffer(protocol.DesiredReceiveBufferSize); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err := conn.SetWriteBuffer(protocol.DesiredSendBufferSize); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
raddr, err := net.ResolveUDPAddr("udp", opts.RemoteAddr)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
packetDropper := NoDropper
|
p.logger.Debugf("Starting UDP Proxy %s <-> %s", p.Conn.LocalAddr(), p.ServerAddr)
|
||||||
if opts.DropPacket != nil {
|
|
||||||
packetDropper = opts.DropPacket
|
|
||||||
}
|
|
||||||
|
|
||||||
packetDelayer := NoDelay
|
|
||||||
if opts.DelayPacket != nil {
|
|
||||||
packetDelayer = opts.DelayPacket
|
|
||||||
}
|
|
||||||
|
|
||||||
p := QuicProxy{
|
|
||||||
clientDict: make(map[string]*connection),
|
|
||||||
conn: conn,
|
|
||||||
closeChan: make(chan struct{}),
|
|
||||||
serverAddr: raddr,
|
|
||||||
dropPacket: packetDropper,
|
|
||||||
delayPacket: packetDelayer,
|
|
||||||
logger: utils.DefaultLogger.WithPrefix("proxy"),
|
|
||||||
}
|
|
||||||
|
|
||||||
p.logger.Debugf("Starting UDP Proxy %s <-> %s", conn.LocalAddr(), raddr)
|
|
||||||
go p.runProxy()
|
go p.runProxy()
|
||||||
return &p, nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close stops the UDP Proxy
|
// Close stops the UDP Proxy
|
||||||
func (p *QuicProxy) Close() error {
|
func (p *Proxy) Close() error {
|
||||||
p.mutex.Lock()
|
p.mutex.Lock()
|
||||||
defer p.mutex.Unlock()
|
defer p.mutex.Unlock()
|
||||||
|
|
||||||
|
@ -218,16 +170,14 @@ func (p *QuicProxy) Close() error {
|
||||||
c.Incoming.Close()
|
c.Incoming.Close()
|
||||||
c.Outgoing.Close()
|
c.Outgoing.Close()
|
||||||
}
|
}
|
||||||
return p.conn.Close()
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// LocalAddr is the address the proxy is listening on.
|
// LocalAddr is the address the proxy is listening on.
|
||||||
func (p *QuicProxy) LocalAddr() net.Addr {
|
func (p *Proxy) LocalAddr() net.Addr { return p.Conn.LocalAddr() }
|
||||||
return p.conn.LocalAddr()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *QuicProxy) newConnection(cliAddr *net.UDPAddr) (*connection, error) {
|
func (p *Proxy) newConnection(cliAddr *net.UDPAddr) (*connection, error) {
|
||||||
conn, err := net.DialUDP("udp", nil, p.serverAddr)
|
conn, err := net.DialUDP("udp", nil, p.ServerAddr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -247,10 +197,10 @@ func (p *QuicProxy) newConnection(cliAddr *net.UDPAddr) (*connection, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// runProxy listens on the proxy address and handles incoming packets.
|
// runProxy listens on the proxy address and handles incoming packets.
|
||||||
func (p *QuicProxy) runProxy() error {
|
func (p *Proxy) runProxy() error {
|
||||||
for {
|
for {
|
||||||
buffer := make([]byte, protocol.MaxPacketBufferSize)
|
buffer := make([]byte, protocol.MaxPacketBufferSize)
|
||||||
n, cliaddr, err := p.conn.ReadFromUDP(buffer)
|
n, cliaddr, err := p.Conn.ReadFromUDP(buffer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -272,14 +222,17 @@ func (p *QuicProxy) runProxy() error {
|
||||||
}
|
}
|
||||||
p.mutex.Unlock()
|
p.mutex.Unlock()
|
||||||
|
|
||||||
if p.dropPacket(DirectionIncoming, raw) {
|
if p.DropPacket != nil && p.DropPacket(DirectionIncoming, raw) {
|
||||||
if p.logger.Debug() {
|
if p.logger.Debug() {
|
||||||
p.logger.Debugf("dropping incoming packet(%d bytes)", n)
|
p.logger.Debugf("dropping incoming packet(%d bytes)", n)
|
||||||
}
|
}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
delay := p.delayPacket(DirectionIncoming, raw)
|
var delay time.Duration
|
||||||
|
if p.DelayPacket != nil {
|
||||||
|
delay = p.DelayPacket(DirectionIncoming, raw)
|
||||||
|
}
|
||||||
if delay == 0 {
|
if delay == 0 {
|
||||||
if p.logger.Debug() {
|
if p.logger.Debug() {
|
||||||
p.logger.Debugf("forwarding incoming packet (%d bytes) to %s", len(raw), conn.ServerConn.RemoteAddr())
|
p.logger.Debugf("forwarding incoming packet (%d bytes) to %s", len(raw), conn.ServerConn.RemoteAddr())
|
||||||
|
@ -298,7 +251,7 @@ func (p *QuicProxy) runProxy() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
// runConnection handles packets from server to a single client
|
// runConnection handles packets from server to a single client
|
||||||
func (p *QuicProxy) runOutgoingConnection(conn *connection) error {
|
func (p *Proxy) runOutgoingConnection(conn *connection) error {
|
||||||
outgoingPackets := make(chan packetEntry, 10)
|
outgoingPackets := make(chan packetEntry, 10)
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
|
@ -309,19 +262,22 @@ func (p *QuicProxy) runOutgoingConnection(conn *connection) error {
|
||||||
}
|
}
|
||||||
raw := buffer[0:n]
|
raw := buffer[0:n]
|
||||||
|
|
||||||
if p.dropPacket(DirectionOutgoing, raw) {
|
if p.DropPacket != nil && p.DropPacket(DirectionOutgoing, raw) {
|
||||||
if p.logger.Debug() {
|
if p.logger.Debug() {
|
||||||
p.logger.Debugf("dropping outgoing packet(%d bytes)", n)
|
p.logger.Debugf("dropping outgoing packet(%d bytes)", n)
|
||||||
}
|
}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
delay := p.delayPacket(DirectionOutgoing, raw)
|
var delay time.Duration
|
||||||
|
if p.DelayPacket != nil {
|
||||||
|
delay = p.DelayPacket(DirectionOutgoing, raw)
|
||||||
|
}
|
||||||
if delay == 0 {
|
if delay == 0 {
|
||||||
if p.logger.Debug() {
|
if p.logger.Debug() {
|
||||||
p.logger.Debugf("forwarding outgoing packet (%d bytes) to %s", len(raw), conn.ClientAddr)
|
p.logger.Debugf("forwarding outgoing packet (%d bytes) to %s", len(raw), conn.ClientAddr)
|
||||||
}
|
}
|
||||||
if _, err := p.conn.WriteToUDP(raw, conn.ClientAddr); err != nil {
|
if _, err := p.Conn.WriteToUDP(raw, conn.ClientAddr); err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -342,14 +298,14 @@ func (p *QuicProxy) runOutgoingConnection(conn *connection) error {
|
||||||
conn.Outgoing.Add(e)
|
conn.Outgoing.Add(e)
|
||||||
case <-conn.Outgoing.Timer():
|
case <-conn.Outgoing.Timer():
|
||||||
conn.Outgoing.SetTimerRead()
|
conn.Outgoing.SetTimerRead()
|
||||||
if _, err := p.conn.WriteTo(conn.Outgoing.Get(), conn.ClientAddr); err != nil {
|
if _, err := p.Conn.WriteTo(conn.Outgoing.Get(), conn.ClientAddr); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *QuicProxy) runIncomingConnection(conn *connection) error {
|
func (p *Proxy) runIncomingConnection(conn *connection) error {
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case <-p.closeChan:
|
case <-p.closeChan:
|
||||||
|
|
|
@ -1,11 +1,8 @@
|
||||||
package quicproxy
|
package quicproxy
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"net"
|
"net"
|
||||||
"runtime/pprof"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
@ -16,6 +13,14 @@ import (
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func newUPDConnLocalhost(t testing.TB) *net.UDPConn {
|
||||||
|
t.Helper()
|
||||||
|
conn, err := net.ListenUDP("udp", &net.UDPAddr{IP: net.IPv4(127, 0, 0, 1), Port: 0})
|
||||||
|
require.NoError(t, err)
|
||||||
|
t.Cleanup(func() { conn.Close() })
|
||||||
|
return conn
|
||||||
|
}
|
||||||
|
|
||||||
type packetData []byte
|
type packetData []byte
|
||||||
|
|
||||||
func makePacket(t *testing.T, p protocol.PacketNumber, payload []byte) []byte {
|
func makePacket(t *testing.T, p protocol.PacketNumber, payload []byte) []byte {
|
||||||
|
@ -47,37 +52,6 @@ func readPacketNumber(t *testing.T, b []byte) protocol.PacketNumber {
|
||||||
return extHdr.PacketNumber
|
return extHdr.PacketNumber
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestProxyShutdown(t *testing.T) {
|
|
||||||
isProxyRunning := func() bool {
|
|
||||||
var b bytes.Buffer
|
|
||||||
pprof.Lookup("goroutine").WriteTo(&b, 1)
|
|
||||||
return strings.Contains(b.String(), "proxy.(*QuicProxy).runProxy")
|
|
||||||
}
|
|
||||||
|
|
||||||
proxy, err := NewQuicProxy("localhost:0", nil)
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.Eventually(t, func() bool { return isProxyRunning() }, time.Second, 10*time.Millisecond)
|
|
||||||
|
|
||||||
conn, err := net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
|
||||||
require.NoError(t, err)
|
|
||||||
_, err = conn.Write(makePacket(t, 1, []byte("foobar")))
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
require.NoError(t, proxy.Close())
|
|
||||||
|
|
||||||
// check that the proxy port is not in use anymore
|
|
||||||
// sometimes it takes a while for the OS to free the port
|
|
||||||
require.Eventually(t, func() bool {
|
|
||||||
ln, err := net.ListenUDP("udp", proxy.LocalAddr().(*net.UDPAddr))
|
|
||||||
if err != nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
ln.Close()
|
|
||||||
return true
|
|
||||||
}, time.Second, 10*time.Millisecond)
|
|
||||||
require.Eventually(t, func() bool { return !isProxyRunning() }, time.Second, 10*time.Millisecond)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set up a dumb UDP server.
|
// Set up a dumb UDP server.
|
||||||
// In production this would be a QUIC server.
|
// In production this would be a QUIC server.
|
||||||
func runServer(t *testing.T) (*net.UDPAddr, chan packetData) {
|
func runServer(t *testing.T) (*net.UDPAddr, chan packetData) {
|
||||||
|
@ -116,25 +90,19 @@ func runServer(t *testing.T) (*net.UDPAddr, chan packetData) {
|
||||||
return serverConn.LocalAddr().(*net.UDPAddr), serverReceivedPackets
|
return serverConn.LocalAddr().(*net.UDPAddr), serverReceivedPackets
|
||||||
}
|
}
|
||||||
|
|
||||||
func startProxy(t *testing.T, opts *Opts) (clientConn *net.UDPConn) {
|
|
||||||
proxy, err := NewQuicProxy("localhost:0", opts)
|
|
||||||
require.NoError(t, err)
|
|
||||||
clientConn, err = net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
t.Cleanup(func() {
|
|
||||||
require.NoError(t, proxy.Close())
|
|
||||||
require.NoError(t, clientConn.Close())
|
|
||||||
})
|
|
||||||
return clientConn
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestProxyyingBackAndForth(t *testing.T) {
|
func TestProxyyingBackAndForth(t *testing.T) {
|
||||||
serverAddr, _ := runServer(t)
|
serverAddr, _ := runServer(t)
|
||||||
clientConn := startProxy(t, &Opts{RemoteAddr: serverAddr.String()})
|
proxy := Proxy{
|
||||||
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: serverAddr,
|
||||||
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
|
defer proxy.Close()
|
||||||
|
clientConn, err := net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
// send the first packet
|
// send the first packet
|
||||||
_, err := clientConn.Write(makePacket(t, 1, []byte("foobar")))
|
_, err = clientConn.Write(makePacket(t, 1, []byte("foobar")))
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
// send the second packet
|
// send the second packet
|
||||||
_, err = clientConn.Write(makePacket(t, 2, []byte("decafbad")))
|
_, err = clientConn.Write(makePacket(t, 2, []byte("decafbad")))
|
||||||
|
@ -153,15 +121,20 @@ func TestDropIncomingPackets(t *testing.T) {
|
||||||
const numPackets = 6
|
const numPackets = 6
|
||||||
serverAddr, serverReceivedPackets := runServer(t)
|
serverAddr, serverReceivedPackets := runServer(t)
|
||||||
var counter atomic.Int32
|
var counter atomic.Int32
|
||||||
clientConn := startProxy(t, &Opts{
|
proxy := Proxy{
|
||||||
RemoteAddr: serverAddr.String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: serverAddr,
|
||||||
DropPacket: func(d Direction, _ []byte) bool {
|
DropPacket: func(d Direction, _ []byte) bool {
|
||||||
if d != DirectionIncoming {
|
if d != DirectionIncoming {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return counter.Add(1)%2 == 1
|
return counter.Add(1)%2 == 1
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
|
defer proxy.Close()
|
||||||
|
clientConn, err := net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
for i := 1; i <= numPackets; i++ {
|
for i := 1; i <= numPackets; i++ {
|
||||||
_, err := clientConn.Write(makePacket(t, protocol.PacketNumber(i), []byte("foobar"+strconv.Itoa(i))))
|
_, err := clientConn.Write(makePacket(t, protocol.PacketNumber(i), []byte("foobar"+strconv.Itoa(i))))
|
||||||
|
@ -186,15 +159,20 @@ func TestDropOutgoingPackets(t *testing.T) {
|
||||||
const numPackets = 6
|
const numPackets = 6
|
||||||
serverAddr, serverReceivedPackets := runServer(t)
|
serverAddr, serverReceivedPackets := runServer(t)
|
||||||
var counter atomic.Int32
|
var counter atomic.Int32
|
||||||
clientConn := startProxy(t, &Opts{
|
proxy := Proxy{
|
||||||
RemoteAddr: serverAddr.String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: serverAddr,
|
||||||
DropPacket: func(d Direction, _ []byte) bool {
|
DropPacket: func(d Direction, _ []byte) bool {
|
||||||
if d != DirectionOutgoing {
|
if d != DirectionOutgoing {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return counter.Add(1)%2 == 1
|
return counter.Add(1)%2 == 1
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
|
defer proxy.Close()
|
||||||
|
clientConn, err := net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
clientReceivedPackets := make(chan struct{}, numPackets)
|
clientReceivedPackets := make(chan struct{}, numPackets)
|
||||||
// receive the packets echoed by the server on client side
|
// receive the packets echoed by the server on client side
|
||||||
|
@ -234,19 +212,24 @@ func TestDelayIncomingPackets(t *testing.T) {
|
||||||
const delay = 200 * time.Millisecond
|
const delay = 200 * time.Millisecond
|
||||||
serverAddr, serverReceivedPackets := runServer(t)
|
serverAddr, serverReceivedPackets := runServer(t)
|
||||||
var counter atomic.Int32
|
var counter atomic.Int32
|
||||||
clientConn := startProxy(t, &Opts{
|
proxy := Proxy{
|
||||||
RemoteAddr: serverAddr.String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
// delay packet 1 by 200 ms
|
ServerAddr: serverAddr,
|
||||||
// delay packet 2 by 400 ms
|
|
||||||
// ...
|
|
||||||
DelayPacket: func(d Direction, _ []byte) time.Duration {
|
DelayPacket: func(d Direction, _ []byte) time.Duration {
|
||||||
|
// delay packet 1 by 200 ms
|
||||||
|
// delay packet 2 by 400 ms
|
||||||
|
// ...
|
||||||
if d == DirectionOutgoing {
|
if d == DirectionOutgoing {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
p := counter.Add(1)
|
p := counter.Add(1)
|
||||||
return time.Duration(p) * delay
|
return time.Duration(p) * delay
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
|
defer proxy.Close()
|
||||||
|
clientConn, err := net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
for i := 1; i <= numPackets; i++ {
|
for i := 1; i <= numPackets; i++ {
|
||||||
|
@ -276,19 +259,24 @@ func TestPacketReordering(t *testing.T) {
|
||||||
|
|
||||||
serverAddr, serverReceivedPackets := runServer(t)
|
serverAddr, serverReceivedPackets := runServer(t)
|
||||||
var counter atomic.Int32
|
var counter atomic.Int32
|
||||||
clientConn := startProxy(t, &Opts{
|
proxy := Proxy{
|
||||||
RemoteAddr: serverAddr.String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
// delay packet 1 by 600 ms
|
ServerAddr: serverAddr,
|
||||||
// delay packet 2 by 400 ms
|
|
||||||
// delay packet 3 by 200 ms
|
|
||||||
DelayPacket: func(d Direction, _ []byte) time.Duration {
|
DelayPacket: func(d Direction, _ []byte) time.Duration {
|
||||||
|
// delay packet 1 by 600 ms
|
||||||
|
// delay packet 2 by 400 ms
|
||||||
|
// delay packet 3 by 200 ms
|
||||||
if d == DirectionOutgoing {
|
if d == DirectionOutgoing {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
p := counter.Add(1)
|
p := counter.Add(1)
|
||||||
return 600*time.Millisecond - time.Duration(p-1)*delay
|
return 600*time.Millisecond - time.Duration(p-1)*delay
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
|
defer proxy.Close()
|
||||||
|
clientConn, err := net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
// send 3 packets
|
// send 3 packets
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
|
@ -310,15 +298,20 @@ func TestPacketReordering(t *testing.T) {
|
||||||
|
|
||||||
func TestConstantDelay(t *testing.T) { // no reordering expected here
|
func TestConstantDelay(t *testing.T) { // no reordering expected here
|
||||||
serverAddr, serverReceivedPackets := runServer(t)
|
serverAddr, serverReceivedPackets := runServer(t)
|
||||||
clientConn := startProxy(t, &Opts{
|
proxy := Proxy{
|
||||||
RemoteAddr: serverAddr.String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
|
ServerAddr: serverAddr,
|
||||||
DelayPacket: func(d Direction, _ []byte) time.Duration {
|
DelayPacket: func(d Direction, _ []byte) time.Duration {
|
||||||
if d == DirectionOutgoing {
|
if d == DirectionOutgoing {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
return 100 * time.Millisecond
|
return 100 * time.Millisecond
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
|
defer proxy.Close()
|
||||||
|
clientConn, err := net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
// send 100 packets
|
// send 100 packets
|
||||||
for i := 0; i < 100; i++ {
|
for i := 0; i < 100; i++ {
|
||||||
|
@ -343,19 +336,24 @@ func TestDelayOutgoingPackets(t *testing.T) {
|
||||||
|
|
||||||
serverAddr, serverReceivedPackets := runServer(t)
|
serverAddr, serverReceivedPackets := runServer(t)
|
||||||
var counter atomic.Int32
|
var counter atomic.Int32
|
||||||
clientConn := startProxy(t, &Opts{
|
proxy := Proxy{
|
||||||
RemoteAddr: serverAddr.String(),
|
Conn: newUPDConnLocalhost(t),
|
||||||
// delay packet 1 by 200 ms
|
ServerAddr: serverAddr,
|
||||||
// delay packet 2 by 400 ms
|
|
||||||
// ...
|
|
||||||
DelayPacket: func(d Direction, _ []byte) time.Duration {
|
DelayPacket: func(d Direction, _ []byte) time.Duration {
|
||||||
|
// delay packet 1 by 200 ms
|
||||||
|
// delay packet 2 by 400 ms
|
||||||
|
// ...
|
||||||
if d == DirectionIncoming {
|
if d == DirectionIncoming {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
p := counter.Add(1)
|
p := counter.Add(1)
|
||||||
return time.Duration(p) * delay
|
return time.Duration(p) * delay
|
||||||
},
|
},
|
||||||
})
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
|
defer proxy.Close()
|
||||||
|
clientConn, err := net.DialUDP("udp", nil, proxy.LocalAddr().(*net.UDPAddr))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
clientReceivedPackets := make(chan packetData, numPackets)
|
clientReceivedPackets := make(chan packetData, numPackets)
|
||||||
// receive the packets echoed by the server on client side
|
// receive the packets echoed by the server on client side
|
||||||
|
|
|
@ -2,6 +2,7 @@ package versionnegotiation
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"net"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -32,12 +33,17 @@ func TestVersionNegotiationFailure(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer ln.Close()
|
defer ln.Close()
|
||||||
|
|
||||||
// start the proxy
|
proxyConn, err := net.ListenUDP("udp", &net.UDPAddr{IP: net.IPv4(127, 0, 0, 1), Port: 0})
|
||||||
proxy, err := quicproxy.NewQuicProxy("localhost:0", &quicproxy.Opts{
|
|
||||||
RemoteAddr: ln.Addr().String(),
|
|
||||||
DelayPacket: func(_ quicproxy.Direction, _ []byte) time.Duration { return rtt / 2 },
|
|
||||||
})
|
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
defer proxyConn.Close()
|
||||||
|
// start the proxy
|
||||||
|
proxy := quicproxy.Proxy{
|
||||||
|
Conn: proxyConn,
|
||||||
|
ServerAddr: ln.Addr().(*net.UDPAddr),
|
||||||
|
DelayPacket: func(quicproxy.Direction, []byte) time.Duration { return rtt / 2 },
|
||||||
|
}
|
||||||
|
require.NoError(t, proxy.Start())
|
||||||
|
defer proxy.Close()
|
||||||
|
|
||||||
startTime := time.Now()
|
startTime := time.Now()
|
||||||
_, err = quic.DialAddr(
|
_, err = quic.DialAddr(
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue