mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-25 11:33:11 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			114 lines
		
	
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			114 lines
		
	
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2017 The Forgejo Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package unittest
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"net/http"
 | |
| 	"net/http/httptest"
 | |
| 	"net/url"
 | |
| 	"os"
 | |
| 	"slices"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"code.gitea.io/gitea/modules/log"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| // Mocks HTTP responses of a third-party service (such as GitHub, GitLab…)
 | |
| // This has two modes:
 | |
| //   - live mode: the requests made to the mock HTTP server are transmitted to the live
 | |
| //     service, and responses are saved as test data files
 | |
| //   - test mode: the responses to requests to the mock HTTP server are read from the
 | |
| //     test data files
 | |
| func NewMockWebServer(t *testing.T, liveServerBaseURL, testDataDir string, liveMode bool) *httptest.Server {
 | |
| 	mockServerBaseURL := ""
 | |
| 	ignoredHeaders := []string{"cf-ray", "server", "date", "report-to", "nel", "x-request-id"}
 | |
| 
 | |
| 	server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		path := NormalizedFullPath(r.URL)
 | |
| 		log.Info("Mock HTTP Server: got request for path %s", r.URL.Path)
 | |
| 		// TODO check request method (support POST?)
 | |
| 		fixturePath := fmt.Sprintf("%s/%s_%s", testDataDir, r.Method, url.PathEscape(path))
 | |
| 		if liveMode {
 | |
| 			liveURL := fmt.Sprintf("%s%s", liveServerBaseURL, path)
 | |
| 
 | |
| 			request, err := http.NewRequest(r.Method, liveURL, nil)
 | |
| 			assert.NoError(t, err, "constructing an HTTP request to %s failed", liveURL)
 | |
| 			for headerName, headerValues := range r.Header {
 | |
| 				// do not pass on the encoding: let the Transport of the HTTP client handle that for us
 | |
| 				if strings.ToLower(headerName) != "accept-encoding" {
 | |
| 					for _, headerValue := range headerValues {
 | |
| 						request.Header.Add(headerName, headerValue)
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			response, err := http.DefaultClient.Do(request)
 | |
| 			assert.NoError(t, err, "HTTP request to %s failed: %s", liveURL)
 | |
| 			assert.Less(t, response.StatusCode, 400, "unexpected status code for %s", liveURL)
 | |
| 
 | |
| 			fixture, err := os.Create(fixturePath)
 | |
| 			assert.NoError(t, err, "failed to open the fixture file %s for writing", fixturePath)
 | |
| 			defer fixture.Close()
 | |
| 			fixtureWriter := bufio.NewWriter(fixture)
 | |
| 
 | |
| 			for headerName, headerValues := range response.Header {
 | |
| 				for _, headerValue := range headerValues {
 | |
| 					if !slices.Contains(ignoredHeaders, strings.ToLower(headerName)) {
 | |
| 						_, err := fixtureWriter.WriteString(fmt.Sprintf("%s: %s\n", headerName, headerValue))
 | |
| 						assert.NoError(t, err, "writing the header of the HTTP response to the fixture file failed")
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			_, err = fixtureWriter.WriteString("\n")
 | |
| 			assert.NoError(t, err, "writing the header of the HTTP response to the fixture file failed")
 | |
| 			fixtureWriter.Flush()
 | |
| 
 | |
| 			log.Info("Mock HTTP Server: writing response to %s", fixturePath)
 | |
| 			_, err = io.Copy(fixture, response.Body)
 | |
| 			assert.NoError(t, err, "writing the body of the HTTP response to %s failed", liveURL)
 | |
| 
 | |
| 			err = fixture.Sync()
 | |
| 			assert.NoError(t, err, "writing the body of the HTTP response to the fixture file failed")
 | |
| 		}
 | |
| 
 | |
| 		fixture, err := os.ReadFile(fixturePath)
 | |
| 		assert.NoError(t, err, "missing mock HTTP response: "+fixturePath)
 | |
| 
 | |
| 		w.WriteHeader(http.StatusOK)
 | |
| 
 | |
| 		// replace any mention of the live HTTP service by the mocked host
 | |
| 		stringFixture := strings.ReplaceAll(string(fixture), liveServerBaseURL, mockServerBaseURL)
 | |
| 		// parse back the fixture file into a series of HTTP headers followed by response body
 | |
| 		lines := strings.Split(stringFixture, "\n")
 | |
| 		for idx, line := range lines {
 | |
| 			colonIndex := strings.Index(line, ": ")
 | |
| 			if colonIndex != -1 {
 | |
| 				w.Header().Set(line[0:colonIndex], line[colonIndex+2:])
 | |
| 			} else {
 | |
| 				// we reached the end of the headers (empty line), so what follows is the body
 | |
| 				responseBody := strings.Join(lines[idx+1:], "\n")
 | |
| 				_, err := w.Write([]byte(responseBody))
 | |
| 				assert.NoError(t, err, "writing the body of the HTTP response failed")
 | |
| 				break
 | |
| 			}
 | |
| 		}
 | |
| 	}))
 | |
| 	mockServerBaseURL = server.URL
 | |
| 	return server
 | |
| }
 | |
| 
 | |
| func NormalizedFullPath(url *url.URL) string {
 | |
| 	// TODO normalize path (remove trailing slash?)
 | |
| 	// TODO normalize RawQuery (order query parameters?)
 | |
| 	if len(url.Query()) == 0 {
 | |
| 		return url.EscapedPath()
 | |
| 	}
 | |
| 	return fmt.Sprintf("%s?%s", url.EscapedPath(), url.RawQuery)
 | |
| }
 |