Commit bd42e919 authored by Nick Thomas's avatar Nick Thomas

Test the behaviour of SendURL with less-common upstreams

Two cases in particular, lacking a `Content-Type`,  gave us trouble:

* Transfer-Encoding: chunked
* No content-type and no transfer-encoding

Both of these are permitted by the HTTP RFC (cases 3 and 7), and we
can talk to arbitrary HTTP servers via sendurl, so it's imperative that
we handle them correctly. This commit adds tests for both cases.

Responses of the latter type are transparently converted to responses
of the former type. This is an automatic behaviour of the Go stdlib,
which doesn't really support making the second type of response
directly. Since Transfer-Encoding is a hop-by-hop header, this type of
encoding is extremely common, and we're still streaming, instead of
accumulating, the data, I think this is acceptable.
parent 05bf7368
...@@ -148,7 +148,7 @@ func (e *entry) Inject(w http.ResponseWriter, r *http.Request, sendData string) ...@@ -148,7 +148,7 @@ func (e *entry) Inject(w http.ResponseWriter, r *http.Request, sendData string)
return return
} }
// fix issue #267, where Content-Length was set prior to injection // Prevent Go from adding a Content-Length header automatically
w.Header().Del("Content-Length") w.Header().Del("Content-Length")
// copy response headers and body, except the headers from preserveHeaderKeys // copy response headers and body, except the headers from preserveHeaderKeys
...@@ -165,7 +165,7 @@ func (e *entry) Inject(w http.ResponseWriter, r *http.Request, sendData string) ...@@ -165,7 +165,7 @@ func (e *entry) Inject(w http.ResponseWriter, r *http.Request, sendData string)
if err != nil { if err != nil {
sendURLRequestsRequestFailed.Inc() sendURLRequestsRequestFailed.Inc()
helper.Fail500(w, r, fmt.Errorf("SendURL: Copy response: %v", err)) helper.LogError(r, fmt.Errorf("SendURL: Copy response: %v", err))
return return
} }
......
...@@ -14,6 +14,8 @@ import ( ...@@ -14,6 +14,8 @@ import (
"os/exec" "os/exec"
"path" "path"
"regexp" "regexp"
"strconv"
"strings"
"testing" "testing"
"time" "time"
...@@ -366,21 +368,58 @@ func TestArtifactsGetSingleFile(t *testing.T) { ...@@ -366,21 +368,58 @@ func TestArtifactsGetSingleFile(t *testing.T) {
} }
func TestSendURLForArtifacts(t *testing.T) { func TestSendURLForArtifacts(t *testing.T) {
fileContents := "12345678901234567890\n" expectedBody := strings.Repeat("CONTENT!", 1024)
server := httptest.NewServer(http.FileServer(http.Dir("testdata"))) regularHandler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Header().Set("Content-Length", strconv.Itoa(len(expectedBody)))
w.Write([]byte(expectedBody))
})
chunkedHandler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Header().Set("Transfer-Encoding", "chunked")
w.Write([]byte(expectedBody))
})
rawHandler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
hj, ok := w.(http.Hijacker)
require.Equal(t, true, ok)
conn, buf, err := hj.Hijack()
require.NoError(t, err)
defer conn.Close()
defer buf.Flush()
fmt.Fprint(buf, "HTTP/1.1 200 OK\r\nContent-Type: application/zip\r\n\r\n")
fmt.Fprint(buf, expectedBody)
})
for _, tc := range []struct {
name string
handler http.Handler
transferEncoding []string
contentLength int
}{
{"No content-length, chunked TE", chunkedHandler, []string{"chunked"}, -1}, // Case 3 in https://tools.ietf.org/html/rfc7230#section-3.3.2
{"Known content-length, identity TE", regularHandler, nil, len(expectedBody)}, // Case 5 in https://tools.ietf.org/html/rfc7230#section-3.3.2
{"No content-length, identity TE", rawHandler, []string{"chunked"}, -1}, // Case 7 in https://tools.ietf.org/html/rfc7230#section-3.3.2
} {
t.Run(tc.name, func(t *testing.T) {
server := httptest.NewServer(tc.handler)
defer server.Close() defer server.Close()
// We manually created this txt file in the gitlab-workhorse Git repository jsonParams := fmt.Sprintf(`{"URL":%q}`, server.URL)
url := server.URL + "/test-file.txt"
jsonParams := fmt.Sprintf(`{"URL":%q}`, url)
resourcePath := `/namespace/project/builds/123/artifacts/file/download` resourcePath := `/namespace/project/builds/123/artifacts/file/download`
resp, body, err := doSendDataRequest(resourcePath, "send-url", jsonParams) resp, body, err := doSendDataRequest(resourcePath, "send-url", jsonParams)
require.NoError(t, err) require.NoError(t, err)
assert.Equal(t, 200, resp.StatusCode, "GET %q: status code", resourcePath) require.Equal(t, http.StatusOK, resp.StatusCode, "GET %q: status code", resourcePath)
assert.Equal(t, fileContents, string(body), "GET %q: response body", resourcePath) require.Equal(t, int64(tc.contentLength), resp.ContentLength, "GET %q: Content-Length", resourcePath)
require.Equal(t, tc.transferEncoding, resp.TransferEncoding, "GET %q: Transfer-Encoding", resourcePath)
require.Equal(t, expectedBody, string(body), "GET %q: response body", resourcePath)
assertNginxResponseBuffering(t, "no", resp, "GET %q: nginx response buffering", resourcePath)
})
}
} }
func TestApiContentTypeBlock(t *testing.T) { func TestApiContentTypeBlock(t *testing.T) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment