| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201 |
- 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://[email protected]: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://[email protected]: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://[email protected]: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://[email protected]: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://[email protected]: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://[email protected]: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://[email protected]: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)
- }
- }
|