tls_client_test.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. package runtime
  2. import (
  3. "context"
  4. "crypto/sha256"
  5. "encoding/base64"
  6. "encoding/hex"
  7. "net/http"
  8. "net/http/httptest"
  9. "net/url"
  10. "strconv"
  11. "strings"
  12. "testing"
  13. "github.com/mhsanaei/3x-ui/v3/internal/database/model"
  14. )
  15. // nodeForServer builds a node pointing at a loopback test server (loopback is
  16. // SSRF-blocked, so AllowPrivateAddress is set for the guarded dialer).
  17. func nodeForServer(t *testing.T, srv *httptest.Server, mode, pin string) *model.Node {
  18. t.Helper()
  19. u, err := url.Parse(srv.URL)
  20. if err != nil {
  21. t.Fatalf("parse server url: %v", err)
  22. }
  23. port, err := strconv.Atoi(u.Port())
  24. if err != nil {
  25. t.Fatalf("parse server port: %v", err)
  26. }
  27. return &model.Node{
  28. Id: 1,
  29. Name: "n1",
  30. Scheme: "https",
  31. Address: u.Hostname(),
  32. Port: port,
  33. BasePath: "/",
  34. ApiToken: "token",
  35. Enable: true,
  36. AllowPrivateAddress: true,
  37. TlsVerifyMode: mode,
  38. PinnedCertSha256: pin,
  39. }
  40. }
  41. func leafPinBase64(srv *httptest.Server) string {
  42. sum := sha256.Sum256(srv.Certificate().Raw)
  43. return base64.StdEncoding.EncodeToString(sum[:])
  44. }
  45. // A self-signed node must be reachable by Remote ops under skip/pin and
  46. // rejected under verify — the split issue #5264 reported.
  47. func TestRemoteHonorsTLSVerifyMode(t *testing.T) {
  48. srv := httptest.NewTLSServer(http.HandlerFunc(func(w http.ResponseWriter, _ *http.Request) {
  49. w.Header().Set("Content-Type", "application/json")
  50. _, _ = w.Write([]byte(`{"success":true,"obj":[]}`))
  51. }))
  52. defer srv.Close()
  53. goodPin := leafPinBase64(srv)
  54. wrongPin := base64.StdEncoding.EncodeToString(make([]byte, sha256.Size))
  55. cases := []struct {
  56. name string
  57. mode string
  58. pin string
  59. wantErr bool
  60. }{
  61. {"verify rejects self-signed", "verify", "", true},
  62. {"skip accepts self-signed", "skip", "", false},
  63. {"pin accepts matching cert", "pin", goodPin, false},
  64. {"pin rejects mismatched cert", "pin", wrongPin, true},
  65. }
  66. for _, c := range cases {
  67. t.Run(c.name, func(t *testing.T) {
  68. r := NewRemote(nodeForServer(t, srv, c.mode, c.pin))
  69. _, err := r.ListInboundOptions(context.Background())
  70. if c.wantErr && err == nil {
  71. t.Fatalf("mode %q: expected error, got nil", c.mode)
  72. }
  73. if !c.wantErr && err != nil {
  74. t.Fatalf("mode %q: unexpected error: %v", c.mode, err)
  75. }
  76. })
  77. }
  78. }
  79. // The lazily-built client is cached for the Remote's lifetime so repeated
  80. // operations reuse one pooled transport rather than rebuilding TLS each call.
  81. func TestRemoteClientCached(t *testing.T) {
  82. r := NewRemote(&model.Node{Scheme: "https", TlsVerifyMode: "skip"})
  83. c1, err1 := r.httpClient()
  84. c2, err2 := r.httpClient()
  85. if err1 != nil || err2 != nil {
  86. t.Fatalf("httpClient errors: %v %v", err1, err2)
  87. }
  88. if c1 != c2 {
  89. t.Fatal("expected the same cached client across calls")
  90. }
  91. }
  92. func TestHTTPClientForNodeVerifyShared(t *testing.T) {
  93. // verify mode and plain http both reuse the shared default client.
  94. for _, n := range []*model.Node{
  95. {Scheme: "https", TlsVerifyMode: "verify"},
  96. {Scheme: "https", TlsVerifyMode: ""},
  97. {Scheme: "http", TlsVerifyMode: "skip"},
  98. } {
  99. c, err := HTTPClientForNode(n)
  100. if err != nil {
  101. t.Fatalf("HTTPClientForNode(%+v): %v", n, err)
  102. }
  103. if c != defaultNodeHTTPClient {
  104. t.Fatalf("HTTPClientForNode(%+v) = %p, want shared default %p", n, c, defaultNodeHTTPClient)
  105. }
  106. }
  107. }
  108. func TestHTTPClientForNodePinInvalid(t *testing.T) {
  109. if _, err := HTTPClientForNode(&model.Node{Scheme: "https", TlsVerifyMode: "pin", PinnedCertSha256: "not-a-pin"}); err == nil {
  110. t.Fatal("expected error for invalid pin")
  111. }
  112. }
  113. func TestDecodeCertPin(t *testing.T) {
  114. raw := sha256.Sum256([]byte("cert"))
  115. hexColon := strings.ToUpper(hex.EncodeToString(raw[:]))
  116. // reinsert colons in openssl -fingerprint style
  117. var withColons strings.Builder
  118. for i := 0; i < len(hexColon); i += 2 {
  119. if i > 0 {
  120. withColons.WriteByte(':')
  121. }
  122. withColons.WriteString(hexColon[i : i+2])
  123. }
  124. cases := []struct {
  125. name string
  126. in string
  127. wantErr bool
  128. }{
  129. {"base64 std", base64.StdEncoding.EncodeToString(raw[:]), false},
  130. {"base64 raw url", base64.RawURLEncoding.EncodeToString(raw[:]), false},
  131. {"hex bare", hex.EncodeToString(raw[:]), false},
  132. {"hex colon openssl", withColons.String(), false},
  133. {"empty", "", true},
  134. {"garbage", "not-a-pin", true},
  135. }
  136. for _, c := range cases {
  137. t.Run(c.name, func(t *testing.T) {
  138. got, err := DecodeCertPin(c.in)
  139. if c.wantErr {
  140. if err == nil {
  141. t.Fatalf("expected error for %q", c.in)
  142. }
  143. return
  144. }
  145. if err != nil {
  146. t.Fatalf("unexpected error for %q: %v", c.in, err)
  147. }
  148. if string(got) != string(raw[:]) {
  149. t.Fatalf("decoded bytes mismatch for %q", c.in)
  150. }
  151. })
  152. }
  153. }