2014-12-10 05:25:54 +00:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
2015-04-17 20:32:51 +00:00
|
|
|
"encoding/json"
|
2014-12-10 05:25:54 +00:00
|
|
|
"fmt"
|
2015-04-17 20:32:51 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
2014-12-10 05:25:54 +00:00
|
|
|
|
2015-04-17 20:32:51 +00:00
|
|
|
"github.com/docker/distribution/registry/api/v2"
|
2014-12-10 05:25:54 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// BlobUploadNotFoundError is returned when making a blob upload operation against an
|
|
|
|
// invalid blob upload location url.
|
|
|
|
// This may be the result of using a cancelled, completed, or stale upload
|
|
|
|
// location.
|
|
|
|
type BlobUploadNotFoundError struct {
|
|
|
|
Location string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *BlobUploadNotFoundError) Error() string {
|
|
|
|
return fmt.Sprintf("No blob upload found at Location: %s", e.Location)
|
|
|
|
}
|
|
|
|
|
|
|
|
// BlobUploadInvalidRangeError is returned when attempting to upload an image
|
|
|
|
// blob chunk that is out of order.
|
|
|
|
// This provides the known BlobSize and LastValidRange which can be used to
|
|
|
|
// resume the upload.
|
|
|
|
type BlobUploadInvalidRangeError struct {
|
|
|
|
Location string
|
|
|
|
LastValidRange int
|
|
|
|
BlobSize int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *BlobUploadInvalidRangeError) Error() string {
|
|
|
|
return fmt.Sprintf(
|
|
|
|
"Invalid range provided for upload at Location: %s. Last Valid Range: %d, Blob Size: %d",
|
|
|
|
e.Location, e.LastValidRange, e.BlobSize)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnexpectedHTTPStatusError is returned when an unexpected HTTP status is
|
|
|
|
// returned when making a registry api call.
|
|
|
|
type UnexpectedHTTPStatusError struct {
|
|
|
|
Status string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *UnexpectedHTTPStatusError) Error() string {
|
|
|
|
return fmt.Sprintf("Received unexpected HTTP status: %s", e.Status)
|
|
|
|
}
|
2015-04-17 20:32:51 +00:00
|
|
|
|
|
|
|
// UnexpectedHTTPResponseError is returned when an expected HTTP status code
|
|
|
|
// is returned, but the content was unexpected and failed to be parsed.
|
|
|
|
type UnexpectedHTTPResponseError struct {
|
|
|
|
ParseErr error
|
|
|
|
Response []byte
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *UnexpectedHTTPResponseError) Error() string {
|
|
|
|
shortenedResponse := string(e.Response)
|
|
|
|
if len(shortenedResponse) > 15 {
|
|
|
|
shortenedResponse = shortenedResponse[:12] + "..."
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("Error parsing HTTP response: %s: %q", e.ParseErr.Error(), shortenedResponse)
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseHTTPErrorResponse(response *http.Response) error {
|
|
|
|
var errors v2.Errors
|
|
|
|
body, err := ioutil.ReadAll(response.Body)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-05-07 23:11:04 +00:00
|
|
|
|
|
|
|
if err := json.Unmarshal(body, &errors); err != nil {
|
2015-04-17 20:32:51 +00:00
|
|
|
return &UnexpectedHTTPResponseError{
|
|
|
|
ParseErr: err,
|
|
|
|
Response: body,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return &errors
|
|
|
|
}
|
2015-05-09 00:40:30 +00:00
|
|
|
|
|
|
|
func handleErrorResponse(resp *http.Response) error {
|
|
|
|
if resp.StatusCode >= 400 && resp.StatusCode < 500 {
|
|
|
|
return parseHTTPErrorResponse(resp)
|
|
|
|
}
|
|
|
|
return &UnexpectedHTTPStatusError{Status: resp.Status}
|
|
|
|
}
|