shared.go 2.9 KB

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