From fce5950bb63a1d9290feee371276bb87bd9f6dbb Mon Sep 17 00:00:00 2001 From: Brian Goff Date: Tue, 30 Sep 2014 22:06:31 -0400 Subject: [PATCH] Fix #6231 - Accept chunked encoding on start Docker-DCO-1.1-Signed-off-by: Brian Goff (github: cpuguy83) Upstream-commit: 86bfb9bca6a5e1e78b01741b076505c137f52673 Component: engine --- components/engine/api/server/server.go | 7 +- components/engine/integration/api_test.go | 80 +++++++++++++++++++++++ 2 files changed, 86 insertions(+), 1 deletion(-) diff --git a/components/engine/api/server/server.go b/components/engine/api/server/server.go index 3e5510e11c..27fe305106 100644 --- a/components/engine/api/server/server.go +++ b/components/engine/api/server/server.go @@ -760,8 +760,13 @@ func postContainersStart(eng *engine.Engine, version version.Version, w http.Res job = eng.Job("start", name) ) + // If contentLength is -1, we can assumed chunked encoding + // or more technically that the length is unknown + // http://golang.org/src/pkg/net/http/request.go#L139 + // net/http otherwise seems to swallow any headers related to chunked encoding + // including r.TransferEncoding // allow a nil body for backwards compatibility - if r.Body != nil && r.ContentLength > 0 { + if r.Body != nil && (r.ContentLength > 0 || r.ContentLength == -1) { if err := checkForJson(r); err != nil { return err } diff --git a/components/engine/integration/api_test.go b/components/engine/integration/api_test.go index c326465526..8fa295e7b1 100644 --- a/components/engine/integration/api_test.go +++ b/components/engine/integration/api_test.go @@ -1072,6 +1072,86 @@ func TestPostContainersCopyWhenContainerNotFound(t *testing.T) { } } +// Regression test for https://github.com/docker/docker/issues/6231 +func TestConstainersStartChunkedEncodingHostConfig(t *testing.T) { + eng := NewTestEngine(t) + defer mkDaemonFromEngine(eng, t).Nuke() + + r := httptest.NewRecorder() + + var testData engine.Env + testData.Set("Image", "docker-test-image") + testData.SetAuto("Volumes", map[string]struct{}{"/foo": {}}) + testData.Set("Cmd", "true") + jsonData := bytes.NewBuffer(nil) + if err := testData.Encode(jsonData); err != nil { + t.Fatal(err) + } + + req, err := http.NewRequest("POST", "/containers/create?name=chunk_test", jsonData) + if err != nil { + t.Fatal(err) + } + + req.Header.Add("Content-Type", "application/json") + if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil { + t.Fatal(err) + } + assertHttpNotError(r, t) + + var testData2 engine.Env + testData2.SetAuto("Binds", []string{"/tmp:/foo"}) + jsonData = bytes.NewBuffer(nil) + if err := testData2.Encode(jsonData); err != nil { + t.Fatal(err) + } + + req, err = http.NewRequest("POST", "/containers/chunk_test/start", jsonData) + if err != nil { + t.Fatal(err) + } + + req.Header.Add("Content-Type", "application/json") + // This is a cheat to make the http request do chunked encoding + // Otherwise (just setting the Content-Encoding to chunked) net/http will overwrite + // http://golang.org/src/pkg/net/http/request.go?s=11980:12172 + req.ContentLength = -1 + if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil { + t.Fatal(err) + } + assertHttpNotError(r, t) + + type config struct { + HostConfig struct { + Binds []string + } + } + + req, err = http.NewRequest("GET", "/containers/chunk_test/json", nil) + if err != nil { + t.Fatal(err) + } + + r2 := httptest.NewRecorder() + req.Header.Add("Content-Type", "application/json") + if err := server.ServeRequest(eng, api.APIVERSION, r2, req); err != nil { + t.Fatal(err) + } + assertHttpNotError(r, t) + + c := config{} + + json.Unmarshal(r2.Body.Bytes(), &c) + + if len(c.HostConfig.Binds) == 0 { + t.Fatal("Chunked Encoding not handled") + } + + if c.HostConfig.Binds[0] != "/tmp:/foo" { + t.Fatal("Chunked encoding not properly handled, execpted binds to be /tmp:/foo, got:", c.HostConfig.Binds[0]) + } +} + // Mocked types for tests type NopConn struct { io.ReadCloser