clash_test.go 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518
  1. package main
  2. import (
  3. "context"
  4. "crypto/md5"
  5. "crypto/rand"
  6. "errors"
  7. "io"
  8. "net"
  9. "net/http"
  10. _ "net/http/pprof"
  11. "net/netip"
  12. "sync"
  13. "testing"
  14. "time"
  15. C "github.com/sagernet/sing-box/constant"
  16. "github.com/sagernet/sing-box/log"
  17. "github.com/sagernet/sing/common/control"
  18. F "github.com/sagernet/sing/common/format"
  19. "github.com/docker/docker/api/types"
  20. "github.com/docker/docker/client"
  21. "github.com/stretchr/testify/assert"
  22. "github.com/stretchr/testify/require"
  23. )
  24. // kanged from clash
  25. const (
  26. ImageShadowsocksRustServer = "ghcr.io/shadowsocks/ssserver-rust:latest"
  27. ImageShadowsocksRustClient = "ghcr.io/shadowsocks/sslocal-rust:latest"
  28. ImageV2RayCore = "v2fly/v2fly-core:latest"
  29. ImageTrojan = "trojangfw/trojan:latest"
  30. ImageNaive = "pocat/naiveproxy:client"
  31. ImageBoringTun = "ghcr.io/ntkme/boringtun:edge"
  32. ImageHysteria = "tobyxdd/hysteria:latest"
  33. ImageNginx = "nginx:stable"
  34. )
  35. var allImages = []string{
  36. ImageShadowsocksRustServer,
  37. ImageShadowsocksRustClient,
  38. ImageV2RayCore,
  39. ImageTrojan,
  40. ImageNaive,
  41. ImageBoringTun,
  42. ImageHysteria,
  43. // ImageNginx,
  44. }
  45. var localIP = netip.MustParseAddr("127.0.0.1")
  46. func init() {
  47. if C.IsDarwin {
  48. var err error
  49. localIP, err = defaultRouteIP()
  50. if err != nil {
  51. panic(err)
  52. }
  53. }
  54. dockerClient, err := client.NewClientWithOpts(client.FromEnv, client.WithAPIVersionNegotiation())
  55. if err != nil {
  56. panic(err)
  57. }
  58. defer dockerClient.Close()
  59. list, err := dockerClient.ImageList(context.Background(), types.ImageListOptions{All: true})
  60. if err != nil {
  61. panic(err)
  62. }
  63. imageExist := func(image string) bool {
  64. for _, item := range list {
  65. for _, tag := range item.RepoTags {
  66. if image == tag {
  67. return true
  68. }
  69. }
  70. }
  71. return false
  72. }
  73. for _, image := range allImages {
  74. if imageExist(image) {
  75. continue
  76. }
  77. log.Info("pulling image: ", image)
  78. imageStream, err := dockerClient.ImagePull(context.Background(), image, types.ImagePullOptions{})
  79. if err != nil {
  80. panic(err)
  81. }
  82. io.Copy(io.Discard, imageStream)
  83. }
  84. go func() {
  85. err = http.ListenAndServe("0.0.0.0:8965", nil)
  86. if err != nil {
  87. log.Debug(err)
  88. }
  89. }()
  90. }
  91. func newPingPongPair() (chan []byte, chan []byte, func(t *testing.T) error) {
  92. pingCh := make(chan []byte)
  93. pongCh := make(chan []byte)
  94. test := func(t *testing.T) error {
  95. defer close(pingCh)
  96. defer close(pongCh)
  97. pingOpen := false
  98. pongOpen := false
  99. var recv []byte
  100. for {
  101. if pingOpen && pongOpen {
  102. break
  103. }
  104. select {
  105. case recv, pingOpen = <-pingCh:
  106. assert.True(t, pingOpen)
  107. assert.Equal(t, []byte("ping"), recv)
  108. case recv, pongOpen = <-pongCh:
  109. assert.True(t, pongOpen)
  110. assert.Equal(t, []byte("pong"), recv)
  111. case <-time.After(10 * time.Second):
  112. return errors.New("timeout")
  113. }
  114. }
  115. return nil
  116. }
  117. return pingCh, pongCh, test
  118. }
  119. func newLargeDataPair() (chan hashPair, chan hashPair, func(t *testing.T) error) {
  120. pingCh := make(chan hashPair)
  121. pongCh := make(chan hashPair)
  122. test := func(t *testing.T) error {
  123. defer close(pingCh)
  124. defer close(pongCh)
  125. pingOpen := false
  126. pongOpen := false
  127. var serverPair hashPair
  128. var clientPair hashPair
  129. for {
  130. if pingOpen && pongOpen {
  131. break
  132. }
  133. select {
  134. case serverPair, pingOpen = <-pingCh:
  135. assert.True(t, pingOpen)
  136. case clientPair, pongOpen = <-pongCh:
  137. assert.True(t, pongOpen)
  138. case <-time.After(10 * time.Second):
  139. return errors.New("timeout")
  140. }
  141. }
  142. assert.Equal(t, serverPair.recvHash, clientPair.sendHash)
  143. assert.Equal(t, serverPair.sendHash, clientPair.recvHash)
  144. return nil
  145. }
  146. return pingCh, pongCh, test
  147. }
  148. func testPingPongWithConn(t *testing.T, port uint16, cc func() (net.Conn, error)) error {
  149. l, err := listen("tcp", ":"+F.ToString(port))
  150. if err != nil {
  151. return err
  152. }
  153. defer l.Close()
  154. c, err := cc()
  155. if err != nil {
  156. return err
  157. }
  158. pingCh, pongCh, test := newPingPongPair()
  159. go func() {
  160. c, err := l.Accept()
  161. if err != nil {
  162. return
  163. }
  164. buf := make([]byte, 4)
  165. if _, err := io.ReadFull(c, buf); err != nil {
  166. return
  167. }
  168. pingCh <- buf
  169. if _, err := c.Write([]byte("pong")); err != nil {
  170. return
  171. }
  172. }()
  173. go func() {
  174. if _, err := c.Write([]byte("ping")); err != nil {
  175. return
  176. }
  177. buf := make([]byte, 4)
  178. if _, err := io.ReadFull(c, buf); err != nil {
  179. return
  180. }
  181. pongCh <- buf
  182. }()
  183. return test(t)
  184. }
  185. func testPingPongWithPacketConn(t *testing.T, port uint16, pcc func() (net.PacketConn, error)) error {
  186. l, err := listenPacket("udp", ":"+F.ToString(port))
  187. if err != nil {
  188. return err
  189. }
  190. defer l.Close()
  191. rAddr := &net.UDPAddr{IP: localIP.AsSlice(), Port: int(port)}
  192. pingCh, pongCh, test := newPingPongPair()
  193. go func() {
  194. buf := make([]byte, 1024)
  195. n, rAddr, err := l.ReadFrom(buf)
  196. if err != nil {
  197. return
  198. }
  199. pingCh <- buf[:n]
  200. if _, err := l.WriteTo([]byte("pong"), rAddr); err != nil {
  201. return
  202. }
  203. }()
  204. pc, err := pcc()
  205. if err != nil {
  206. return err
  207. }
  208. go func() {
  209. if _, err := pc.WriteTo([]byte("ping"), rAddr); err != nil {
  210. return
  211. }
  212. buf := make([]byte, 1024)
  213. n, _, err := pc.ReadFrom(buf)
  214. if err != nil {
  215. return
  216. }
  217. pongCh <- buf[:n]
  218. }()
  219. return test(t)
  220. }
  221. type hashPair struct {
  222. sendHash map[int][]byte
  223. recvHash map[int][]byte
  224. }
  225. func testLargeDataWithConn(t *testing.T, port uint16, cc func() (net.Conn, error)) error {
  226. l, err := listen("tcp", ":"+F.ToString(port))
  227. require.NoError(t, err)
  228. defer l.Close()
  229. times := 100
  230. chunkSize := int64(64 * 1024)
  231. pingCh, pongCh, test := newLargeDataPair()
  232. writeRandData := func(conn net.Conn) (map[int][]byte, error) {
  233. buf := make([]byte, chunkSize)
  234. hashMap := map[int][]byte{}
  235. for i := 0; i < times; i++ {
  236. if _, err := rand.Read(buf[1:]); err != nil {
  237. return nil, err
  238. }
  239. buf[0] = byte(i)
  240. hash := md5.Sum(buf)
  241. hashMap[i] = hash[:]
  242. if _, err := conn.Write(buf); err != nil {
  243. return nil, err
  244. }
  245. }
  246. return hashMap, nil
  247. }
  248. c, err := cc()
  249. if err != nil {
  250. return err
  251. }
  252. go func() {
  253. c, err := l.Accept()
  254. if err != nil {
  255. return
  256. }
  257. defer c.Close()
  258. hashMap := map[int][]byte{}
  259. buf := make([]byte, chunkSize)
  260. for i := 0; i < times; i++ {
  261. _, err := io.ReadFull(c, buf)
  262. if err != nil {
  263. t.Log(err.Error())
  264. return
  265. }
  266. hash := md5.Sum(buf)
  267. hashMap[int(buf[0])] = hash[:]
  268. }
  269. sendHash, err := writeRandData(c)
  270. if err != nil {
  271. t.Log(err.Error())
  272. return
  273. }
  274. pingCh <- hashPair{
  275. sendHash: sendHash,
  276. recvHash: hashMap,
  277. }
  278. }()
  279. go func() {
  280. sendHash, err := writeRandData(c)
  281. if err != nil {
  282. t.Log(err.Error())
  283. return
  284. }
  285. hashMap := map[int][]byte{}
  286. buf := make([]byte, chunkSize)
  287. for i := 0; i < times; i++ {
  288. _, err := io.ReadFull(c, buf)
  289. if err != nil {
  290. t.Log(err.Error())
  291. return
  292. }
  293. hash := md5.Sum(buf)
  294. hashMap[int(buf[0])] = hash[:]
  295. }
  296. pongCh <- hashPair{
  297. sendHash: sendHash,
  298. recvHash: hashMap,
  299. }
  300. }()
  301. return test(t)
  302. }
  303. func testLargeDataWithPacketConn(t *testing.T, port uint16, pcc func() (net.PacketConn, error)) error {
  304. l, err := listenPacket("udp", ":"+F.ToString(port))
  305. if err != nil {
  306. return err
  307. }
  308. defer l.Close()
  309. rAddr := &net.UDPAddr{IP: localIP.AsSlice(), Port: int(port)}
  310. times := 50
  311. chunkSize := int64(1024)
  312. pingCh, pongCh, test := newLargeDataPair()
  313. writeRandData := func(pc net.PacketConn, addr net.Addr) (map[int][]byte, error) {
  314. hashMap := map[int][]byte{}
  315. mux := sync.Mutex{}
  316. for i := 0; i < times; i++ {
  317. buf := make([]byte, chunkSize)
  318. if _, err = rand.Read(buf[1:]); err != nil {
  319. t.Log(err.Error())
  320. continue
  321. }
  322. buf[0] = byte(i)
  323. hash := md5.Sum(buf)
  324. mux.Lock()
  325. hashMap[i] = hash[:]
  326. mux.Unlock()
  327. if _, err = pc.WriteTo(buf, addr); err != nil {
  328. t.Log(err)
  329. continue
  330. }
  331. }
  332. return hashMap, nil
  333. }
  334. go func() {
  335. var rAddr net.Addr
  336. hashMap := map[int][]byte{}
  337. buf := make([]byte, 64*1024)
  338. for i := 0; i < times; i++ {
  339. _, rAddr, err = l.ReadFrom(buf)
  340. if err != nil {
  341. t.Log(err.Error())
  342. return
  343. }
  344. hash := md5.Sum(buf[:chunkSize])
  345. hashMap[int(buf[0])] = hash[:]
  346. }
  347. sendHash, err := writeRandData(l, rAddr)
  348. if err != nil {
  349. t.Log(err.Error())
  350. return
  351. }
  352. pingCh <- hashPair{
  353. sendHash: sendHash,
  354. recvHash: hashMap,
  355. }
  356. }()
  357. pc, err := pcc()
  358. if err != nil {
  359. return err
  360. }
  361. go func() {
  362. sendHash, err := writeRandData(pc, rAddr)
  363. if err != nil {
  364. t.Log(err.Error())
  365. return
  366. }
  367. hashMap := map[int][]byte{}
  368. buf := make([]byte, 64*1024)
  369. for i := 0; i < times; i++ {
  370. _, _, err := pc.ReadFrom(buf)
  371. if err != nil {
  372. t.Log(err.Error())
  373. return
  374. }
  375. hash := md5.Sum(buf[:chunkSize])
  376. hashMap[int(buf[0])] = hash[:]
  377. }
  378. pongCh <- hashPair{
  379. sendHash: sendHash,
  380. recvHash: hashMap,
  381. }
  382. }()
  383. return test(t)
  384. }
  385. func testPacketConnTimeout(t *testing.T, pcc func() (net.PacketConn, error)) error {
  386. pc, err := pcc()
  387. if err != nil {
  388. return err
  389. }
  390. err = pc.SetReadDeadline(time.Now().Add(time.Millisecond * 300))
  391. require.NoError(t, err)
  392. errCh := make(chan error, 1)
  393. go func() {
  394. buf := make([]byte, 1024)
  395. _, _, err := pc.ReadFrom(buf)
  396. errCh <- err
  397. }()
  398. select {
  399. case <-errCh:
  400. return nil
  401. case <-time.After(time.Second * 10):
  402. return errors.New("timeout")
  403. }
  404. }
  405. func listen(network, address string) (net.Listener, error) {
  406. var lc net.ListenConfig
  407. lc.Control = control.ReuseAddr()
  408. var lastErr error
  409. for i := 0; i < 5; i++ {
  410. l, err := lc.Listen(context.Background(), network, address)
  411. if err == nil {
  412. return l, nil
  413. }
  414. lastErr = err
  415. time.Sleep(5 * time.Millisecond)
  416. }
  417. return nil, lastErr
  418. }
  419. func listenPacket(network, address string) (net.PacketConn, error) {
  420. var lc net.ListenConfig
  421. lc.Control = control.ReuseAddr()
  422. var lastErr error
  423. for i := 0; i < 5; i++ {
  424. l, err := lc.ListenPacket(context.Background(), network, address)
  425. if err == nil {
  426. return l, nil
  427. }
  428. lastErr = err
  429. time.Sleep(5 * time.Millisecond)
  430. }
  431. return nil, lastErr
  432. }