mirror of
https://github.com/XTLS/Xray-core.git
synced 2025-01-25 19:14:12 +00:00
94c249a8c8
* Fix some tests to PickPort correctly * Add retry logic for two socks tests Two socks tests listen for both TCP and UDP with same port number, in some cases the port is not available Add retry logic for server core, start server core and client core separately Extract a common method * Add retry logic for two dokodemo tests Two dokodemo tests listen for 5 ports, in some cases they are not available Add retry logic for client core, start server and client separately
256 lines
5.7 KiB
Go
256 lines
5.7 KiB
Go
package scenarios
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/rand"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"runtime"
|
|
"sync"
|
|
"syscall"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
|
|
"github.com/xtls/xray-core/app/dispatcher"
|
|
"github.com/xtls/xray-core/app/proxyman"
|
|
"github.com/xtls/xray-core/common"
|
|
"github.com/xtls/xray-core/common/errors"
|
|
"github.com/xtls/xray-core/common/log"
|
|
"github.com/xtls/xray-core/common/net"
|
|
"github.com/xtls/xray-core/common/retry"
|
|
"github.com/xtls/xray-core/common/serial"
|
|
"github.com/xtls/xray-core/common/units"
|
|
core "github.com/xtls/xray-core/core"
|
|
)
|
|
|
|
func xor(b []byte) []byte {
|
|
r := make([]byte, len(b))
|
|
for i, v := range b {
|
|
r[i] = v ^ 'c'
|
|
}
|
|
return r
|
|
}
|
|
|
|
func readFrom(conn net.Conn, timeout time.Duration, length int) []byte {
|
|
b := make([]byte, length)
|
|
deadline := time.Now().Add(timeout)
|
|
conn.SetReadDeadline(deadline)
|
|
n, err := io.ReadFull(conn, b[:length])
|
|
if err != nil {
|
|
fmt.Println("Unexpected error from readFrom:", err)
|
|
}
|
|
return b[:n]
|
|
}
|
|
|
|
func readFrom2(conn net.Conn, timeout time.Duration, length int) ([]byte, error) {
|
|
b := make([]byte, length)
|
|
deadline := time.Now().Add(timeout)
|
|
conn.SetReadDeadline(deadline)
|
|
n, err := io.ReadFull(conn, b[:length])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return b[:n], nil
|
|
}
|
|
|
|
func InitializeServerConfigs(configs ...*core.Config) ([]*exec.Cmd, error) {
|
|
servers := make([]*exec.Cmd, 0, 10)
|
|
|
|
for _, config := range configs {
|
|
server, err := InitializeServerConfig(config)
|
|
if err != nil {
|
|
CloseAllServers(servers)
|
|
return nil, err
|
|
}
|
|
servers = append(servers, server)
|
|
}
|
|
|
|
time.Sleep(time.Second * 2)
|
|
|
|
return servers, nil
|
|
}
|
|
|
|
func InitializeServerConfig(config *core.Config) (*exec.Cmd, error) {
|
|
err := BuildXray()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
config = withDefaultApps(config)
|
|
configBytes, err := proto.Marshal(config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
proc := RunXrayProtobuf(configBytes)
|
|
|
|
if err := proc.Start(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return proc, nil
|
|
}
|
|
|
|
var (
|
|
testBinaryPath string
|
|
testBinaryPathGen sync.Once
|
|
)
|
|
|
|
func genTestBinaryPath() {
|
|
testBinaryPathGen.Do(func() {
|
|
var tempDir string
|
|
common.Must(retry.Timed(5, 100).On(func() error {
|
|
dir, err := os.MkdirTemp("", "xray")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
tempDir = dir
|
|
return nil
|
|
}))
|
|
file := filepath.Join(tempDir, "xray.test")
|
|
if runtime.GOOS == "windows" {
|
|
file += ".exe"
|
|
}
|
|
testBinaryPath = file
|
|
fmt.Printf("Generated binary path: %s\n", file)
|
|
})
|
|
}
|
|
|
|
func GetSourcePath() string {
|
|
return filepath.Join("github.com", "xtls", "xray-core", "main")
|
|
}
|
|
|
|
func CloseAllServers(servers []*exec.Cmd) {
|
|
log.Record(&log.GeneralMessage{
|
|
Severity: log.Severity_Info,
|
|
Content: "Closing all servers.",
|
|
})
|
|
for _, server := range servers {
|
|
if runtime.GOOS == "windows" {
|
|
server.Process.Kill()
|
|
} else {
|
|
server.Process.Signal(syscall.SIGTERM)
|
|
}
|
|
}
|
|
for _, server := range servers {
|
|
server.Process.Wait()
|
|
}
|
|
log.Record(&log.GeneralMessage{
|
|
Severity: log.Severity_Info,
|
|
Content: "All server closed.",
|
|
})
|
|
}
|
|
|
|
func CloseServer(server *exec.Cmd) {
|
|
log.Record(&log.GeneralMessage{
|
|
Severity: log.Severity_Info,
|
|
Content: "Closing server.",
|
|
})
|
|
if runtime.GOOS == "windows" {
|
|
server.Process.Kill()
|
|
} else {
|
|
server.Process.Signal(syscall.SIGTERM)
|
|
}
|
|
server.Process.Wait()
|
|
log.Record(&log.GeneralMessage{
|
|
Severity: log.Severity_Info,
|
|
Content: "Server closed.",
|
|
})
|
|
}
|
|
|
|
func withDefaultApps(config *core.Config) *core.Config {
|
|
config.App = append(config.App, serial.ToTypedMessage(&dispatcher.Config{}))
|
|
config.App = append(config.App, serial.ToTypedMessage(&proxyman.InboundConfig{}))
|
|
config.App = append(config.App, serial.ToTypedMessage(&proxyman.OutboundConfig{}))
|
|
return config
|
|
}
|
|
|
|
func testTCPConn(port net.Port, payloadSize int, timeout time.Duration) func() error {
|
|
return func() error {
|
|
conn, err := net.DialTCP("tcp", nil, &net.TCPAddr{
|
|
IP: []byte{127, 0, 0, 1},
|
|
Port: int(port),
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer conn.Close()
|
|
|
|
return testTCPConn2(conn, payloadSize, timeout)()
|
|
}
|
|
}
|
|
|
|
func testUDPConn(port net.Port, payloadSize int, timeout time.Duration) func() error {
|
|
return func() error {
|
|
conn, err := net.DialUDP("udp", nil, &net.UDPAddr{
|
|
IP: []byte{127, 0, 0, 1},
|
|
Port: int(port),
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer conn.Close()
|
|
|
|
return testTCPConn2(conn, payloadSize, timeout)()
|
|
}
|
|
}
|
|
|
|
func testTCPConn2(conn net.Conn, payloadSize int, timeout time.Duration) func() error {
|
|
return func() (err1 error) {
|
|
start := time.Now()
|
|
defer func() {
|
|
var m runtime.MemStats
|
|
runtime.ReadMemStats(&m)
|
|
// For info on each, see: https://golang.org/pkg/runtime/#MemStats
|
|
fmt.Println("testConn finishes:", time.Since(start).Milliseconds(), "ms\t",
|
|
err1, "\tAlloc =", units.ByteSize(m.Alloc).String(),
|
|
"\tTotalAlloc =", units.ByteSize(m.TotalAlloc).String(),
|
|
"\tSys =", units.ByteSize(m.Sys).String(),
|
|
"\tNumGC =", m.NumGC)
|
|
}()
|
|
payload := make([]byte, payloadSize)
|
|
common.Must2(rand.Read(payload))
|
|
|
|
nBytes, err := conn.Write(payload)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if nBytes != len(payload) {
|
|
return errors.New("expect ", len(payload), " written, but actually ", nBytes)
|
|
}
|
|
|
|
response, err := readFrom2(conn, timeout, payloadSize)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_ = response
|
|
|
|
if r := bytes.Compare(response, xor(payload)); r != 0 {
|
|
return errors.New(r)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func WaitConnAvailableWithTest(t *testing.T, testFunc func() error) bool {
|
|
for i := 1; ; i++ {
|
|
if i > 10 {
|
|
t.Log("All attempts failed to test tcp conn")
|
|
return false
|
|
}
|
|
time.Sleep(time.Millisecond * 10)
|
|
if err := testFunc(); err != nil {
|
|
t.Log("err ", err)
|
|
} else {
|
|
t.Log("success with", i, "attempts")
|
|
break
|
|
}
|
|
}
|
|
return true
|
|
}
|