|
@@ -24,6 +24,7 @@ type serverResponse struct {
|
|
body io.ReadCloser
|
|
body io.ReadCloser
|
|
header http.Header
|
|
header http.Header
|
|
statusCode int
|
|
statusCode int
|
|
|
|
+ reqURL *url.URL
|
|
}
|
|
}
|
|
|
|
|
|
// head sends an http request to the docker API using the method HEAD.
|
|
// head sends an http request to the docker API using the method HEAD.
|
|
@@ -118,11 +119,18 @@ func (cli *Client) sendRequest(ctx context.Context, method, path string, query u
|
|
if err != nil {
|
|
if err != nil {
|
|
return serverResponse{}, err
|
|
return serverResponse{}, err
|
|
}
|
|
}
|
|
- return cli.doRequest(ctx, req)
|
|
|
|
|
|
+ resp, err := cli.doRequest(ctx, req)
|
|
|
|
+ if err != nil {
|
|
|
|
+ return resp, err
|
|
|
|
+ }
|
|
|
|
+ if err := cli.checkResponseErr(resp); err != nil {
|
|
|
|
+ return resp, err
|
|
|
|
+ }
|
|
|
|
+ return resp, nil
|
|
}
|
|
}
|
|
|
|
|
|
func (cli *Client) doRequest(ctx context.Context, req *http.Request) (serverResponse, error) {
|
|
func (cli *Client) doRequest(ctx context.Context, req *http.Request) (serverResponse, error) {
|
|
- serverResp := serverResponse{statusCode: -1}
|
|
|
|
|
|
+ serverResp := serverResponse{statusCode: -1, reqURL: req.URL}
|
|
|
|
|
|
resp, err := ctxhttp.Do(ctx, cli.client, req)
|
|
resp, err := ctxhttp.Do(ctx, cli.client, req)
|
|
if err != nil {
|
|
if err != nil {
|
|
@@ -179,35 +187,42 @@ func (cli *Client) doRequest(ctx context.Context, req *http.Request) (serverResp
|
|
|
|
|
|
if resp != nil {
|
|
if resp != nil {
|
|
serverResp.statusCode = resp.StatusCode
|
|
serverResp.statusCode = resp.StatusCode
|
|
|
|
+ serverResp.body = resp.Body
|
|
|
|
+ serverResp.header = resp.Header
|
|
}
|
|
}
|
|
|
|
+ return serverResp, nil
|
|
|
|
+}
|
|
|
|
|
|
- if serverResp.statusCode < 200 || serverResp.statusCode >= 400 {
|
|
|
|
- body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
- if err != nil {
|
|
|
|
- return serverResp, err
|
|
|
|
- }
|
|
|
|
- if len(body) == 0 {
|
|
|
|
- return serverResp, fmt.Errorf("Error: request returned %s for API route and version %s, check if the server supports the requested API version", http.StatusText(serverResp.statusCode), req.URL)
|
|
|
|
- }
|
|
|
|
|
|
+func (cli *Client) checkResponseErr(serverResp serverResponse) error {
|
|
|
|
+ if serverResp.statusCode >= 200 && serverResp.statusCode < 400 {
|
|
|
|
+ return nil
|
|
|
|
+ }
|
|
|
|
|
|
- var errorMessage string
|
|
|
|
- if (cli.version == "" || versions.GreaterThan(cli.version, "1.23")) &&
|
|
|
|
- resp.Header.Get("Content-Type") == "application/json" {
|
|
|
|
- var errorResponse types.ErrorResponse
|
|
|
|
- if err := json.Unmarshal(body, &errorResponse); err != nil {
|
|
|
|
- return serverResp, fmt.Errorf("Error reading JSON: %v", err)
|
|
|
|
- }
|
|
|
|
- errorMessage = errorResponse.Message
|
|
|
|
- } else {
|
|
|
|
- errorMessage = string(body)
|
|
|
|
- }
|
|
|
|
|
|
+ body, err := ioutil.ReadAll(serverResp.body)
|
|
|
|
+ if err != nil {
|
|
|
|
+ return err
|
|
|
|
+ }
|
|
|
|
+ if len(body) == 0 {
|
|
|
|
+ return fmt.Errorf("Error: request returned %s for API route and version %s, check if the server supports the requested API version", http.StatusText(serverResp.statusCode), serverResp.reqURL)
|
|
|
|
+ }
|
|
|
|
|
|
- return serverResp, fmt.Errorf("Error response from daemon: %s", strings.TrimSpace(errorMessage))
|
|
|
|
|
|
+ var ct string
|
|
|
|
+ if serverResp.header != nil {
|
|
|
|
+ ct = serverResp.header.Get("Content-Type")
|
|
}
|
|
}
|
|
|
|
|
|
- serverResp.body = resp.Body
|
|
|
|
- serverResp.header = resp.Header
|
|
|
|
- return serverResp, nil
|
|
|
|
|
|
+ var errorMessage string
|
|
|
|
+ if (cli.version == "" || versions.GreaterThan(cli.version, "1.23")) && ct == "application/json" {
|
|
|
|
+ var errorResponse types.ErrorResponse
|
|
|
|
+ if err := json.Unmarshal(body, &errorResponse); err != nil {
|
|
|
|
+ return fmt.Errorf("Error reading JSON: %v", err)
|
|
|
|
+ }
|
|
|
|
+ errorMessage = errorResponse.Message
|
|
|
|
+ } else {
|
|
|
|
+ errorMessage = string(body)
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return fmt.Errorf("Error response from daemon: %s", strings.TrimSpace(errorMessage))
|
|
}
|
|
}
|
|
|
|
|
|
func (cli *Client) addHeaders(req *http.Request, headers headers) *http.Request {
|
|
func (cli *Client) addHeaders(req *http.Request, headers headers) *http.Request {
|
|
@@ -239,9 +254,9 @@ func encodeData(data interface{}) (*bytes.Buffer, error) {
|
|
}
|
|
}
|
|
|
|
|
|
func ensureReaderClosed(response serverResponse) {
|
|
func ensureReaderClosed(response serverResponse) {
|
|
- if body := response.body; body != nil {
|
|
|
|
|
|
+ if response.body != nil {
|
|
// Drain up to 512 bytes and close the body to let the Transport reuse the connection
|
|
// Drain up to 512 bytes and close the body to let the Transport reuse the connection
|
|
- io.CopyN(ioutil.Discard, body, 512)
|
|
|
|
|
|
+ io.CopyN(ioutil.Discard, response.body, 512)
|
|
response.body.Close()
|
|
response.body.Close()
|
|
}
|
|
}
|
|
}
|
|
}
|