command_client.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. package libbox
  2. import (
  3. "encoding/binary"
  4. "net"
  5. "os"
  6. "path/filepath"
  7. "time"
  8. C "github.com/sagernet/sing-box/constant"
  9. "github.com/sagernet/sing/common"
  10. E "github.com/sagernet/sing/common/exceptions"
  11. )
  12. type CommandClient struct {
  13. handler CommandClientHandler
  14. conn net.Conn
  15. options CommandClientOptions
  16. }
  17. type CommandClientOptions struct {
  18. Command int32
  19. StatusInterval int64
  20. }
  21. type CommandClientHandler interface {
  22. Connected()
  23. Disconnected(message string)
  24. ClearLogs()
  25. WriteLogs(messageList StringIterator)
  26. WriteStatus(message *StatusMessage)
  27. WriteGroups(message OutboundGroupIterator)
  28. InitializeClashMode(modeList StringIterator, currentMode string)
  29. UpdateClashMode(newMode string)
  30. WriteConnections(message *Connections)
  31. }
  32. func NewStandaloneCommandClient() *CommandClient {
  33. return new(CommandClient)
  34. }
  35. func NewCommandClient(handler CommandClientHandler, options *CommandClientOptions) *CommandClient {
  36. return &CommandClient{
  37. handler: handler,
  38. options: common.PtrValueOrDefault(options),
  39. }
  40. }
  41. func (c *CommandClient) directConnect() (net.Conn, error) {
  42. if !sTVOS {
  43. return net.DialUnix("unix", nil, &net.UnixAddr{
  44. Name: filepath.Join(sBasePath, "command.sock"),
  45. Net: "unix",
  46. })
  47. } else {
  48. return net.Dial("tcp", "127.0.0.1:8964")
  49. }
  50. }
  51. func (c *CommandClient) directConnectWithRetry() (net.Conn, error) {
  52. var (
  53. conn net.Conn
  54. err error
  55. )
  56. for i := 0; i < 10; i++ {
  57. conn, err = c.directConnect()
  58. if err == nil {
  59. return conn, nil
  60. }
  61. time.Sleep(time.Duration(100+i*50) * time.Millisecond)
  62. }
  63. return nil, err
  64. }
  65. func (c *CommandClient) Connect() error {
  66. common.Close(c.conn)
  67. conn, err := c.directConnectWithRetry()
  68. if err != nil {
  69. return err
  70. }
  71. c.conn = conn
  72. err = binary.Write(conn, binary.BigEndian, uint8(c.options.Command))
  73. if err != nil {
  74. return err
  75. }
  76. switch c.options.Command {
  77. case CommandLog:
  78. err = binary.Write(conn, binary.BigEndian, c.options.StatusInterval)
  79. if err != nil {
  80. return E.Cause(err, "write interval")
  81. }
  82. c.handler.Connected()
  83. go c.handleLogConn(conn)
  84. case CommandStatus:
  85. err = binary.Write(conn, binary.BigEndian, c.options.StatusInterval)
  86. if err != nil {
  87. return E.Cause(err, "write interval")
  88. }
  89. c.handler.Connected()
  90. go c.handleStatusConn(conn)
  91. case CommandGroup:
  92. err = binary.Write(conn, binary.BigEndian, c.options.StatusInterval)
  93. if err != nil {
  94. return E.Cause(err, "write interval")
  95. }
  96. c.handler.Connected()
  97. go c.handleGroupConn(conn)
  98. case CommandClashMode:
  99. var (
  100. modeList []string
  101. currentMode string
  102. )
  103. modeList, currentMode, err = readClashModeList(conn)
  104. if err != nil {
  105. return err
  106. }
  107. if C.FixAndroidStack {
  108. go func() {
  109. c.handler.Connected()
  110. c.handler.InitializeClashMode(newIterator(modeList), currentMode)
  111. if len(modeList) == 0 {
  112. conn.Close()
  113. c.handler.Disconnected(os.ErrInvalid.Error())
  114. }
  115. }()
  116. } else {
  117. c.handler.Connected()
  118. c.handler.InitializeClashMode(newIterator(modeList), currentMode)
  119. if len(modeList) == 0 {
  120. conn.Close()
  121. c.handler.Disconnected(os.ErrInvalid.Error())
  122. }
  123. }
  124. if len(modeList) == 0 {
  125. return nil
  126. }
  127. go c.handleModeConn(conn)
  128. case CommandConnections:
  129. err = binary.Write(conn, binary.BigEndian, c.options.StatusInterval)
  130. if err != nil {
  131. return E.Cause(err, "write interval")
  132. }
  133. c.handler.Connected()
  134. go c.handleConnectionsConn(conn)
  135. }
  136. return nil
  137. }
  138. func (c *CommandClient) Disconnect() error {
  139. return common.Close(c.conn)
  140. }