2020-12-04 04:36:16 +03:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"os"
|
2020-12-04 18:35:08 +03:00
|
|
|
"reflect"
|
2020-12-04 04:36:16 +03:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2024-03-03 18:52:22 +03:00
|
|
|
"google.golang.org/grpc/credentials/insecure"
|
|
|
|
"google.golang.org/protobuf/encoding/protojson"
|
|
|
|
|
2020-12-04 04:36:16 +03:00
|
|
|
"github.com/xtls/xray-core/common/buf"
|
|
|
|
"github.com/xtls/xray-core/main/commands/base"
|
2022-05-18 10:29:01 +03:00
|
|
|
"google.golang.org/grpc"
|
|
|
|
"google.golang.org/protobuf/proto"
|
2020-12-04 04:36:16 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
type serviceHandler func(ctx context.Context, conn *grpc.ClientConn, cmd *base.Command, args []string) string
|
|
|
|
|
|
|
|
var (
|
|
|
|
apiServerAddrPtr string
|
|
|
|
apiTimeout int
|
2024-02-18 06:51:37 +03:00
|
|
|
apiJSON bool
|
2020-12-04 04:36:16 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
func setSharedFlags(cmd *base.Command) {
|
|
|
|
cmd.Flag.StringVar(&apiServerAddrPtr, "s", "127.0.0.1:8080", "")
|
|
|
|
cmd.Flag.StringVar(&apiServerAddrPtr, "server", "127.0.0.1:8080", "")
|
|
|
|
cmd.Flag.IntVar(&apiTimeout, "t", 3, "")
|
|
|
|
cmd.Flag.IntVar(&apiTimeout, "timeout", 3, "")
|
2024-02-18 06:51:37 +03:00
|
|
|
cmd.Flag.BoolVar(&apiJSON, "json", false, "")
|
2020-12-04 04:36:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func dialAPIServer() (conn *grpc.ClientConn, ctx context.Context, close func()) {
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Duration(apiTimeout)*time.Second)
|
2024-03-03 18:52:22 +03:00
|
|
|
conn, err := grpc.DialContext(ctx, apiServerAddrPtr, grpc.WithTransportCredentials(insecure.NewCredentials()), grpc.WithBlock())
|
2020-12-04 04:36:16 +03:00
|
|
|
if err != nil {
|
|
|
|
base.Fatalf("failed to dial %s", apiServerAddrPtr)
|
|
|
|
}
|
|
|
|
close = func() {
|
|
|
|
cancel()
|
|
|
|
conn.Close()
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// loadArg loads one arg, maybe an remote url, or local file path
|
|
|
|
func loadArg(arg string) (out io.Reader, err error) {
|
|
|
|
var data []byte
|
|
|
|
switch {
|
|
|
|
case strings.HasPrefix(arg, "http://"), strings.HasPrefix(arg, "https://"):
|
|
|
|
data, err = fetchHTTPContent(arg)
|
|
|
|
|
|
|
|
case arg == "stdin:":
|
2021-09-28 21:49:34 +03:00
|
|
|
data, err = io.ReadAll(os.Stdin)
|
2020-12-04 04:36:16 +03:00
|
|
|
|
|
|
|
default:
|
2021-09-28 21:49:34 +03:00
|
|
|
data, err = os.ReadFile(arg)
|
2020-12-04 04:36:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
out = bytes.NewBuffer(data)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// fetchHTTPContent dials https for remote content
|
|
|
|
func fetchHTTPContent(target string) ([]byte, error) {
|
|
|
|
parsedTarget, err := url.Parse(target)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if s := strings.ToLower(parsedTarget.Scheme); s != "http" && s != "https" {
|
|
|
|
return nil, fmt.Errorf("invalid scheme: %s", parsedTarget.Scheme)
|
|
|
|
}
|
|
|
|
|
|
|
|
client := &http.Client{
|
|
|
|
Timeout: 30 * time.Second,
|
|
|
|
}
|
|
|
|
resp, err := client.Do(&http.Request{
|
|
|
|
Method: "GET",
|
|
|
|
URL: parsedTarget,
|
|
|
|
Close: true,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to dial to %s", target)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return nil, fmt.Errorf("unexpected HTTP status code: %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
content, err := buf.ReadAllToBytes(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to read HTTP response")
|
|
|
|
}
|
|
|
|
|
|
|
|
return content, nil
|
|
|
|
}
|
|
|
|
|
2024-02-18 06:51:37 +03:00
|
|
|
func protoToJSONString(m proto.Message, prefix, indent string) (string, error) {
|
|
|
|
return strings.TrimSpace(protojson.MarshalOptions{Indent: indent}.Format(m)), nil
|
2022-07-16 04:45:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func showJSONResponse(m proto.Message) {
|
2020-12-04 17:03:07 +03:00
|
|
|
if isNil(m) {
|
|
|
|
return
|
|
|
|
}
|
2022-07-16 04:45:13 +03:00
|
|
|
output, err := protoToJSONString(m, "", " ")
|
2020-12-04 04:36:16 +03:00
|
|
|
if err != nil {
|
2022-07-16 04:45:13 +03:00
|
|
|
fmt.Fprintf(os.Stdout, "%v\n", m)
|
|
|
|
base.Fatalf("error encode json: %s", err)
|
2020-12-04 04:36:16 +03:00
|
|
|
}
|
2022-07-16 04:45:13 +03:00
|
|
|
fmt.Println(output)
|
2020-12-04 04:36:16 +03:00
|
|
|
}
|
2020-12-04 17:03:07 +03:00
|
|
|
|
|
|
|
func isNil(i interface{}) bool {
|
|
|
|
vi := reflect.ValueOf(i)
|
|
|
|
if vi.Kind() == reflect.Ptr {
|
|
|
|
return vi.IsNil()
|
|
|
|
}
|
|
|
|
return i == nil
|
|
|
|
}
|