123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237 |
- package scenarios
- import (
- "bytes"
- "crypto/rand"
- "fmt"
- "io"
- "os"
- "os/exec"
- "path/filepath"
- "runtime"
- "sync"
- "syscall"
- "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
- }
- }
|