Merge pull request #750 from aaronlehmann/response-code-tolerance
Make the registry client more tolerant about HTTP status codesmaster
						commit
						e833456266
					
				|  | @ -10,6 +10,7 @@ import ( | |||
| 	"sync" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/docker/distribution/registry/client" | ||||
| 	"github.com/docker/distribution/registry/client/transport" | ||||
| ) | ||||
| 
 | ||||
|  | @ -209,7 +210,7 @@ func (th *tokenHandler) fetchToken(params map[string]string) (token string, err | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	if resp.StatusCode != http.StatusOK { | ||||
| 	if !client.SuccessStatus(resp.StatusCode) { | ||||
| 		return "", fmt.Errorf("token auth attempt for registry: %s request failed with status: %d %s", req.URL, resp.StatusCode, http.StatusText(resp.StatusCode)) | ||||
| 	} | ||||
| 
 | ||||
|  |  | |||
|  | @ -44,7 +44,7 @@ func (hbu *httpBlobUpload) ReadFrom(r io.Reader) (n int64, err error) { | |||
| 		return 0, err | ||||
| 	} | ||||
| 
 | ||||
| 	if resp.StatusCode != http.StatusAccepted { | ||||
| 	if !SuccessStatus(resp.StatusCode) { | ||||
| 		return 0, hbu.handleErrorResponse(resp) | ||||
| 	} | ||||
| 
 | ||||
|  | @ -79,7 +79,7 @@ func (hbu *httpBlobUpload) Write(p []byte) (n int, err error) { | |||
| 		return 0, err | ||||
| 	} | ||||
| 
 | ||||
| 	if resp.StatusCode != http.StatusAccepted { | ||||
| 	if !SuccessStatus(resp.StatusCode) { | ||||
| 		return 0, hbu.handleErrorResponse(resp) | ||||
| 	} | ||||
| 
 | ||||
|  | @ -142,7 +142,7 @@ func (hbu *httpBlobUpload) Commit(ctx context.Context, desc distribution.Descrip | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	if resp.StatusCode != http.StatusCreated { | ||||
| 	if !SuccessStatus(resp.StatusCode) { | ||||
| 		return distribution.Descriptor{}, hbu.handleErrorResponse(resp) | ||||
| 	} | ||||
| 
 | ||||
|  | @ -160,12 +160,10 @@ func (hbu *httpBlobUpload) Cancel(ctx context.Context) error { | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusNoContent, http.StatusNotFound: | ||||
| 	if resp.StatusCode == http.StatusNotFound || SuccessStatus(resp.StatusCode) { | ||||
| 		return nil | ||||
| 	default: | ||||
| 		return hbu.handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return hbu.handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| func (hbu *httpBlobUpload) Close() error { | ||||
|  |  | |||
|  | @ -61,3 +61,9 @@ func handleErrorResponse(resp *http.Response) error { | |||
| 	} | ||||
| 	return &UnexpectedHTTPStatusError{Status: resp.Status} | ||||
| } | ||||
| 
 | ||||
| // SuccessStatus returns true if the argument is a successful HTTP response
 | ||||
| // code (in the range 200 - 399 inclusive).
 | ||||
| func SuccessStatus(status int) bool { | ||||
| 	return status >= 200 && status <= 399 | ||||
| } | ||||
|  |  | |||
|  | @ -70,8 +70,7 @@ func (r *registry) Repositories(ctx context.Context, entries []string, last stri | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusOK: | ||||
| 	if SuccessStatus(resp.StatusCode) { | ||||
| 		var ctlg struct { | ||||
| 			Repositories []string `json:"repositories"` | ||||
| 		} | ||||
|  | @ -90,8 +89,7 @@ func (r *registry) Repositories(ctx context.Context, entries []string, last stri | |||
| 		if link == "" { | ||||
| 			returnErr = io.EOF | ||||
| 		} | ||||
| 
 | ||||
| 	default: | ||||
| 	} else { | ||||
| 		return 0, handleErrorResponse(resp) | ||||
| 	} | ||||
| 
 | ||||
|  | @ -199,8 +197,7 @@ func (ms *manifests) Tags() ([]string, error) { | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusOK: | ||||
| 	if SuccessStatus(resp.StatusCode) { | ||||
| 		b, err := ioutil.ReadAll(resp.Body) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
|  | @ -214,11 +211,10 @@ func (ms *manifests) Tags() ([]string, error) { | |||
| 		} | ||||
| 
 | ||||
| 		return tagsResponse.Tags, nil | ||||
| 	case http.StatusNotFound: | ||||
| 	} else if resp.StatusCode == http.StatusNotFound { | ||||
| 		return nil, nil | ||||
| 	default: | ||||
| 		return nil, handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return nil, handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| func (ms *manifests) Exists(dgst digest.Digest) (bool, error) { | ||||
|  | @ -238,14 +234,12 @@ func (ms *manifests) ExistsByTag(tag string) (bool, error) { | |||
| 		return false, err | ||||
| 	} | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusOK: | ||||
| 	if SuccessStatus(resp.StatusCode) { | ||||
| 		return true, nil | ||||
| 	case http.StatusNotFound: | ||||
| 	} else if resp.StatusCode == http.StatusNotFound { | ||||
| 		return false, nil | ||||
| 	default: | ||||
| 		return false, handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return false, handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| func (ms *manifests) Get(dgst digest.Digest) (*manifest.SignedManifest, error) { | ||||
|  | @ -294,8 +288,9 @@ func (ms *manifests) GetByTag(tag string, options ...distribution.ManifestServic | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusOK: | ||||
| 	if resp.StatusCode == http.StatusNotModified { | ||||
| 		return nil, nil | ||||
| 	} else if SuccessStatus(resp.StatusCode) { | ||||
| 		var sm manifest.SignedManifest | ||||
| 		decoder := json.NewDecoder(resp.Body) | ||||
| 
 | ||||
|  | @ -303,11 +298,8 @@ func (ms *manifests) GetByTag(tag string, options ...distribution.ManifestServic | |||
| 			return nil, err | ||||
| 		} | ||||
| 		return &sm, nil | ||||
| 	case http.StatusNotModified: | ||||
| 		return nil, nil | ||||
| 	default: | ||||
| 		return nil, handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return nil, handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| func (ms *manifests) Put(m *manifest.SignedManifest) error { | ||||
|  | @ -329,13 +321,11 @@ func (ms *manifests) Put(m *manifest.SignedManifest) error { | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusAccepted: | ||||
| 	if SuccessStatus(resp.StatusCode) { | ||||
| 		// TODO(dmcgowan): make use of digest header
 | ||||
| 		return nil | ||||
| 	default: | ||||
| 		return handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| func (ms *manifests) Delete(dgst digest.Digest) error { | ||||
|  | @ -354,12 +344,10 @@ func (ms *manifests) Delete(dgst digest.Digest) error { | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusAccepted: | ||||
| 	if SuccessStatus(resp.StatusCode) { | ||||
| 		return nil | ||||
| 	default: | ||||
| 		return handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| type blobs struct { | ||||
|  | @ -461,8 +449,7 @@ func (bs *blobs) Create(ctx context.Context) (distribution.BlobWriter, error) { | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusAccepted: | ||||
| 	if SuccessStatus(resp.StatusCode) { | ||||
| 		// TODO(dmcgowan): Check for invalid UUID
 | ||||
| 		uuid := resp.Header.Get("Docker-Upload-UUID") | ||||
| 		location, err := sanitizeLocation(resp.Header.Get("Location"), u) | ||||
|  | @ -477,9 +464,8 @@ func (bs *blobs) Create(ctx context.Context) (distribution.BlobWriter, error) { | |||
| 			startedAt: time.Now(), | ||||
| 			location:  location, | ||||
| 		}, nil | ||||
| 	default: | ||||
| 		return nil, handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return nil, handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| func (bs *blobs) Resume(ctx context.Context, id string) (distribution.BlobWriter, error) { | ||||
|  | @ -508,8 +494,7 @@ func (bs *blobStatter) Stat(ctx context.Context, dgst digest.Digest) (distributi | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusOK: | ||||
| 	if SuccessStatus(resp.StatusCode) { | ||||
| 		lengthHeader := resp.Header.Get("Content-Length") | ||||
| 		length, err := strconv.ParseInt(lengthHeader, 10, 64) | ||||
| 		if err != nil { | ||||
|  | @ -521,11 +506,10 @@ func (bs *blobStatter) Stat(ctx context.Context, dgst digest.Digest) (distributi | |||
| 			Size:      length, | ||||
| 			Digest:    dgst, | ||||
| 		}, nil | ||||
| 	case http.StatusNotFound: | ||||
| 	} else if resp.StatusCode == http.StatusNotFound { | ||||
| 		return distribution.Descriptor{}, distribution.ErrBlobUnknown | ||||
| 	default: | ||||
| 		return distribution.Descriptor{}, handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return distribution.Descriptor{}, handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| func buildCatalogValues(maxEntries int, last string) url.Values { | ||||
|  | @ -559,12 +543,10 @@ func (bs *blobStatter) Clear(ctx context.Context, dgst digest.Digest) error { | |||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 
 | ||||
| 	switch resp.StatusCode { | ||||
| 	case http.StatusAccepted: | ||||
| 	if SuccessStatus(resp.StatusCode) { | ||||
| 		return nil | ||||
| 	default: | ||||
| 		return handleErrorResponse(resp) | ||||
| 	} | ||||
| 	return handleErrorResponse(resp) | ||||
| } | ||||
| 
 | ||||
| func (bs *blobStatter) SetDescriptor(ctx context.Context, dgst digest.Digest, desc distribution.Descriptor) error { | ||||
|  |  | |||
|  | @ -154,10 +154,11 @@ func (hrs *httpReadSeeker) reader() (io.Reader, error) { | |||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	switch { | ||||
| 	case resp.StatusCode == 200: | ||||
| 	// Normally would use client.SuccessStatus, but that would be a cyclic
 | ||||
| 	// import
 | ||||
| 	if resp.StatusCode >= 200 && resp.StatusCode <= 399 { | ||||
| 		hrs.rc = resp.Body | ||||
| 	default: | ||||
| 	} else { | ||||
| 		defer resp.Body.Close() | ||||
| 		return nil, fmt.Errorf("unexpected status resolving reader: %v", resp.Status) | ||||
| 	} | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue