| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133 |
- package external
- import (
- "bytes"
- "context"
- "net"
- "io"
- "net/http"
- "net/url"
- "os"
- "strings"
- "time"
- "github.com/xtls/xray-core/common/buf"
- "github.com/xtls/xray-core/common/errors"
- "github.com/xtls/xray-core/main/confloader"
- )
- func ConfigLoader(arg string) (out io.Reader, err error) {
- var data []byte
- switch {
- case strings.HasPrefix(arg, "http+unix://"):
- data, err = FetchUnixSocketHTTPContent(arg)
- case strings.HasPrefix(arg, "http://"), strings.HasPrefix(arg, "https://"):
- data, err = FetchHTTPContent(arg)
- case arg == "stdin:":
- data, err = io.ReadAll(os.Stdin)
- default:
- data, err = os.ReadFile(arg)
- }
- if err != nil {
- return
- }
- out = bytes.NewBuffer(data)
- return
- }
- func FetchHTTPContent(target string) ([]byte, error) {
- parsedTarget, err := url.Parse(target)
- if err != nil {
- return nil, errors.New("invalid URL: ", target).Base(err)
- }
- if s := strings.ToLower(parsedTarget.Scheme); s != "http" && s != "https" {
- return nil, errors.New("invalid scheme: ", 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, errors.New("failed to dial to ", target).Base(err)
- }
- defer resp.Body.Close()
- if resp.StatusCode != 200 {
- return nil, errors.New("unexpected HTTP status code: ", resp.StatusCode)
- }
- content, err := buf.ReadAllToBytes(resp.Body)
- if err != nil {
- return nil, errors.New("failed to read HTTP response").Base(err)
- }
- return content, nil
- }
- // Format: http+unix:///path/to/socket.sock/api/endpoint
- func FetchUnixSocketHTTPContent(target string) ([]byte, error) {
- path := strings.TrimPrefix(target, "http+unix://")
-
- if !strings.HasPrefix(path, "/") {
- return nil, errors.New("unix socket path must be absolute")
- }
-
- var socketPath, httpPath string
-
- sockIdx := strings.Index(path, ".sock")
- if sockIdx != -1 {
- socketPath = path[:sockIdx+5]
- httpPath = path[sockIdx+5:]
- if httpPath == "" {
- httpPath = "/"
- }
- } else {
- return nil, errors.New("cannot determine socket path, socket file should have .sock extension")
- }
-
- if _, err := os.Stat(socketPath); err != nil {
- return nil, errors.New("socket file not found: ", socketPath).Base(err)
- }
-
- client := &http.Client{
- Timeout: 30 * time.Second,
- Transport: &http.Transport{
- DialContext: func(ctx context.Context, _, _ string) (net.Conn, error) {
- var d net.Dialer
- return d.DialContext(ctx, "unix", socketPath)
- },
- },
- }
- defer client.CloseIdleConnections()
-
- resp, err := client.Get("http://localhost" + httpPath)
- if err != nil {
- return nil, errors.New("failed to fetch from unix socket: ", socketPath).Base(err)
- }
- defer resp.Body.Close()
-
- if resp.StatusCode != 200 {
- return nil, errors.New("unexpected HTTP status code: ", resp.StatusCode)
- }
-
- content, err := buf.ReadAllToBytes(resp.Body)
- if err != nil {
- return nil, errors.New("failed to read response").Base(err)
- }
-
- return content, nil
- }
- func init() {
- confloader.EffectiveConfigFileLoader = ConfigLoader
- }
|