| package proxyurl |
|
|
| import ( |
| "strings" |
| "testing" |
| ) |
|
|
| func TestParse_空字符串直连(t *testing.T) { |
| trimmed, parsed, err := Parse("") |
| if err != nil { |
| t.Fatalf("空字符串应直连: %v", err) |
| } |
| if trimmed != "" { |
| t.Errorf("trimmed 应为空: got %q", trimmed) |
| } |
| if parsed != nil { |
| t.Errorf("parsed 应为 nil: got %v", parsed) |
| } |
| } |
|
|
| func TestParse_空白字符串直连(t *testing.T) { |
| trimmed, parsed, err := Parse(" ") |
| if err != nil { |
| t.Fatalf("空白字符串应直连: %v", err) |
| } |
| if trimmed != "" { |
| t.Errorf("trimmed 应为空: got %q", trimmed) |
| } |
| if parsed != nil { |
| t.Errorf("parsed 应为 nil: got %v", parsed) |
| } |
| } |
|
|
| func TestParse_有效HTTP代理(t *testing.T) { |
| trimmed, parsed, err := Parse("http://proxy.example.com:8080") |
| if err != nil { |
| t.Fatalf("有效 HTTP 代理应成功: %v", err) |
| } |
| if trimmed != "http://proxy.example.com:8080" { |
| t.Errorf("trimmed 不匹配: got %q", trimmed) |
| } |
| if parsed == nil { |
| t.Fatal("parsed 不应为 nil") |
| } |
| if parsed.Host != "proxy.example.com:8080" { |
| t.Errorf("Host 不匹配: got %q", parsed.Host) |
| } |
| } |
|
|
| func TestParse_有效HTTPS代理(t *testing.T) { |
| _, parsed, err := Parse("https://proxy.example.com:443") |
| if err != nil { |
| t.Fatalf("有效 HTTPS 代理应成功: %v", err) |
| } |
| if parsed.Scheme != "https" { |
| t.Errorf("Scheme 不匹配: got %q", parsed.Scheme) |
| } |
| } |
|
|
| func TestParse_有效SOCKS5代理_自动升级为SOCKS5H(t *testing.T) { |
| trimmed, parsed, err := Parse("socks5://127.0.0.1:1080") |
| if err != nil { |
| t.Fatalf("有效 SOCKS5 代理应成功: %v", err) |
| } |
| |
| if trimmed != "socks5h://127.0.0.1:1080" { |
| t.Errorf("trimmed 应升级为 socks5h: got %q", trimmed) |
| } |
| if parsed.Scheme != "socks5h" { |
| t.Errorf("Scheme 应升级为 socks5h: got %q", parsed.Scheme) |
| } |
| } |
|
|
| func TestParse_无效URL(t *testing.T) { |
| _, _, err := Parse("://invalid") |
| if err == nil { |
| t.Fatal("无效 URL 应返回错误") |
| } |
| if !strings.Contains(err.Error(), "invalid proxy URL") { |
| t.Errorf("错误信息应包含 'invalid proxy URL': got %s", err.Error()) |
| } |
| } |
|
|
| func TestParse_缺少Host(t *testing.T) { |
| _, _, err := Parse("http://") |
| if err == nil { |
| t.Fatal("缺少 host 应返回错误") |
| } |
| if !strings.Contains(err.Error(), "missing host") { |
| t.Errorf("错误信息应包含 'missing host': got %s", err.Error()) |
| } |
| } |
|
|
| func TestParse_不支持的Scheme(t *testing.T) { |
| _, _, err := Parse("ftp://proxy.example.com:21") |
| if err == nil { |
| t.Fatal("不支持的 scheme 应返回错误") |
| } |
| if !strings.Contains(err.Error(), "unsupported proxy scheme") { |
| t.Errorf("错误信息应包含 'unsupported proxy scheme': got %s", err.Error()) |
| } |
| } |
|
|
| func TestParse_含密码URL脱敏(t *testing.T) { |
| |
| trimmed, parsed, err := Parse("socks5://user:secret_password@proxy.local:1080") |
| if err != nil { |
| t.Fatalf("含密码的有效 URL 应成功: %v", err) |
| } |
| if trimmed == "" || parsed == nil { |
| t.Fatal("应返回非空结果") |
| } |
| if parsed.Scheme != "socks5h" { |
| t.Errorf("Scheme 应升级为 socks5h: got %q", parsed.Scheme) |
| } |
| if !strings.HasPrefix(trimmed, "socks5h://") { |
| t.Errorf("trimmed 应以 socks5h:// 开头: got %q", trimmed) |
| } |
| if parsed.User == nil { |
| t.Error("升级后应保留 UserInfo") |
| } |
|
|
| |
| _, _, err = Parse("http://user:secret_password@:0/") |
| if err == nil { |
| t.Fatal("缺少 host 应返回错误") |
| } |
| if strings.Contains(err.Error(), "secret_password") { |
| t.Error("错误信息不应包含明文密码") |
| } |
| if !strings.Contains(err.Error(), "missing host") { |
| t.Errorf("错误信息应包含 'missing host': got %s", err.Error()) |
| } |
| } |
|
|
| func TestParse_带空白的有效URL(t *testing.T) { |
| trimmed, parsed, err := Parse(" http://proxy.example.com:8080 ") |
| if err != nil { |
| t.Fatalf("带空白的有效 URL 应成功: %v", err) |
| } |
| if trimmed != "http://proxy.example.com:8080" { |
| t.Errorf("trimmed 应去除空白: got %q", trimmed) |
| } |
| if parsed == nil { |
| t.Fatal("parsed 不应为 nil") |
| } |
| } |
|
|
| func TestParse_Scheme大小写不敏感(t *testing.T) { |
| |
| trimmed, parsed, err := Parse("SOCKS5://proxy.example.com:1080") |
| if err != nil { |
| t.Fatalf("大写 SOCKS5 应被接受: %v", err) |
| } |
| if parsed.Scheme != "socks5h" { |
| t.Errorf("大写 SOCKS5 Scheme 应升级为 socks5h: got %q", parsed.Scheme) |
| } |
| if !strings.HasPrefix(trimmed, "socks5h://") { |
| t.Errorf("大写 SOCKS5 trimmed 应升级为 socks5h://: got %q", trimmed) |
| } |
|
|
| |
| _, _, err = Parse("HTTP://proxy.example.com:8080") |
| if err != nil { |
| t.Fatalf("大写 HTTP 应被接受: %v", err) |
| } |
| } |
|
|
| func TestParse_带认证的有效代理(t *testing.T) { |
| trimmed, parsed, err := Parse("http://user:pass@proxy.example.com:8080") |
| if err != nil { |
| t.Fatalf("带认证的代理 URL 应成功: %v", err) |
| } |
| if parsed.User == nil { |
| t.Error("应保留 UserInfo") |
| } |
| if trimmed != "http://user:pass@proxy.example.com:8080" { |
| t.Errorf("trimmed 不匹配: got %q", trimmed) |
| } |
| } |
|
|
| func TestParse_IPv6地址(t *testing.T) { |
| trimmed, parsed, err := Parse("http://[::1]:8080") |
| if err != nil { |
| t.Fatalf("IPv6 代理 URL 应成功: %v", err) |
| } |
| if parsed.Hostname() != "::1" { |
| t.Errorf("Hostname 不匹配: got %q", parsed.Hostname()) |
| } |
| if trimmed != "http://[::1]:8080" { |
| t.Errorf("trimmed 不匹配: got %q", trimmed) |
| } |
| } |
|
|
| func TestParse_SOCKS5H保持不变(t *testing.T) { |
| trimmed, parsed, err := Parse("socks5h://proxy.local:1080") |
| if err != nil { |
| t.Fatalf("有效 SOCKS5H 代理应成功: %v", err) |
| } |
| |
| if trimmed != "socks5h://proxy.local:1080" { |
| t.Errorf("trimmed 不应变化: got %q", trimmed) |
| } |
| if parsed.Scheme != "socks5h" { |
| t.Errorf("Scheme 应保持 socks5h: got %q", parsed.Scheme) |
| } |
| } |
|
|
| func TestParse_无Scheme裸地址(t *testing.T) { |
| |
| _, _, err := Parse("proxy.example.com:8080") |
| if err == nil { |
| t.Fatal("无 scheme 的裸地址应返回错误") |
| } |
| } |
|
|