mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
Disambiguate mirror -> other endpoint fallbacks from V2 -> V1
Signed-off-by: Jake Sanders <jsand@google.com>
This commit is contained in:
parent
3a9ab941ad
commit
305801f58f
4 changed files with 96 additions and 7 deletions
|
@ -126,21 +126,25 @@ func TranslatePullError(err error, ref reference.Named) error {
|
||||||
|
|
||||||
// continueOnError returns true if we should fallback to the next endpoint
|
// continueOnError returns true if we should fallback to the next endpoint
|
||||||
// as a result of this error.
|
// as a result of this error.
|
||||||
func continueOnError(err error) bool {
|
func continueOnError(err error, mirrorEndpoint bool) bool {
|
||||||
switch v := err.(type) {
|
switch v := err.(type) {
|
||||||
case errcode.Errors:
|
case errcode.Errors:
|
||||||
if len(v) == 0 {
|
if len(v) == 0 {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
return continueOnError(v[0])
|
return continueOnError(v[0], mirrorEndpoint)
|
||||||
case ErrNoSupport:
|
case ErrNoSupport:
|
||||||
return continueOnError(v.Err)
|
return continueOnError(v.Err, mirrorEndpoint)
|
||||||
case errcode.Error:
|
case errcode.Error:
|
||||||
return shouldV2Fallback(v)
|
return mirrorEndpoint || shouldV2Fallback(v)
|
||||||
case *client.UnexpectedHTTPResponseError:
|
case *client.UnexpectedHTTPResponseError:
|
||||||
return true
|
return true
|
||||||
case ImageConfigPullError:
|
case ImageConfigPullError:
|
||||||
return false
|
// ImageConfigPullError only happens with v2 images, v1 fallback is
|
||||||
|
// unnecessary.
|
||||||
|
// Failures from a mirror endpoint should result in fallback to the
|
||||||
|
// canonical repo.
|
||||||
|
return mirrorEndpoint
|
||||||
case error:
|
case error:
|
||||||
return !strings.Contains(err.Error(), strings.ToLower(syscall.ESRCH.Error()))
|
return !strings.Contains(err.Error(), strings.ToLower(syscall.ESRCH.Error()))
|
||||||
}
|
}
|
||||||
|
|
85
distribution/errors_test.go
Normal file
85
distribution/errors_test.go
Normal file
|
@ -0,0 +1,85 @@
|
||||||
|
package distribution
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"strings"
|
||||||
|
"syscall"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/docker/distribution/registry/api/errcode"
|
||||||
|
"github.com/docker/distribution/registry/api/v2"
|
||||||
|
"github.com/docker/distribution/registry/client"
|
||||||
|
)
|
||||||
|
|
||||||
|
var always_continue = []error{
|
||||||
|
&client.UnexpectedHTTPResponseError{},
|
||||||
|
|
||||||
|
// Some errcode.Errors that don't disprove the existence of a V1 image
|
||||||
|
errcode.Error{Code: errcode.ErrorCodeUnauthorized},
|
||||||
|
errcode.Error{Code: v2.ErrorCodeManifestUnknown},
|
||||||
|
errcode.Error{Code: v2.ErrorCodeNameUnknown},
|
||||||
|
|
||||||
|
errors.New("some totally unexpected error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
var continue_from_mirror_endpoint = []error{
|
||||||
|
ImageConfigPullError{},
|
||||||
|
|
||||||
|
// Some other errcode.Error that doesn't indicate we should search for a V1 image.
|
||||||
|
errcode.Error{Code: errcode.ErrorCodeTooManyRequests},
|
||||||
|
}
|
||||||
|
|
||||||
|
var never_continue = []error{
|
||||||
|
errors.New(strings.ToLower(syscall.ESRCH.Error())), // No such process
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestContinueOnError_NonMirrorEndpoint(t *testing.T) {
|
||||||
|
for _, err := range always_continue {
|
||||||
|
if !continueOnError(err, false) {
|
||||||
|
t.Errorf("Should continue from non-mirror endpoint: %T: '%s'", err, err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, err := range continue_from_mirror_endpoint {
|
||||||
|
if continueOnError(err, false) {
|
||||||
|
t.Errorf("Should only continue from mirror endpoint: %T: '%s'", err, err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestContinueOnError_MirrorEndpoint(t *testing.T) {
|
||||||
|
errs := []error{}
|
||||||
|
errs = append(errs, always_continue...)
|
||||||
|
errs = append(errs, continue_from_mirror_endpoint...)
|
||||||
|
for _, err := range errs {
|
||||||
|
if !continueOnError(err, true) {
|
||||||
|
t.Errorf("Should continue from mirror endpoint: %T: '%s'", err, err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestContinueOnError_NeverContinue(t *testing.T) {
|
||||||
|
for _, is_mirror_endpoint := range []bool{true, false} {
|
||||||
|
for _, err := range never_continue {
|
||||||
|
if continueOnError(err, is_mirror_endpoint) {
|
||||||
|
t.Errorf("Should never continue: %T: '%s'", err, err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestContinueOnError_UnnestsErrors(t *testing.T) {
|
||||||
|
// ContinueOnError should evaluate nested errcode.Errors.
|
||||||
|
|
||||||
|
// Assumes that v2.ErrorCodeNameUnknown is a continueable error code.
|
||||||
|
err := errcode.Errors{errcode.Error{Code: v2.ErrorCodeNameUnknown}}
|
||||||
|
if !continueOnError(err, false) {
|
||||||
|
t.Fatal("ContinueOnError should unnest, base return value on errcode.Errors")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Assumes that errcode.ErrorCodeTooManyRequests is not a V1-fallback indication
|
||||||
|
err = errcode.Errors{errcode.Error{Code: errcode.ErrorCodeTooManyRequests}}
|
||||||
|
if continueOnError(err, false) {
|
||||||
|
t.Fatal("ContinueOnError should unnest, base return value on errcode.Errors")
|
||||||
|
}
|
||||||
|
}
|
|
@ -74,7 +74,7 @@ func (p *v2Puller) Pull(ctx context.Context, ref reference.Named, platform strin
|
||||||
if _, ok := err.(fallbackError); ok {
|
if _, ok := err.(fallbackError); ok {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if continueOnError(err) {
|
if continueOnError(err, p.endpoint.Mirror) {
|
||||||
return fallbackError{
|
return fallbackError{
|
||||||
err: err,
|
err: err,
|
||||||
confirmedV2: p.confirmedV2,
|
confirmedV2: p.confirmedV2,
|
||||||
|
|
|
@ -67,7 +67,7 @@ func (p *v2Pusher) Push(ctx context.Context) (err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = p.pushV2Repository(ctx); err != nil {
|
if err = p.pushV2Repository(ctx); err != nil {
|
||||||
if continueOnError(err) {
|
if continueOnError(err, p.endpoint.Mirror) {
|
||||||
return fallbackError{
|
return fallbackError{
|
||||||
err: err,
|
err: err,
|
||||||
confirmedV2: p.pushState.confirmedV2,
|
confirmedV2: p.pushState.confirmedV2,
|
||||||
|
|
Loading…
Add table
Reference in a new issue