shared.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. package api
  2. import (
  3. "bytes"
  4. "context"
  5. "encoding/json"
  6. "fmt"
  7. "io"
  8. "io/ioutil"
  9. "net/http"
  10. "net/url"
  11. "os"
  12. "reflect"
  13. "strings"
  14. "time"
  15. "google.golang.org/grpc"
  16. "google.golang.org/protobuf/proto"
  17. "github.com/xtls/xray-core/common/buf"
  18. "github.com/xtls/xray-core/main/commands/base"
  19. )
  20. type serviceHandler func(ctx context.Context, conn *grpc.ClientConn, cmd *base.Command, args []string) string
  21. var (
  22. apiServerAddrPtr string
  23. apiTimeout int
  24. )
  25. func setSharedFlags(cmd *base.Command) {
  26. cmd.Flag.StringVar(&apiServerAddrPtr, "s", "127.0.0.1:8080", "")
  27. cmd.Flag.StringVar(&apiServerAddrPtr, "server", "127.0.0.1:8080", "")
  28. cmd.Flag.IntVar(&apiTimeout, "t", 3, "")
  29. cmd.Flag.IntVar(&apiTimeout, "timeout", 3, "")
  30. }
  31. func dialAPIServer() (conn *grpc.ClientConn, ctx context.Context, close func()) {
  32. ctx, cancel := context.WithTimeout(context.Background(), time.Duration(apiTimeout)*time.Second)
  33. conn, err := grpc.DialContext(ctx, apiServerAddrPtr, grpc.WithInsecure(), grpc.WithBlock())
  34. if err != nil {
  35. base.Fatalf("failed to dial %s", apiServerAddrPtr)
  36. }
  37. close = func() {
  38. cancel()
  39. conn.Close()
  40. }
  41. return
  42. }
  43. // loadArg loads one arg, maybe an remote url, or local file path
  44. func loadArg(arg string) (out io.Reader, err error) {
  45. var data []byte
  46. switch {
  47. case strings.HasPrefix(arg, "http://"), strings.HasPrefix(arg, "https://"):
  48. data, err = fetchHTTPContent(arg)
  49. case arg == "stdin:":
  50. data, err = ioutil.ReadAll(os.Stdin)
  51. default:
  52. data, err = ioutil.ReadFile(arg)
  53. }
  54. if err != nil {
  55. return
  56. }
  57. out = bytes.NewBuffer(data)
  58. return
  59. }
  60. // fetchHTTPContent dials https for remote content
  61. func fetchHTTPContent(target string) ([]byte, error) {
  62. parsedTarget, err := url.Parse(target)
  63. if err != nil {
  64. return nil, err
  65. }
  66. if s := strings.ToLower(parsedTarget.Scheme); s != "http" && s != "https" {
  67. return nil, fmt.Errorf("invalid scheme: %s", parsedTarget.Scheme)
  68. }
  69. client := &http.Client{
  70. Timeout: 30 * time.Second,
  71. }
  72. resp, err := client.Do(&http.Request{
  73. Method: "GET",
  74. URL: parsedTarget,
  75. Close: true,
  76. })
  77. if err != nil {
  78. return nil, fmt.Errorf("failed to dial to %s", target)
  79. }
  80. defer resp.Body.Close()
  81. if resp.StatusCode != 200 {
  82. return nil, fmt.Errorf("unexpected HTTP status code: %d", resp.StatusCode)
  83. }
  84. content, err := buf.ReadAllToBytes(resp.Body)
  85. if err != nil {
  86. return nil, fmt.Errorf("failed to read HTTP response")
  87. }
  88. return content, nil
  89. }
  90. func showResponese(m proto.Message) {
  91. if isNil(m) {
  92. return
  93. }
  94. b := new(strings.Builder)
  95. e := json.NewEncoder(b)
  96. e.SetIndent("", " ")
  97. e.SetEscapeHTML(false)
  98. err := e.Encode(m)
  99. msg := ""
  100. if err != nil {
  101. msg = fmt.Sprintf("error: %s\n\n%v", err, m)
  102. } else {
  103. msg = strings.TrimSpace(b.String())
  104. }
  105. if msg == "" {
  106. return
  107. }
  108. fmt.Println(msg)
  109. }
  110. func isNil(i interface{}) bool {
  111. vi := reflect.ValueOf(i)
  112. if vi.Kind() == reflect.Ptr {
  113. return vi.IsNil()
  114. }
  115. return i == nil
  116. }