mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-07-05 04:14:21 +00:00
7fe082a7f1
Xray counts client traffic globally per email, so a client attached to several of a node's inbounds has its single shared counter copied onto every inbound by the node's enriched inbound list. When those copies diverge (legacy per-inbound rows surviving a v3.2.x->v3.3.x upgrade, or any drift) the per-inbound delta loop read the lower sibling as a node-counter reset and re-added its full value, inflating the client far past real usage (#5274). Fold each email to its per-field node-wide max before the delta loop so every occurrence is equal: the per-email baseline dedup then holds and the reset clamp never misfires.
202 lines
6.8 KiB
Go
202 lines
6.8 KiB
Go
package link
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"net/url"
|
|
"reflect"
|
|
"testing"
|
|
)
|
|
|
|
func TestDefaultPort(t *testing.T) {
|
|
cases := []struct {
|
|
in string
|
|
def int
|
|
want int
|
|
}{
|
|
{"", 443, 443},
|
|
{"8080", 443, 8080},
|
|
{"0", 443, 443}, // non-positive falls back
|
|
{"-1", 443, 443}, // negative falls back
|
|
{"abc", 443, 443}, // unparseable falls back
|
|
{"65535", 443, 65535},
|
|
}
|
|
for _, c := range cases {
|
|
if got := defaultPort(c.in, c.def); got != c.want {
|
|
t.Errorf("defaultPort(%q,%d) = %d, want %d", c.in, c.def, got, c.want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestFirstNonEmptyAndParam(t *testing.T) {
|
|
if got := firstNonEmpty("a", "b"); got != "a" {
|
|
t.Errorf("firstNonEmpty(a,b) = %q, want a", got)
|
|
}
|
|
if got := firstNonEmpty("", "b"); got != "b" {
|
|
t.Errorf("firstNonEmpty(,b) = %q, want b", got)
|
|
}
|
|
p := url.Values{"x": {""}, "y": {"hit"}, "z": {"z"}}
|
|
if got := firstParam(p, "x", "y", "z"); got != "hit" {
|
|
t.Errorf("firstParam = %q, want hit (first non-empty)", got)
|
|
}
|
|
if got := firstParam(p, "x"); got != "" {
|
|
t.Errorf("firstParam(only empty) = %q, want empty", got)
|
|
}
|
|
}
|
|
|
|
func TestSplitComma(t *testing.T) {
|
|
if got := splitComma(""); got != nil {
|
|
t.Errorf("splitComma(empty) = %v, want nil", got)
|
|
}
|
|
if got := splitComma("a, ,b ,, c"); !reflect.DeepEqual(got, []string{"a", "b", "c"}) {
|
|
t.Errorf("splitComma trim/skip = %v, want [a b c]", got)
|
|
}
|
|
if got := splitCommaOrDefault("", []string{"d"}); !reflect.DeepEqual(got, []string{"d"}) {
|
|
t.Errorf("splitCommaOrDefault(empty) = %v, want [d]", got)
|
|
}
|
|
if got := splitCommaOrDefault("x,y", []string{"d"}); !reflect.DeepEqual(got, []string{"x", "y"}) {
|
|
t.Errorf("splitCommaOrDefault(x,y) = %v, want [x y]", got)
|
|
}
|
|
}
|
|
|
|
func TestPadAndBase64DecodeFlexible(t *testing.T) {
|
|
if got := padBase64("abc"); got != "abc=" {
|
|
t.Errorf("padBase64(abc) = %q, want abc=", got)
|
|
}
|
|
if got := padBase64("abcd"); got != "abcd" {
|
|
t.Errorf("padBase64(abcd) = %q, want unchanged", got)
|
|
}
|
|
std := base64.StdEncoding.EncodeToString([]byte("aes-256-gcm:secret"))
|
|
if got, err := base64DecodeFlexible(std); err != nil || got != "aes-256-gcm:secret" {
|
|
t.Errorf("base64DecodeFlexible(std) = (%q,%v), want (aes-256-gcm:secret,nil)", got, err)
|
|
}
|
|
rawURL := base64.RawURLEncoding.EncodeToString([]byte("m:p"))
|
|
if got, err := base64DecodeFlexible(rawURL); err != nil || got != "m:p" {
|
|
t.Errorf("base64DecodeFlexible(rawurl) = (%q,%v), want (m:p,nil)", got, err)
|
|
}
|
|
if _, err := base64DecodeFlexible("!!!not!!!"); err == nil {
|
|
t.Error("base64DecodeFlexible(garbage) should error")
|
|
}
|
|
}
|
|
|
|
func TestDecodeHash(t *testing.T) {
|
|
if got := decodeHash(""); got != "" {
|
|
t.Errorf("decodeHash(empty) = %q, want empty", got)
|
|
}
|
|
if got := decodeHash("a%20b"); got != "a b" {
|
|
t.Errorf("decodeHash(a%%20b) = %q, want 'a b'", got)
|
|
}
|
|
if got := decodeHash("plain"); got != "plain" {
|
|
t.Errorf("decodeHash(plain) = %q, want plain", got)
|
|
}
|
|
}
|
|
|
|
func TestCanonicalQuery_SortsKeys(t *testing.T) {
|
|
// unsorted input must come out key-sorted for a stable identity
|
|
got := canonicalQuery(url.Values{"c": {"3"}, "a": {"1"}, "b": {"2"}})
|
|
if got != "a=1&b=2&c=3" {
|
|
t.Fatalf("canonicalQuery = %q, want a=1&b=2&c=3", got)
|
|
}
|
|
}
|
|
|
|
// stream navigates res.Outbound["streamSettings"][key] as a map.
|
|
func streamSub(t *testing.T, res *ParseResult, key string) map[string]any {
|
|
t.Helper()
|
|
ss, _ := res.Outbound["streamSettings"].(map[string]any)
|
|
m, ok := ss[key].(map[string]any)
|
|
if !ok {
|
|
t.Fatalf("streamSettings.%s missing/not a map: %#v", key, ss)
|
|
}
|
|
return m
|
|
}
|
|
|
|
func TestParse_RealitySecurityMapped(t *testing.T) {
|
|
res, err := ParseLink("vless://uuid@h.com:443?type=tcp&security=reality&pbk=PBK&sid=SID&sni=SNI&fp=firefox&spx=%2Fspx&pqv=PQV")
|
|
if err != nil {
|
|
t.Fatalf("parse: %v", err)
|
|
}
|
|
re := streamSub(t, res, "realitySettings")
|
|
for k, want := range map[string]string{"publicKey": "PBK", "shortId": "SID", "serverName": "SNI", "fingerprint": "firefox", "spiderX": "/spx", "mldsa65Verify": "PQV"} {
|
|
if re[k] != want {
|
|
t.Errorf("realitySettings[%q] = %v, want %q", k, re[k], want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestParse_TLSSecurityMapped(t *testing.T) {
|
|
res, err := ParseLink("trojan://pw@h.com:443?type=tcp&security=tls&sni=SNI&fp=chrome&alpn=h2,http/1.1&ech=ECH&pcs=PCS")
|
|
if err != nil {
|
|
t.Fatalf("parse: %v", err)
|
|
}
|
|
tls := streamSub(t, res, "tlsSettings")
|
|
if tls["serverName"] != "SNI" || tls["fingerprint"] != "chrome" || tls["echConfigList"] != "ECH" || tls["pinnedPeerCertSha256"] != "PCS" {
|
|
t.Errorf("tlsSettings fields = %#v", tls)
|
|
}
|
|
if alpn, _ := tls["alpn"].([]string); !reflect.DeepEqual(alpn, []string{"h2", "http/1.1"}) {
|
|
t.Errorf("alpn = %#v, want [h2 http/1.1]", tls["alpn"])
|
|
}
|
|
}
|
|
|
|
func TestParse_WSAndGRPCTransport(t *testing.T) {
|
|
ws, err := ParseLink("vless://uuid@h.com:443?type=ws&host=H&path=%2Fwspath")
|
|
if err != nil {
|
|
t.Fatalf("parse ws: %v", err)
|
|
}
|
|
wss := streamSub(t, ws, "wsSettings")
|
|
if wss["host"] != "H" || wss["path"] != "/wspath" {
|
|
t.Errorf("wsSettings = %#v, want host=H path=/wspath", wss)
|
|
}
|
|
|
|
grpc, err := ParseLink("vless://uuid@h.com:443?type=grpc&serviceName=svc&authority=auth&mode=multi")
|
|
if err != nil {
|
|
t.Fatalf("parse grpc: %v", err)
|
|
}
|
|
gs := streamSub(t, grpc, "grpcSettings")
|
|
if gs["serviceName"] != "svc" || gs["authority"] != "auth" || gs["multiMode"] != true {
|
|
t.Errorf("grpcSettings = %#v, want serviceName=svc authority=auth multiMode=true", gs)
|
|
}
|
|
}
|
|
|
|
func TestParse_TCPHTTPHeader(t *testing.T) {
|
|
res, err := ParseLink("vless://uuid@h.com:443?type=tcp&headerType=http&host=ex.com&path=%2F")
|
|
if err != nil {
|
|
t.Fatalf("parse: %v", err)
|
|
}
|
|
tcp := streamSub(t, res, "tcpSettings")
|
|
header, _ := tcp["header"].(map[string]any)
|
|
if header["type"] != "http" {
|
|
t.Errorf("tcp header type = %v, want http", header["type"])
|
|
}
|
|
}
|
|
|
|
func TestParseVless_CoreFields(t *testing.T) {
|
|
res, err := ParseLink("vless://the-uuid@9.9.9.9:8443?type=tcp&security=none&flow=xtls-rprx-vision#tag1")
|
|
if err != nil {
|
|
t.Fatalf("parse: %v", err)
|
|
}
|
|
st, _ := res.Outbound["settings"].(map[string]any)
|
|
if st["address"] != "9.9.9.9" || st["port"] != 8443 || st["id"] != "the-uuid" || st["flow"] != "xtls-rprx-vision" {
|
|
t.Errorf("vless settings = %#v", st)
|
|
}
|
|
}
|
|
|
|
func TestParseTrojanAndSS_CoreFields(t *testing.T) {
|
|
tr, err := ParseLink("trojan://secret@t.com:443?type=tcp&security=tls#tj")
|
|
if err != nil {
|
|
t.Fatalf("parse trojan: %v", err)
|
|
}
|
|
srv := tr.Outbound["settings"].(map[string]any)["servers"].([]any)[0].(map[string]any)
|
|
if srv["address"] != "t.com" || srv["port"] != 443 || srv["password"] != "secret" {
|
|
t.Errorf("trojan server = %#v", srv)
|
|
}
|
|
|
|
ssLink := "ss://" + base64.StdEncoding.EncodeToString([]byte("aes-256-gcm:sspass")) + "@s.com:8388#ss1"
|
|
ss, err := ParseLink(ssLink)
|
|
if err != nil {
|
|
t.Fatalf("parse ss: %v", err)
|
|
}
|
|
ssrv := ss.Outbound["settings"].(map[string]any)["servers"].([]any)[0].(map[string]any)
|
|
if ssrv["address"] != "s.com" || ssrv["port"] != 8388 || ssrv["password"] != "sspass" || ssrv["method"] != "aes-256-gcm" {
|
|
t.Errorf("ss server = %#v", ssrv)
|
|
}
|
|
}
|