dae/cmd/internal/subscription.go

185 lines
4.5 KiB
Go

package internal
import (
"bufio"
"bytes"
"encoding/json"
"fmt"
"github.com/sirupsen/logrus"
"github.com/v2rayA/dae/common"
"io"
"net"
"net/http"
"net/url"
"os"
"path/filepath"
"strconv"
"strings"
)
type sip008 struct {
Version int `json:"version"`
Servers []sip008Server `json:"servers"`
BytesUsed int64 `json:"bytes_used"`
BytesRemaining int64 `json:"bytes_remaining"`
}
type sip008Server struct {
Id string `json:"id"`
Remarks string `json:"remarks"`
Server string `json:"server"`
ServerPort int `json:"server_port"`
Password string `json:"password"`
Method string `json:"method"`
Plugin string `json:"plugin"`
PluginOpts string `json:"plugin_opts"`
}
func resolveSubscriptionAsBase64(log *logrus.Logger, b []byte) (nodes []string) {
log.Debugln("Try to resolve as base64")
// base64 decode
raw, e := common.Base64StdDecode(string(b))
if e != nil {
raw, _ = common.Base64UrlDecode(string(b))
}
// Simply check and preprocess.
lines := strings.Split(raw, "\n")
for _, line := range lines {
line = strings.TrimSpace(line)
if line == "" {
continue
}
protocol, suffix, _ := strings.Cut(line, "://")
if len(protocol) == 0 || len(suffix) == 0 {
continue
}
nodes = append(nodes, line)
}
return nodes
}
func resolveSubscriptionAsSIP008(log *logrus.Logger, b []byte) (nodes []string, err error) {
log.Debugln("Try to resolve as sip008")
var sip sip008
err = json.Unmarshal(b, &sip)
if err != nil {
return nil, fmt.Errorf("failed to unmarshal json to sip008")
}
if sip.Version != 1 || sip.Servers == nil {
return nil, fmt.Errorf("does not seems like a standard sip008 subscription")
}
for _, server := range sip.Servers {
u := url.URL{
Scheme: "ss",
User: url.UserPassword(server.Method, server.Password),
Host: net.JoinHostPort(server.Server, strconv.Itoa(server.ServerPort)),
RawQuery: url.Values{"plugin": []string{server.PluginOpts}}.Encode(),
Fragment: server.Remarks,
}
nodes = append(nodes, u.String())
}
return nodes, nil
}
func resolveFile(u *url.URL, configDir string) (b []byte, err error) {
if u.Host == "" {
return nil, fmt.Errorf("not support absolute path")
}
/// Relative location.
// Make sure path is secure.
path := filepath.Join(configDir, u.Host, u.Path)
if err = common.EnsureFileInSubDir(path, configDir); err != nil {
return nil, err
}
/// Read and resolve.
f, err := os.Open(path)
if err != nil {
return nil, err
}
// Check file access.
fi, err := f.Stat()
if err != nil {
return nil, err
}
if fi.IsDir() {
return nil, fmt.Errorf("subscription file cannot be a directory: %v", path)
}
if fi.Mode()&0037 > 0 {
return nil, fmt.Errorf("permissions %04o for '%v' are too open; requires the file is NOT writable by the same group and NOT accessible by others; suggest 0640 or 0600", fi.Mode()&0777, path)
}
// Resolve the first line instruction.
fReader := bufio.NewReader(f)
b, err = fReader.Peek(1)
if err != nil {
return nil, err
}
if string(b[0]) == "@" {
// Instruction line. But not support yet.
_, _, err = fReader.ReadLine()
if err != nil {
return nil, err
}
}
b, err = io.ReadAll(fReader)
if err != nil {
return nil, err
}
return bytes.TrimSpace(b), err
}
func ResolveSubscription(log *logrus.Logger, configDir string, subscription string) (tag string, nodes []string, err error) {
/// Get tag.
iColon := strings.Index(subscription, ":")
if iColon == -1 {
goto parseUrl
}
// If first colon is like "://" in "scheme://linkbody", no tag is present.
if strings.HasPrefix(subscription[iColon:], "://") {
goto parseUrl
}
// Else tag is the part before colon.
tag = subscription[:iColon]
subscription = subscription[iColon+1:]
/// Parse url.
parseUrl:
u, err := url.Parse(subscription)
if err != nil {
return tag, nil, fmt.Errorf("failed to parse subscription \"%v\": %w", subscription, err)
}
log.Debugf("ResolveSubscription: %v", subscription)
var (
b []byte
resp *http.Response
)
switch u.Scheme {
case "file":
b, err = resolveFile(u, configDir)
if err != nil {
return "", nil, err
}
goto resolve
default:
}
resp, err = http.Get(subscription)
if err != nil {
return "", nil, err
}
defer resp.Body.Close()
b, err = io.ReadAll(resp.Body)
if err != nil {
return "", nil, err
}
resolve:
if nodes, err = resolveSubscriptionAsSIP008(log, b); err == nil {
return tag, nodes, nil
} else {
log.Debugln(err)
}
return tag, resolveSubscriptionAsBase64(log, b), nil
}