command_client.go 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807
  1. package libbox
  2. import (
  3. "context"
  4. "net"
  5. "os"
  6. "path/filepath"
  7. "strconv"
  8. "sync"
  9. "time"
  10. "github.com/sagernet/sing-box/daemon"
  11. "github.com/sagernet/sing/common"
  12. E "github.com/sagernet/sing/common/exceptions"
  13. "google.golang.org/grpc"
  14. "google.golang.org/grpc/codes"
  15. "google.golang.org/grpc/credentials/insecure"
  16. "google.golang.org/grpc/metadata"
  17. "google.golang.org/grpc/status"
  18. "google.golang.org/protobuf/types/known/emptypb"
  19. )
  20. type CommandClient struct {
  21. handler CommandClientHandler
  22. grpcConn *grpc.ClientConn
  23. grpcClient daemon.StartedServiceClient
  24. options CommandClientOptions
  25. ctx context.Context
  26. cancel context.CancelFunc
  27. clientMutex sync.RWMutex
  28. standalone bool
  29. }
  30. type CommandClientOptions struct {
  31. commands []int32
  32. StatusInterval int64
  33. }
  34. func (o *CommandClientOptions) AddCommand(command int32) {
  35. o.commands = append(o.commands, command)
  36. }
  37. type CommandClientHandler interface {
  38. Connected()
  39. Disconnected(message string)
  40. SetDefaultLogLevel(level int32)
  41. ClearLogs()
  42. WriteLogs(messageList LogIterator)
  43. WriteStatus(message *StatusMessage)
  44. WriteGroups(message OutboundGroupIterator)
  45. WriteOutbounds(message OutboundGroupItemIterator)
  46. InitializeClashMode(modeList StringIterator, currentMode string)
  47. UpdateClashMode(newMode string)
  48. WriteConnectionEvents(events *ConnectionEvents)
  49. }
  50. type LogEntry struct {
  51. Level int32
  52. Message string
  53. }
  54. type LogIterator interface {
  55. Len() int32
  56. HasNext() bool
  57. Next() *LogEntry
  58. }
  59. type XPCDialer interface {
  60. DialXPC() (int32, error)
  61. }
  62. var sXPCDialer XPCDialer
  63. func SetXPCDialer(dialer XPCDialer) {
  64. sXPCDialer = dialer
  65. }
  66. func NewStandaloneCommandClient() *CommandClient {
  67. return &CommandClient{standalone: true}
  68. }
  69. func NewCommandClient(handler CommandClientHandler, options *CommandClientOptions) *CommandClient {
  70. return &CommandClient{
  71. handler: handler,
  72. options: common.PtrValueOrDefault(options),
  73. }
  74. }
  75. func unaryClientAuthInterceptor(ctx context.Context, method string, req, reply any, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error {
  76. if sCommandServerSecret != "" {
  77. ctx = metadata.AppendToOutgoingContext(ctx, "x-command-secret", sCommandServerSecret)
  78. }
  79. return invoker(ctx, method, req, reply, cc, opts...)
  80. }
  81. func streamClientAuthInterceptor(ctx context.Context, desc *grpc.StreamDesc, cc *grpc.ClientConn, method string, streamer grpc.Streamer, opts ...grpc.CallOption) (grpc.ClientStream, error) {
  82. if sCommandServerSecret != "" {
  83. ctx = metadata.AppendToOutgoingContext(ctx, "x-command-secret", sCommandServerSecret)
  84. }
  85. return streamer(ctx, desc, cc, method, opts...)
  86. }
  87. const (
  88. commandClientDialAttempts = 10
  89. commandClientDialBaseDelay = 100 * time.Millisecond
  90. commandClientDialStepDelay = 50 * time.Millisecond
  91. )
  92. func commandClientDialDelay(attempt int) time.Duration {
  93. return commandClientDialBaseDelay + time.Duration(attempt)*commandClientDialStepDelay
  94. }
  95. func dialTarget() (string, func(context.Context, string) (net.Conn, error)) {
  96. if sXPCDialer != nil {
  97. return "passthrough:///xpc", func(ctx context.Context, _ string) (net.Conn, error) {
  98. fileDescriptor, err := sXPCDialer.DialXPC()
  99. if err != nil {
  100. return nil, E.Cause(err, "dial xpc")
  101. }
  102. return networkConnectionFromFileDescriptor(fileDescriptor)
  103. }
  104. }
  105. if sCommandServerListenPort == 0 {
  106. socketPath := filepath.Join(sBasePath, "command.sock")
  107. return "passthrough:///command-socket", func(ctx context.Context, _ string) (net.Conn, error) {
  108. var networkDialer net.Dialer
  109. return networkDialer.DialContext(ctx, "unix", socketPath)
  110. }
  111. }
  112. return net.JoinHostPort("127.0.0.1", strconv.Itoa(int(sCommandServerListenPort))), nil
  113. }
  114. func networkConnectionFromFileDescriptor(fileDescriptor int32) (net.Conn, error) {
  115. file := os.NewFile(uintptr(fileDescriptor), "xpc-command-socket")
  116. if file == nil {
  117. return nil, E.New("invalid file descriptor")
  118. }
  119. networkConnection, err := net.FileConn(file)
  120. if err != nil {
  121. file.Close()
  122. return nil, E.Cause(err, "create connection from fd")
  123. }
  124. file.Close()
  125. return networkConnection, nil
  126. }
  127. func (c *CommandClient) dialWithRetry(target string, contextDialer func(context.Context, string) (net.Conn, error), retryDial bool) (*grpc.ClientConn, daemon.StartedServiceClient, error) {
  128. var connection *grpc.ClientConn
  129. var client daemon.StartedServiceClient
  130. var lastError error
  131. for attempt := 0; attempt < commandClientDialAttempts; attempt++ {
  132. if connection == nil {
  133. options := []grpc.DialOption{
  134. grpc.WithTransportCredentials(insecure.NewCredentials()),
  135. grpc.WithUnaryInterceptor(unaryClientAuthInterceptor),
  136. grpc.WithStreamInterceptor(streamClientAuthInterceptor),
  137. }
  138. if contextDialer != nil {
  139. options = append(options, grpc.WithContextDialer(contextDialer))
  140. }
  141. var err error
  142. connection, err = grpc.NewClient(target, options...)
  143. if err != nil {
  144. lastError = err
  145. if !retryDial {
  146. return nil, nil, E.Cause(err, "create command client")
  147. }
  148. time.Sleep(commandClientDialDelay(attempt))
  149. continue
  150. }
  151. client = daemon.NewStartedServiceClient(connection)
  152. }
  153. waitDuration := commandClientDialDelay(attempt)
  154. ctx, cancel := context.WithTimeout(context.Background(), waitDuration)
  155. _, err := client.GetStartedAt(ctx, &emptypb.Empty{}, grpc.WaitForReady(true))
  156. cancel()
  157. if err == nil {
  158. return connection, client, nil
  159. }
  160. lastError = err
  161. }
  162. if connection != nil {
  163. connection.Close()
  164. }
  165. return nil, nil, E.Cause(lastError, "probe command server")
  166. }
  167. func (c *CommandClient) Connect() error {
  168. c.clientMutex.Lock()
  169. common.Close(common.PtrOrNil(c.grpcConn))
  170. target, contextDialer := dialTarget()
  171. connection, client, err := c.dialWithRetry(target, contextDialer, true)
  172. if err != nil {
  173. c.clientMutex.Unlock()
  174. return err
  175. }
  176. c.grpcConn = connection
  177. c.grpcClient = client
  178. c.ctx, c.cancel = context.WithCancel(context.Background())
  179. c.clientMutex.Unlock()
  180. c.handler.Connected()
  181. return c.dispatchCommands()
  182. }
  183. func (c *CommandClient) ConnectWithFD(fd int32) error {
  184. c.clientMutex.Lock()
  185. common.Close(common.PtrOrNil(c.grpcConn))
  186. networkConnection, err := networkConnectionFromFileDescriptor(fd)
  187. if err != nil {
  188. c.clientMutex.Unlock()
  189. return err
  190. }
  191. connection, client, err := c.dialWithRetry("passthrough:///xpc", func(ctx context.Context, _ string) (net.Conn, error) {
  192. return networkConnection, nil
  193. }, false)
  194. if err != nil {
  195. networkConnection.Close()
  196. c.clientMutex.Unlock()
  197. return err
  198. }
  199. c.grpcConn = connection
  200. c.grpcClient = client
  201. c.ctx, c.cancel = context.WithCancel(context.Background())
  202. c.clientMutex.Unlock()
  203. c.handler.Connected()
  204. return c.dispatchCommands()
  205. }
  206. func (c *CommandClient) dispatchCommands() error {
  207. for _, command := range c.options.commands {
  208. switch command {
  209. case CommandLog:
  210. go c.handleLogStream()
  211. case CommandStatus:
  212. go c.handleStatusStream()
  213. case CommandGroup:
  214. go c.handleGroupStream()
  215. case CommandClashMode:
  216. go c.handleClashModeStream()
  217. case CommandConnections:
  218. go c.handleConnectionsStream()
  219. case CommandOutbounds:
  220. go c.handleOutboundsStream()
  221. default:
  222. return E.New("unknown command: ", command)
  223. }
  224. }
  225. return nil
  226. }
  227. func (c *CommandClient) Disconnect() error {
  228. c.clientMutex.Lock()
  229. defer c.clientMutex.Unlock()
  230. if c.cancel != nil {
  231. c.cancel()
  232. }
  233. return common.Close(common.PtrOrNil(c.grpcConn))
  234. }
  235. func (c *CommandClient) getClientForCall() (daemon.StartedServiceClient, error) {
  236. c.clientMutex.RLock()
  237. if c.grpcClient != nil {
  238. defer c.clientMutex.RUnlock()
  239. return c.grpcClient, nil
  240. }
  241. c.clientMutex.RUnlock()
  242. c.clientMutex.Lock()
  243. defer c.clientMutex.Unlock()
  244. if c.grpcClient != nil {
  245. return c.grpcClient, nil
  246. }
  247. target, contextDialer := dialTarget()
  248. connection, client, err := c.dialWithRetry(target, contextDialer, true)
  249. if err != nil {
  250. return nil, E.Cause(err, "get command client")
  251. }
  252. c.grpcConn = connection
  253. c.grpcClient = client
  254. if c.ctx == nil {
  255. c.ctx, c.cancel = context.WithCancel(context.Background())
  256. }
  257. return c.grpcClient, nil
  258. }
  259. func (c *CommandClient) closeConnection() {
  260. c.clientMutex.Lock()
  261. defer c.clientMutex.Unlock()
  262. if c.grpcConn != nil {
  263. c.grpcConn.Close()
  264. c.grpcConn = nil
  265. c.grpcClient = nil
  266. }
  267. }
  268. func callWithResult[T any](c *CommandClient, call func(client daemon.StartedServiceClient) (T, error)) (T, error) {
  269. client, err := c.getClientForCall()
  270. if err != nil {
  271. var zero T
  272. return zero, err
  273. }
  274. if c.standalone {
  275. defer c.closeConnection()
  276. }
  277. return call(client)
  278. }
  279. func (c *CommandClient) getStreamContext() (daemon.StartedServiceClient, context.Context) {
  280. c.clientMutex.RLock()
  281. defer c.clientMutex.RUnlock()
  282. return c.grpcClient, c.ctx
  283. }
  284. func (c *CommandClient) handleLogStream() {
  285. client, ctx := c.getStreamContext()
  286. stream, err := client.SubscribeLog(ctx, &emptypb.Empty{})
  287. if err != nil {
  288. c.handler.Disconnected(E.Cause(err, "subscribe log").Error())
  289. return
  290. }
  291. defaultLogLevel, err := client.GetDefaultLogLevel(ctx, &emptypb.Empty{})
  292. if err != nil {
  293. c.handler.Disconnected(E.Cause(err, "get default log level").Error())
  294. return
  295. }
  296. c.handler.SetDefaultLogLevel(int32(defaultLogLevel.Level))
  297. for {
  298. logMessage, err := stream.Recv()
  299. if err != nil {
  300. c.handler.Disconnected(E.Cause(err, "log stream recv").Error())
  301. return
  302. }
  303. if logMessage.Reset_ {
  304. c.handler.ClearLogs()
  305. }
  306. var messages []*LogEntry
  307. for _, msg := range logMessage.Messages {
  308. messages = append(messages, &LogEntry{
  309. Level: int32(msg.Level),
  310. Message: msg.Message,
  311. })
  312. }
  313. c.handler.WriteLogs(newIterator(messages))
  314. }
  315. }
  316. func (c *CommandClient) handleStatusStream() {
  317. client, ctx := c.getStreamContext()
  318. interval := c.options.StatusInterval
  319. stream, err := client.SubscribeStatus(ctx, &daemon.SubscribeStatusRequest{
  320. Interval: interval,
  321. })
  322. if err != nil {
  323. c.handler.Disconnected(E.Cause(err, "subscribe status").Error())
  324. return
  325. }
  326. for {
  327. status, err := stream.Recv()
  328. if err != nil {
  329. c.handler.Disconnected(E.Cause(err, "status stream recv").Error())
  330. return
  331. }
  332. c.handler.WriteStatus(statusMessageFromGRPC(status))
  333. }
  334. }
  335. func (c *CommandClient) handleGroupStream() {
  336. client, ctx := c.getStreamContext()
  337. stream, err := client.SubscribeGroups(ctx, &emptypb.Empty{})
  338. if err != nil {
  339. c.handler.Disconnected(E.Cause(err, "subscribe groups").Error())
  340. return
  341. }
  342. for {
  343. groups, err := stream.Recv()
  344. if err != nil {
  345. c.handler.Disconnected(E.Cause(err, "groups stream recv").Error())
  346. return
  347. }
  348. c.handler.WriteGroups(outboundGroupIteratorFromGRPC(groups))
  349. }
  350. }
  351. func (c *CommandClient) handleClashModeStream() {
  352. client, ctx := c.getStreamContext()
  353. modeStatus, err := client.GetClashModeStatus(ctx, &emptypb.Empty{})
  354. if err != nil {
  355. c.handler.Disconnected(E.Cause(err, "get clash mode status").Error())
  356. return
  357. }
  358. if sFixAndroidStack {
  359. go func() {
  360. c.handler.InitializeClashMode(newIterator(modeStatus.ModeList), modeStatus.CurrentMode)
  361. if len(modeStatus.ModeList) == 0 {
  362. c.handler.Disconnected(E.Cause(os.ErrInvalid, "empty clash mode list").Error())
  363. }
  364. }()
  365. } else {
  366. c.handler.InitializeClashMode(newIterator(modeStatus.ModeList), modeStatus.CurrentMode)
  367. if len(modeStatus.ModeList) == 0 {
  368. c.handler.Disconnected(E.Cause(os.ErrInvalid, "empty clash mode list").Error())
  369. return
  370. }
  371. }
  372. if len(modeStatus.ModeList) == 0 {
  373. return
  374. }
  375. stream, err := client.SubscribeClashMode(ctx, &emptypb.Empty{})
  376. if err != nil {
  377. c.handler.Disconnected(E.Cause(err, "subscribe clash mode").Error())
  378. return
  379. }
  380. for {
  381. mode, err := stream.Recv()
  382. if err != nil {
  383. c.handler.Disconnected(E.Cause(err, "clash mode stream recv").Error())
  384. return
  385. }
  386. c.handler.UpdateClashMode(mode.Mode)
  387. }
  388. }
  389. func (c *CommandClient) handleConnectionsStream() {
  390. client, ctx := c.getStreamContext()
  391. interval := c.options.StatusInterval
  392. stream, err := client.SubscribeConnections(ctx, &daemon.SubscribeConnectionsRequest{
  393. Interval: interval,
  394. })
  395. if err != nil {
  396. c.handler.Disconnected(E.Cause(err, "subscribe connections").Error())
  397. return
  398. }
  399. for {
  400. events, err := stream.Recv()
  401. if err != nil {
  402. c.handler.Disconnected(E.Cause(err, "connections stream recv").Error())
  403. return
  404. }
  405. libboxEvents := connectionEventsFromGRPC(events)
  406. c.handler.WriteConnectionEvents(libboxEvents)
  407. }
  408. }
  409. func (c *CommandClient) handleOutboundsStream() {
  410. client, ctx := c.getStreamContext()
  411. stream, err := client.SubscribeOutbounds(ctx, &emptypb.Empty{})
  412. if err != nil {
  413. c.handler.Disconnected(E.Cause(err, "subscribe outbounds").Error())
  414. return
  415. }
  416. for {
  417. list, err := stream.Recv()
  418. if err != nil {
  419. c.handler.Disconnected(E.Cause(err, "outbounds stream recv").Error())
  420. return
  421. }
  422. c.handler.WriteOutbounds(outboundGroupItemListFromGRPC(list))
  423. }
  424. }
  425. func (c *CommandClient) SelectOutbound(groupTag string, outboundTag string) error {
  426. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  427. return client.SelectOutbound(context.Background(), &daemon.SelectOutboundRequest{
  428. GroupTag: groupTag,
  429. OutboundTag: outboundTag,
  430. })
  431. })
  432. if err != nil {
  433. return E.Cause(err, "select outbound")
  434. }
  435. return nil
  436. }
  437. func (c *CommandClient) URLTest(groupTag string) error {
  438. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  439. return client.URLTest(context.Background(), &daemon.URLTestRequest{
  440. OutboundTag: groupTag,
  441. })
  442. })
  443. if err != nil {
  444. return E.Cause(err, "url test")
  445. }
  446. return nil
  447. }
  448. func (c *CommandClient) SetClashMode(newMode string) error {
  449. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  450. return client.SetClashMode(context.Background(), &daemon.ClashMode{
  451. Mode: newMode,
  452. })
  453. })
  454. if err != nil {
  455. return E.Cause(err, "set clash mode")
  456. }
  457. return nil
  458. }
  459. func (c *CommandClient) CloseConnection(connId string) error {
  460. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  461. return client.CloseConnection(context.Background(), &daemon.CloseConnectionRequest{
  462. Id: connId,
  463. })
  464. })
  465. if err != nil {
  466. return E.Cause(err, "close connection")
  467. }
  468. return nil
  469. }
  470. func (c *CommandClient) CloseConnections() error {
  471. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  472. return client.CloseAllConnections(context.Background(), &emptypb.Empty{})
  473. })
  474. if err != nil {
  475. return E.Cause(err, "close all connections")
  476. }
  477. return nil
  478. }
  479. func (c *CommandClient) ServiceReload() error {
  480. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  481. return client.ReloadService(context.Background(), &emptypb.Empty{})
  482. })
  483. if err != nil {
  484. return E.Cause(err, "reload service")
  485. }
  486. return nil
  487. }
  488. func (c *CommandClient) ServiceClose() error {
  489. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  490. return client.StopService(context.Background(), &emptypb.Empty{})
  491. })
  492. if err != nil {
  493. return E.Cause(err, "stop service")
  494. }
  495. return nil
  496. }
  497. func (c *CommandClient) ClearLogs() error {
  498. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  499. return client.ClearLogs(context.Background(), &emptypb.Empty{})
  500. })
  501. if err != nil {
  502. return E.Cause(err, "clear logs")
  503. }
  504. return nil
  505. }
  506. func (c *CommandClient) GetSystemProxyStatus() (*SystemProxyStatus, error) {
  507. return callWithResult(c, func(client daemon.StartedServiceClient) (*SystemProxyStatus, error) {
  508. status, err := client.GetSystemProxyStatus(context.Background(), &emptypb.Empty{})
  509. if err != nil {
  510. return nil, E.Cause(err, "get system proxy status")
  511. }
  512. return systemProxyStatusFromGRPC(status), nil
  513. })
  514. }
  515. func (c *CommandClient) SetSystemProxyEnabled(isEnabled bool) error {
  516. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  517. return client.SetSystemProxyEnabled(context.Background(), &daemon.SetSystemProxyEnabledRequest{
  518. Enabled: isEnabled,
  519. })
  520. })
  521. if err != nil {
  522. return E.Cause(err, "set system proxy enabled")
  523. }
  524. return nil
  525. }
  526. func (c *CommandClient) TriggerGoCrash() error {
  527. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  528. return client.TriggerDebugCrash(context.Background(), &daemon.DebugCrashRequest{
  529. Type: daemon.DebugCrashRequest_GO,
  530. })
  531. })
  532. if err != nil {
  533. return E.Cause(err, "trigger debug crash")
  534. }
  535. return nil
  536. }
  537. func (c *CommandClient) TriggerNativeCrash() error {
  538. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  539. return client.TriggerDebugCrash(context.Background(), &daemon.DebugCrashRequest{
  540. Type: daemon.DebugCrashRequest_NATIVE,
  541. })
  542. })
  543. if err != nil {
  544. return E.Cause(err, "trigger native crash")
  545. }
  546. return nil
  547. }
  548. func (c *CommandClient) TriggerOOMReport() error {
  549. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  550. return client.TriggerOOMReport(context.Background(), &emptypb.Empty{})
  551. })
  552. if err != nil {
  553. return E.Cause(err, "trigger oom report")
  554. }
  555. return nil
  556. }
  557. func (c *CommandClient) GetDeprecatedNotes() (DeprecatedNoteIterator, error) {
  558. return callWithResult(c, func(client daemon.StartedServiceClient) (DeprecatedNoteIterator, error) {
  559. warnings, err := client.GetDeprecatedWarnings(context.Background(), &emptypb.Empty{})
  560. if err != nil {
  561. return nil, E.Cause(err, "get deprecated warnings")
  562. }
  563. var notes []*DeprecatedNote
  564. for _, warning := range warnings.Warnings {
  565. notes = append(notes, &DeprecatedNote{
  566. Description: warning.Description,
  567. DeprecatedVersion: warning.DeprecatedVersion,
  568. ScheduledVersion: warning.ScheduledVersion,
  569. MigrationLink: warning.MigrationLink,
  570. })
  571. }
  572. return newIterator(notes), nil
  573. })
  574. }
  575. func (c *CommandClient) GetStartedAt() (int64, error) {
  576. return callWithResult(c, func(client daemon.StartedServiceClient) (int64, error) {
  577. startedAt, err := client.GetStartedAt(context.Background(), &emptypb.Empty{})
  578. if err != nil {
  579. return 0, E.Cause(err, "get started at")
  580. }
  581. return startedAt.StartedAt, nil
  582. })
  583. }
  584. func (c *CommandClient) SetGroupExpand(groupTag string, isExpand bool) error {
  585. _, err := callWithResult(c, func(client daemon.StartedServiceClient) (*emptypb.Empty, error) {
  586. return client.SetGroupExpand(context.Background(), &daemon.SetGroupExpandRequest{
  587. GroupTag: groupTag,
  588. IsExpand: isExpand,
  589. })
  590. })
  591. if err != nil {
  592. return E.Cause(err, "set group expand")
  593. }
  594. return nil
  595. }
  596. func (c *CommandClient) StartNetworkQualityTest(configURL string, outboundTag string, serial bool, maxRuntimeSeconds int32, http3 bool, handler NetworkQualityTestHandler) error {
  597. client, err := c.getClientForCall()
  598. if err != nil {
  599. return E.Cause(err, "start network quality test")
  600. }
  601. if c.standalone {
  602. defer c.closeConnection()
  603. }
  604. stream, err := client.StartNetworkQualityTest(context.Background(), &daemon.NetworkQualityTestRequest{
  605. ConfigURL: configURL,
  606. OutboundTag: outboundTag,
  607. Serial: serial,
  608. MaxRuntimeSeconds: maxRuntimeSeconds,
  609. Http3: http3,
  610. })
  611. if err != nil {
  612. return E.Cause(err, "start network quality test")
  613. }
  614. for {
  615. event, recvErr := stream.Recv()
  616. if recvErr != nil {
  617. recvErr = E.Cause(recvErr, "network quality test recv")
  618. handler.OnError(recvErr.Error())
  619. return recvErr
  620. }
  621. if event.IsFinal {
  622. if event.Error != "" {
  623. handler.OnError(event.Error)
  624. } else {
  625. handler.OnResult(&NetworkQualityResult{
  626. DownloadCapacity: event.DownloadCapacity,
  627. UploadCapacity: event.UploadCapacity,
  628. DownloadRPM: event.DownloadRPM,
  629. UploadRPM: event.UploadRPM,
  630. IdleLatencyMs: event.IdleLatencyMs,
  631. DownloadCapacityAccuracy: event.DownloadCapacityAccuracy,
  632. UploadCapacityAccuracy: event.UploadCapacityAccuracy,
  633. DownloadRPMAccuracy: event.DownloadRPMAccuracy,
  634. UploadRPMAccuracy: event.UploadRPMAccuracy,
  635. })
  636. }
  637. return nil
  638. }
  639. handler.OnProgress(networkQualityProgressFromGRPC(event))
  640. }
  641. }
  642. func (c *CommandClient) StartSTUNTest(server string, outboundTag string, handler STUNTestHandler) error {
  643. client, err := c.getClientForCall()
  644. if err != nil {
  645. return E.Cause(err, "start stun test")
  646. }
  647. if c.standalone {
  648. defer c.closeConnection()
  649. }
  650. stream, err := client.StartSTUNTest(context.Background(), &daemon.STUNTestRequest{
  651. Server: server,
  652. OutboundTag: outboundTag,
  653. })
  654. if err != nil {
  655. return E.Cause(err, "start stun test")
  656. }
  657. for {
  658. event, recvErr := stream.Recv()
  659. if recvErr != nil {
  660. recvErr = E.Cause(recvErr, "stun test recv")
  661. handler.OnError(recvErr.Error())
  662. return recvErr
  663. }
  664. if event.IsFinal {
  665. if event.Error != "" {
  666. handler.OnError(event.Error)
  667. } else {
  668. handler.OnResult(&STUNTestResult{
  669. ExternalAddr: event.ExternalAddr,
  670. LatencyMs: event.LatencyMs,
  671. NATMapping: event.NatMapping,
  672. NATFiltering: event.NatFiltering,
  673. NATTypeSupported: event.NatTypeSupported,
  674. })
  675. }
  676. return nil
  677. }
  678. handler.OnProgress(stunTestProgressFromGRPC(event))
  679. }
  680. }
  681. func (c *CommandClient) SubscribeTailscaleStatus(handler TailscaleStatusHandler) error {
  682. client, err := c.getClientForCall()
  683. if err != nil {
  684. return E.Cause(err, "subscribe tailscale status")
  685. }
  686. if c.standalone {
  687. defer c.closeConnection()
  688. }
  689. stream, err := client.SubscribeTailscaleStatus(context.Background(), &emptypb.Empty{})
  690. if err != nil {
  691. return E.Cause(err, "subscribe tailscale status")
  692. }
  693. for {
  694. event, recvErr := stream.Recv()
  695. if recvErr != nil {
  696. if status.Code(recvErr) == codes.NotFound || status.Code(recvErr) == codes.Unavailable {
  697. return nil
  698. }
  699. recvErr = E.Cause(recvErr, "tailscale status recv")
  700. handler.OnError(recvErr.Error())
  701. return recvErr
  702. }
  703. handler.OnStatusUpdate(tailscaleStatusUpdateFromGRPC(event))
  704. }
  705. }
  706. func (c *CommandClient) StartTailscalePing(endpointTag string, peerIP string, handler TailscalePingHandler) error {
  707. client, err := c.getClientForCall()
  708. if err != nil {
  709. return E.Cause(err, "start tailscale ping")
  710. }
  711. if c.standalone {
  712. defer c.closeConnection()
  713. }
  714. stream, err := client.StartTailscalePing(context.Background(), &daemon.TailscalePingRequest{
  715. EndpointTag: endpointTag,
  716. PeerIP: peerIP,
  717. })
  718. if err != nil {
  719. return E.Cause(err, "start tailscale ping")
  720. }
  721. for {
  722. event, recvErr := stream.Recv()
  723. if recvErr != nil {
  724. recvErr = E.Cause(recvErr, "tailscale ping recv")
  725. handler.OnError(recvErr.Error())
  726. return recvErr
  727. }
  728. handler.OnPingResult(tailscalePingResultFromGRPC(event))
  729. }
  730. }