mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-11-04 02:04:11 +01:00 
			
		
		
		
	Backport #35604 by silverwind The test calls out to a web service which may be down or unreachable as seen in the linked issue. It's better for tests to not have such external dependencies to make them absolutely stable. Fixes: https://github.com/go-gitea/gitea/issues/35571 Co-authored-by: silverwind <me@silverwind.io>
		
			
				
	
	
		
			138 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			138 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2023 The Gitea Authors. All rights reserved.
 | 
						|
// SPDX-License-Identifier: MIT
 | 
						|
 | 
						|
package hcaptcha
 | 
						|
 | 
						|
import (
 | 
						|
	"errors"
 | 
						|
	"io"
 | 
						|
	"net/http"
 | 
						|
	"net/url"
 | 
						|
	"os"
 | 
						|
	"strings"
 | 
						|
	"testing"
 | 
						|
	"time"
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	dummySiteKey = "10000000-ffff-ffff-ffff-000000000001"
 | 
						|
	dummySecret  = "0x0000000000000000000000000000000000000000"
 | 
						|
	dummyToken   = "10000000-aaaa-bbbb-cccc-000000000001"
 | 
						|
)
 | 
						|
 | 
						|
func TestMain(m *testing.M) {
 | 
						|
	os.Exit(m.Run())
 | 
						|
}
 | 
						|
 | 
						|
type mockTransport struct{}
 | 
						|
 | 
						|
func (mockTransport) RoundTrip(req *http.Request) (*http.Response, error) {
 | 
						|
	if req.URL.String() != verifyURL {
 | 
						|
		return nil, errors.New("unsupported url")
 | 
						|
	}
 | 
						|
 | 
						|
	body, err := io.ReadAll(req.Body)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	bodyValues, err := url.ParseQuery(string(body))
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	var responseText string
 | 
						|
	if bodyValues.Get("response") == dummyToken {
 | 
						|
		responseText = `{"success":true,"credit":false,"hostname":"dummy-key-pass","challenge_ts":"2025-10-08T16:02:56.136Z"}`
 | 
						|
	} else {
 | 
						|
		responseText = `{"success":false,"error-codes":["invalid-input-response"]}`
 | 
						|
	}
 | 
						|
 | 
						|
	return &http.Response{Request: req, Body: io.NopCloser(strings.NewReader(responseText))}, nil
 | 
						|
}
 | 
						|
 | 
						|
func TestCaptcha(t *testing.T) {
 | 
						|
	tt := []struct {
 | 
						|
		Name   string
 | 
						|
		Secret string
 | 
						|
		Token  string
 | 
						|
		Error  ErrorCode
 | 
						|
	}{
 | 
						|
		{
 | 
						|
			Name:   "Success",
 | 
						|
			Secret: dummySecret,
 | 
						|
			Token:  dummyToken,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			Name:  "Missing Secret",
 | 
						|
			Token: dummyToken,
 | 
						|
			Error: ErrMissingInputSecret,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			Name:   "Missing Token",
 | 
						|
			Secret: dummySecret,
 | 
						|
			Error:  ErrMissingInputResponse,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			Name:   "Invalid Token",
 | 
						|
			Secret: dummySecret,
 | 
						|
			Token:  "test",
 | 
						|
			Error:  ErrInvalidInputResponse,
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	for _, tc := range tt {
 | 
						|
		t.Run(tc.Name, func(t *testing.T) {
 | 
						|
			client, err := New(tc.Secret, WithHTTP(&http.Client{
 | 
						|
				Timeout:   time.Second * 5,
 | 
						|
				Transport: mockTransport{},
 | 
						|
			}))
 | 
						|
			if err != nil {
 | 
						|
				// The only error that can be returned from creating a client
 | 
						|
				if tc.Error == ErrMissingInputSecret && err == ErrMissingInputSecret {
 | 
						|
					return
 | 
						|
				}
 | 
						|
				t.Log(err)
 | 
						|
				t.FailNow()
 | 
						|
			}
 | 
						|
 | 
						|
			resp, err := client.Verify(tc.Token, PostOptions{
 | 
						|
				Sitekey: dummySiteKey,
 | 
						|
			})
 | 
						|
			if err != nil {
 | 
						|
				// The only error that can be returned prior to the request
 | 
						|
				if tc.Error == ErrMissingInputResponse && err == ErrMissingInputResponse {
 | 
						|
					return
 | 
						|
				}
 | 
						|
				t.Log(err)
 | 
						|
				t.FailNow()
 | 
						|
			}
 | 
						|
 | 
						|
			if tc.Error.String() != "" {
 | 
						|
				if resp.Success {
 | 
						|
					t.Log("Verification should fail.")
 | 
						|
					t.Fail()
 | 
						|
				}
 | 
						|
				if len(resp.ErrorCodes) == 0 {
 | 
						|
					t.Log("hCaptcha should have returned an error.")
 | 
						|
					t.Fail()
 | 
						|
				}
 | 
						|
				var hasErr bool
 | 
						|
				for _, err := range resp.ErrorCodes {
 | 
						|
					if strings.EqualFold(err.String(), tc.Error.String()) {
 | 
						|
						hasErr = true
 | 
						|
						break
 | 
						|
					}
 | 
						|
				}
 | 
						|
				if !hasErr {
 | 
						|
					t.Log("hCaptcha did not return the error being tested")
 | 
						|
					t.Fail()
 | 
						|
				}
 | 
						|
			} else if !resp.Success {
 | 
						|
				t.Log("Verification should succeed.")
 | 
						|
				t.Fail()
 | 
						|
			}
 | 
						|
		})
 | 
						|
	}
 | 
						|
}
 |