mirror of
https://github.com/XTLS/Xray-core.git
synced 2026-05-30 17:51:51 +00:00
633 lines
15 KiB
Go
633 lines
15 KiB
Go
package finalmask_test
|
|
|
|
import (
|
|
"bytes"
|
|
"io"
|
|
"net"
|
|
"sync/atomic"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/xtls/xray-core/proxy"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/header/custom"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/header/dns"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/header/srtp"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/header/utp"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/header/wechat"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/header/wireguard"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/mkcp/aes128gcm"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/mkcp/original"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/salamander"
|
|
"github.com/xtls/xray-core/transport/internet/finalmask/sudoku"
|
|
)
|
|
|
|
func mustSendRecv(
|
|
t *testing.T,
|
|
from net.PacketConn,
|
|
to net.PacketConn,
|
|
msg []byte,
|
|
) {
|
|
t.Helper()
|
|
|
|
go func() {
|
|
_, err := from.WriteTo(msg, to.LocalAddr())
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}()
|
|
|
|
buf := make([]byte, 1024)
|
|
n, _, err := to.ReadFrom(buf)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if n != len(msg) {
|
|
t.Fatalf("unexpected size: %d", n)
|
|
}
|
|
|
|
if !bytes.Equal(buf[:n], msg) {
|
|
t.Fatalf("unexpected data")
|
|
}
|
|
}
|
|
|
|
type layerMask struct {
|
|
name string
|
|
mask finalmask.Udpmask
|
|
layers int
|
|
}
|
|
|
|
type countingConn struct {
|
|
net.Conn
|
|
written atomic.Int64
|
|
}
|
|
|
|
func (c *countingConn) Write(p []byte) (int, error) {
|
|
n, err := c.Conn.Write(p)
|
|
c.written.Add(int64(n))
|
|
return n, err
|
|
}
|
|
|
|
func (c *countingConn) Written() int64 {
|
|
return c.written.Load()
|
|
}
|
|
|
|
func TestPacketConnReadWrite(t *testing.T) {
|
|
cases := []layerMask{
|
|
{
|
|
name: "aes128gcm",
|
|
mask: &aes128gcm.Config{Password: "123"},
|
|
layers: 2,
|
|
},
|
|
{
|
|
name: "original",
|
|
mask: &original.Config{},
|
|
layers: 2,
|
|
},
|
|
{
|
|
name: "dns",
|
|
mask: &dns.Config{Domain: "www.baidu.com"},
|
|
layers: 2,
|
|
},
|
|
{
|
|
name: "srtp",
|
|
mask: &srtp.Config{},
|
|
layers: 2,
|
|
},
|
|
{
|
|
name: "utp",
|
|
mask: &utp.Config{},
|
|
layers: 2,
|
|
},
|
|
{
|
|
name: "wechat",
|
|
mask: &wechat.Config{},
|
|
layers: 2,
|
|
},
|
|
{
|
|
name: "wireguard",
|
|
mask: &wireguard.Config{},
|
|
layers: 2,
|
|
},
|
|
{
|
|
name: "salamander",
|
|
mask: &salamander.Config{Password: "1234"},
|
|
layers: 2,
|
|
},
|
|
{
|
|
name: "sudoku-prefer-ascii",
|
|
mask: &sudoku.Config{
|
|
Password: "sudoku-mask",
|
|
Ascii: "prefer_ascii",
|
|
},
|
|
layers: 1,
|
|
},
|
|
{
|
|
name: "sudoku-custom-table",
|
|
mask: &sudoku.Config{
|
|
Password: "sudoku-mask",
|
|
Ascii: "prefer_entropy",
|
|
CustomTable: "xpxvvpvv",
|
|
},
|
|
layers: 1,
|
|
},
|
|
{
|
|
name: "sudoku-custom-tables",
|
|
mask: &sudoku.Config{
|
|
Password: "sudoku-mask",
|
|
Ascii: "prefer_entropy",
|
|
CustomTables: []string{"xpxvvpvv", "vxpvxvvp"},
|
|
},
|
|
layers: 1,
|
|
},
|
|
{
|
|
name: "custom",
|
|
mask: &custom.UDPConfig{
|
|
Client: []*custom.UDPItem{
|
|
{
|
|
Packet: []byte{1},
|
|
},
|
|
{
|
|
Rand: 1,
|
|
},
|
|
},
|
|
Server: []*custom.UDPItem{
|
|
{
|
|
Packet: []byte{1},
|
|
},
|
|
{
|
|
Rand: 1,
|
|
},
|
|
},
|
|
},
|
|
layers: 1,
|
|
},
|
|
{
|
|
name: "salamander-single",
|
|
mask: &salamander.Config{Password: "1234"},
|
|
layers: 1,
|
|
},
|
|
}
|
|
|
|
for _, c := range cases {
|
|
t.Run(c.name, func(t *testing.T) {
|
|
mask := c.mask
|
|
layers := c.layers
|
|
if layers <= 0 {
|
|
layers = 1
|
|
}
|
|
masks := make([]finalmask.Udpmask, 0, layers)
|
|
for i := 0; i < layers; i++ {
|
|
masks = append(masks, mask)
|
|
}
|
|
maskManager := finalmask.NewUdpmaskManager(masks)
|
|
|
|
client, err := net.ListenPacket("udp", "127.0.0.1:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
client, err = maskManager.WrapPacketConnClient(client)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
server, err := net.ListenPacket("udp", "127.0.0.1:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
server, err = maskManager.WrapPacketConnServer(server)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
_ = client.SetDeadline(time.Now().Add(time.Second))
|
|
_ = server.SetDeadline(time.Now().Add(time.Second))
|
|
|
|
mustSendRecv(t, client, server, []byte("client -> server"))
|
|
mustSendRecv(t, server, client, []byte("server -> client"))
|
|
|
|
mustSendRecv(t, client, server, []byte{})
|
|
mustSendRecv(t, server, client, []byte{})
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSudokuBDD(t *testing.T) {
|
|
t.Run("GivenSudokuTCPMask_WhenRoundTripWithAsciiPreference_ThenPayloadMatches", func(t *testing.T) {
|
|
cfg := &sudoku.Config{
|
|
Password: "sudoku-tcp",
|
|
Ascii: "prefer_ascii",
|
|
}
|
|
|
|
clientRaw, serverRaw := net.Pipe()
|
|
defer clientRaw.Close()
|
|
defer serverRaw.Close()
|
|
|
|
clientConn, err := cfg.WrapConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
serverConn, err := cfg.WrapConnServer(serverRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
send := bytes.Repeat([]byte("client->server"), 1024)
|
|
recv := make([]byte, len(send))
|
|
|
|
writeErr := make(chan error, 1)
|
|
go func() {
|
|
_, wErr := clientConn.Write(send)
|
|
writeErr <- wErr
|
|
}()
|
|
|
|
if _, err := io.ReadFull(serverConn, recv); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := <-writeErr; err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !bytes.Equal(send, recv) {
|
|
t.Fatal("tcp sudoku payload mismatch")
|
|
}
|
|
})
|
|
|
|
t.Run("GivenSudokuTCPMask_WhenRoundTrip_ThenBothDirectionsMatch", func(t *testing.T) {
|
|
cfg := &sudoku.Config{
|
|
Password: "sudoku-packed",
|
|
Ascii: "prefer_ascii",
|
|
PaddingMin: 0,
|
|
PaddingMax: 0,
|
|
}
|
|
|
|
clientRaw, serverRaw := net.Pipe()
|
|
defer clientRaw.Close()
|
|
defer serverRaw.Close()
|
|
|
|
clientConn, err := cfg.WrapConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
serverConn, err := cfg.WrapConnServer(serverRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
clientToServer := bytes.Repeat([]byte("client-packed->server"), 257)
|
|
serverToClient := bytes.Repeat([]byte("server-packed->client"), 263)
|
|
|
|
c2sRecv := make([]byte, len(clientToServer))
|
|
c2sErr := make(chan error, 1)
|
|
go func() {
|
|
_, err := clientConn.Write(clientToServer)
|
|
c2sErr <- err
|
|
}()
|
|
if _, err := io.ReadFull(serverConn, c2sRecv); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := <-c2sErr; err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !bytes.Equal(clientToServer, c2sRecv) {
|
|
t.Fatal("tcp client->server payload mismatch")
|
|
}
|
|
|
|
s2cRecv := make([]byte, len(serverToClient))
|
|
s2cErr := make(chan error, 1)
|
|
go func() {
|
|
_, err := serverConn.Write(serverToClient)
|
|
s2cErr <- err
|
|
}()
|
|
if _, err := io.ReadFull(clientConn, s2cRecv); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := <-s2cErr; err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !bytes.Equal(serverToClient, s2cRecv) {
|
|
t.Fatal("tcp server->client payload mismatch")
|
|
}
|
|
})
|
|
|
|
t.Run("GivenSudokuTCPMask_WhenServerWritesDownlink_ThenWireBytesAreReduced", func(t *testing.T) {
|
|
payload := bytes.Repeat([]byte("0123456789abcdef"), 192) // 3072 bytes, divisible by 3.
|
|
|
|
countWireBytes := func(wrapServer func(net.Conn, *sudoku.Config) (net.Conn, error), cfg *sudoku.Config) int64 {
|
|
t.Helper()
|
|
|
|
clientRaw, serverRaw := net.Pipe()
|
|
watchedServerRaw := &countingConn{Conn: serverRaw}
|
|
|
|
clientConn, err := cfg.WrapConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
serverConn, err := wrapServer(watchedServerRaw, cfg)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
readErr := make(chan error, 1)
|
|
go func() {
|
|
_, err := io.CopyN(io.Discard, clientConn, int64(len(payload)))
|
|
readErr <- err
|
|
}()
|
|
|
|
if _, err := serverConn.Write(payload); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := <-readErr; err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
_ = clientConn.Close()
|
|
_ = serverConn.Close()
|
|
return watchedServerRaw.Written()
|
|
}
|
|
|
|
pureUplinkPackedDownlink := &sudoku.Config{
|
|
Password: "sudoku-bandwidth",
|
|
Ascii: "prefer_entropy",
|
|
PaddingMin: 0,
|
|
PaddingMax: 0,
|
|
}
|
|
packedDownlinkBytes := countWireBytes(func(raw net.Conn, cfg *sudoku.Config) (net.Conn, error) {
|
|
return cfg.WrapConnServer(raw)
|
|
}, pureUplinkPackedDownlink)
|
|
legacyPureBytes := countWireBytes(func(raw net.Conn, cfg *sudoku.Config) (net.Conn, error) {
|
|
return sudoku.NewTCPConn(raw, cfg)
|
|
}, pureUplinkPackedDownlink)
|
|
|
|
if packedDownlinkBytes >= legacyPureBytes {
|
|
t.Fatalf("expected default packed downlink bytes < legacy pure bytes, got packed=%d pure=%d", packedDownlinkBytes, legacyPureBytes)
|
|
}
|
|
})
|
|
|
|
t.Run("GivenSudokuMultiTableTCPMask_WhenRoundTrip_ThenPayloadMatches", func(t *testing.T) {
|
|
cfg := &sudoku.Config{
|
|
Password: "sudoku-multi-tcp",
|
|
Ascii: "prefer_entropy",
|
|
CustomTables: []string{"xpxvvpvv", "vxpvxvvp"},
|
|
}
|
|
|
|
clientRaw, serverRaw := net.Pipe()
|
|
defer clientRaw.Close()
|
|
defer serverRaw.Close()
|
|
|
|
clientConn, err := cfg.WrapConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
serverConn, err := cfg.WrapConnServer(serverRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
send := bytes.Repeat([]byte("rotate-table"), 513)
|
|
recv := make([]byte, len(send))
|
|
|
|
writeErr := make(chan error, 1)
|
|
go func() {
|
|
_, wErr := clientConn.Write(send)
|
|
writeErr <- wErr
|
|
}()
|
|
|
|
if _, err := io.ReadFull(serverConn, recv); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := <-writeErr; err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !bytes.Equal(send, recv) {
|
|
t.Fatal("multi-table tcp sudoku payload mismatch")
|
|
}
|
|
})
|
|
|
|
t.Run("GivenSudokuMultiTableTCPMask_WhenPackedDownlink_ThenPayloadMatches", func(t *testing.T) {
|
|
cfg := &sudoku.Config{
|
|
Password: "sudoku-multi-packed",
|
|
Ascii: "prefer_entropy",
|
|
CustomTables: []string{"xpxvvpvv", "vxpvxvvp"},
|
|
PaddingMin: 0,
|
|
PaddingMax: 0,
|
|
}
|
|
|
|
clientRaw, serverRaw := net.Pipe()
|
|
defer clientRaw.Close()
|
|
defer serverRaw.Close()
|
|
|
|
clientConn, err := cfg.WrapConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
serverConn, err := cfg.WrapConnServer(serverRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
send := bytes.Repeat([]byte("packed-rotate"), 257)
|
|
recv := make([]byte, len(send))
|
|
|
|
writeErr := make(chan error, 1)
|
|
go func() {
|
|
_, wErr := clientConn.Write(send)
|
|
writeErr <- wErr
|
|
}()
|
|
|
|
if _, err := io.ReadFull(serverConn, recv); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := <-writeErr; err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !bytes.Equal(send, recv) {
|
|
t.Fatal("multi-table tcp sudoku payload mismatch")
|
|
}
|
|
})
|
|
|
|
t.Run("GivenSudokuUDPMask_WhenNotInnermost_ThenWrapFails", func(t *testing.T) {
|
|
cfg := &sudoku.Config{Password: "sudoku-udp"}
|
|
raw, err := net.ListenPacket("udp", "127.0.0.1:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer raw.Close()
|
|
|
|
if _, err := cfg.WrapPacketConnClient(raw, 0, 1); err == nil {
|
|
t.Fatal("expected innermost check failure")
|
|
}
|
|
})
|
|
|
|
t.Run("GivenSudokuMultiTableUDPMask_WhenClientSendsMultipleDatagrams_ThenPayloadMatches", func(t *testing.T) {
|
|
cfg := &sudoku.Config{
|
|
Password: "sudoku-udp-multi",
|
|
Ascii: "prefer_entropy",
|
|
CustomTables: []string{"xpxvvpvv", "vxpvxvvp"},
|
|
PaddingMin: 0,
|
|
PaddingMax: 0,
|
|
}
|
|
maskManager := finalmask.NewUdpmaskManager([]finalmask.Udpmask{cfg})
|
|
|
|
clientRaw, err := net.ListenPacket("udp", "127.0.0.1:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer clientRaw.Close()
|
|
|
|
serverRaw, err := net.ListenPacket("udp", "127.0.0.1:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer serverRaw.Close()
|
|
|
|
client, err := maskManager.WrapPacketConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
server, err := maskManager.WrapPacketConnServer(serverRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
_ = client.SetDeadline(time.Now().Add(2 * time.Second))
|
|
_ = server.SetDeadline(time.Now().Add(2 * time.Second))
|
|
|
|
mustSendRecv(t, client, server, []byte("first-datagram"))
|
|
mustSendRecv(t, client, server, []byte("second-datagram"))
|
|
mustSendRecv(t, client, server, []byte("third-datagram"))
|
|
})
|
|
|
|
t.Run("GivenSudokuTCPMask_WhenCloseWriteIsCalled_ThenEOFPropagates", func(t *testing.T) {
|
|
cfg := &sudoku.Config{
|
|
Password: "sudoku-closewrite",
|
|
Ascii: "prefer_ascii",
|
|
PaddingMin: 0,
|
|
PaddingMax: 0,
|
|
}
|
|
|
|
listener, err := net.Listen("tcp", "127.0.0.1:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listener.Close()
|
|
|
|
acceptCh := make(chan net.Conn, 1)
|
|
errCh := make(chan error, 1)
|
|
go func() {
|
|
conn, err := listener.Accept()
|
|
if err != nil {
|
|
errCh <- err
|
|
return
|
|
}
|
|
acceptCh <- conn
|
|
}()
|
|
|
|
clientRaw, err := net.Dial("tcp", listener.Addr().String())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer clientRaw.Close()
|
|
|
|
var serverRaw net.Conn
|
|
select {
|
|
case serverRaw = <-acceptCh:
|
|
case err := <-errCh:
|
|
t.Fatal(err)
|
|
case <-time.After(2 * time.Second):
|
|
t.Fatal("accept timeout")
|
|
}
|
|
defer serverRaw.Close()
|
|
|
|
clientConn, err := cfg.WrapConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
serverConn, err := cfg.WrapConnServer(serverRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
closeWriter, ok := clientConn.(interface{ CloseWrite() error })
|
|
if !ok {
|
|
t.Fatalf("wrapped conn does not expose CloseWrite: %T", clientConn)
|
|
}
|
|
|
|
writeErr := make(chan error, 1)
|
|
go func() {
|
|
if _, err := clientConn.Write([]byte("closewrite")); err != nil {
|
|
writeErr <- err
|
|
return
|
|
}
|
|
writeErr <- closeWriter.CloseWrite()
|
|
}()
|
|
|
|
buf := make([]byte, len("closewrite"))
|
|
if _, err := io.ReadFull(serverConn, buf); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !bytes.Equal(buf, []byte("closewrite")) {
|
|
t.Fatal("unexpected payload before closewrite")
|
|
}
|
|
if err := <-writeErr; err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
one := make([]byte, 1)
|
|
n, err := serverConn.Read(one)
|
|
if n != 0 || err != io.EOF {
|
|
t.Fatalf("expected EOF after CloseWrite, got n=%d err=%v", n, err)
|
|
}
|
|
})
|
|
|
|
t.Run("GivenSudokuTCPMask_WhenProxyUnwrapRawConn_ThenMaskConnIsRetained", func(t *testing.T) {
|
|
cfg := &sudoku.Config{
|
|
Password: "sudoku-unwrap",
|
|
Ascii: "prefer_entropy",
|
|
}
|
|
|
|
clientRaw, serverRaw := net.Pipe()
|
|
defer clientRaw.Close()
|
|
defer serverRaw.Close()
|
|
|
|
clientConn, err := cfg.WrapConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
unwrapped, readCounter, writeCounter := proxy.UnwrapRawConn(clientConn)
|
|
if readCounter != nil || writeCounter != nil {
|
|
t.Fatal("unexpected stat counters while unwrapping sudoku conn")
|
|
}
|
|
if unwrapped != clientConn {
|
|
t.Fatalf("expected sudoku conn to stay wrapped, got %T", unwrapped)
|
|
}
|
|
})
|
|
|
|
t.Run("GivenSudokuTCPMask_WhenProxyUnwrapRawConn_AfterDownlinkOptimization_ThenMaskConnIsRetained", func(t *testing.T) {
|
|
cfg := &sudoku.Config{
|
|
Password: "sudoku-packed-unwrap",
|
|
Ascii: "prefer_entropy",
|
|
}
|
|
|
|
clientRaw, serverRaw := net.Pipe()
|
|
defer clientRaw.Close()
|
|
defer serverRaw.Close()
|
|
|
|
clientConn, err := cfg.WrapConnClient(clientRaw)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
unwrapped, readCounter, writeCounter := proxy.UnwrapRawConn(clientConn)
|
|
if readCounter != nil || writeCounter != nil {
|
|
t.Fatal("unexpected stat counters while unwrapping sudoku conn")
|
|
}
|
|
if unwrapped != clientConn {
|
|
t.Fatalf("expected sudoku conn to stay wrapped, got %T", unwrapped)
|
|
}
|
|
})
|
|
}
|