shared.go 2.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. package api
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "os"
  7. "reflect"
  8. "strings"
  9. "time"
  10. "github.com/v2fly/v2ray-core/v4/main/commands/base"
  11. "google.golang.org/grpc"
  12. "google.golang.org/protobuf/proto"
  13. )
  14. var (
  15. apiServerAddrPtr string
  16. apiTimeout int
  17. )
  18. func setSharedFlags(cmd *base.Command) {
  19. cmd.Flag.StringVar(&apiServerAddrPtr, "s", "127.0.0.1:8080", "")
  20. cmd.Flag.StringVar(&apiServerAddrPtr, "server", "127.0.0.1:8080", "")
  21. cmd.Flag.IntVar(&apiTimeout, "t", 3, "")
  22. cmd.Flag.IntVar(&apiTimeout, "timeout", 3, "")
  23. }
  24. func dialAPIServer() (conn *grpc.ClientConn, ctx context.Context, close func()) {
  25. ctx, cancel := context.WithTimeout(context.Background(), time.Duration(apiTimeout)*time.Second)
  26. conn, err := grpc.DialContext(ctx, apiServerAddrPtr, grpc.WithInsecure(), grpc.WithBlock())
  27. if err != nil {
  28. base.Fatalf("failed to dial %s", apiServerAddrPtr)
  29. }
  30. close = func() {
  31. cancel()
  32. conn.Close()
  33. }
  34. return
  35. }
  36. func showResponese(m proto.Message) {
  37. if isEmpty(m) {
  38. // avoid outputs like `{}`, `{"key":{}}`
  39. return
  40. }
  41. b := new(strings.Builder)
  42. e := json.NewEncoder(b)
  43. e.SetIndent("", " ")
  44. e.SetEscapeHTML(false)
  45. err := e.Encode(m)
  46. if err != nil {
  47. fmt.Fprintf(os.Stdout, "%v\n", m)
  48. base.Fatalf("error encode json: %s", err)
  49. return
  50. }
  51. fmt.Println(strings.TrimSpace(b.String()))
  52. }
  53. // isEmpty checks if the response is empty (all zero values).
  54. // proto.Message types always "omitempty" on fields,
  55. // there's no chance for a response to show zero-value messages,
  56. // so we can perform isZero test here
  57. func isEmpty(response interface{}) bool {
  58. s := reflect.Indirect(reflect.ValueOf(response))
  59. if s.Kind() == reflect.Invalid {
  60. return true
  61. }
  62. switch s.Kind() {
  63. case reflect.Struct:
  64. for i := 0; i < s.NumField(); i++ {
  65. f := s.Type().Field(i)
  66. if f.Name[0] < 65 || f.Name[0] > 90 {
  67. // continue if not exported.
  68. continue
  69. }
  70. field := s.Field(i)
  71. if !isEmpty(field.Interface()) {
  72. return false
  73. }
  74. }
  75. case reflect.Array, reflect.Slice:
  76. for i := 0; i < s.Len(); i++ {
  77. if !isEmpty(s.Index(i).Interface()) {
  78. return false
  79. }
  80. }
  81. default:
  82. if !s.IsZero() {
  83. return false
  84. }
  85. }
  86. return true
  87. }