shared.go 2.5 KB

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