diff --git a/.travis.yml b/.travis.yml index 9d00fdd..fddfc4e 100644 --- a/.travis.yml +++ b/.travis.yml @@ -3,3 +3,5 @@ language: go go: - "1.8.x" - "1.10.x" + - "1.13.x" + - "1.14.x" diff --git a/error.go b/error.go index daab7e1..13aa35a 100644 --- a/error.go +++ b/error.go @@ -47,7 +47,6 @@ package errors import ( "bytes" - baseErrors "errors" "fmt" "reflect" "runtime" @@ -140,24 +139,6 @@ func WrapPrefix(e interface{}, prefix string, skip int) *Error { } -// Is detects whether the error is equal to a given error. Errors -// are considered equal by this function if they are matched by errors.Is -// or if their contained errors are matched through errors.Is -func Is(e error, original error) bool { - if baseErrors.Is(e, original) { - return true - } - - if e, ok := e.(*Error); ok { - return Is(e.Err, original) - } - - if original, ok := original.(*Error); ok { - return Is(e, original.Err) - } - - return false -} // Errorf creates a new error with the given message. You can use it // as a drop-in replacement for fmt.Errorf() to provide descriptive diff --git a/error_1_13.go b/error_1_13.go new file mode 100644 index 0000000..a81e24e --- /dev/null +++ b/error_1_13.go @@ -0,0 +1,26 @@ +// +build go1.13 + +package errors + +import ( + baseErrors "errors" +) + +// Is detects whether the error is equal to a given error. Errors +// are considered equal by this function if they are matched by errors.Is +// or if their contained errors are matched through errors.Is +func Is(e error, original error) bool { + if baseErrors.Is(e, original) { + return true + } + + if e, ok := e.(*Error); ok { + return Is(e.Err, original) + } + + if original, ok := original.(*Error); ok { + return Is(e, original.Err) + } + + return false +} diff --git a/error_1_13_test.go b/error_1_13_test.go new file mode 100644 index 0000000..92d3326 --- /dev/null +++ b/error_1_13_test.go @@ -0,0 +1,68 @@ +// +build go1.13 + +package errors + +import ( + "io" + "testing" +) + +// This test should work only for go 1.13 and latter +func TestIs113(t *testing.T) { + custErr := errorWithCustomIs{ + Key: "TestForFun", + Err: io.EOF, + } + + shouldMatch := errorWithCustomIs{ + Key: "TestForFun", + } + + shouldNotMatch := errorWithCustomIs{Key: "notOk"} + + if !Is(custErr, shouldMatch) { + t.Errorf("custErr is not a TestForFun customError") + } + + if Is(custErr, shouldNotMatch) { + t.Errorf("custErr is a notOk customError") + } + + if !Is(custErr, New(shouldMatch)) { + t.Errorf("custErr is not a New(TestForFun customError)") + } + + if Is(custErr, New(shouldNotMatch)) { + t.Errorf("custErr is a New(notOk customError)") + } + + if !Is(New(custErr), shouldMatch) { + t.Errorf("New(custErr) is not a TestForFun customError") + } + + if Is(New(custErr), shouldNotMatch) { + t.Errorf("New(custErr) is a notOk customError") + } + + if !Is(New(custErr), New(shouldMatch)) { + t.Errorf("New(custErr) is not a New(TestForFun customError)") + } + + if Is(New(custErr), New(shouldNotMatch)) { + t.Errorf("New(custErr) is a New(notOk customError)") + } +} + +type errorWithCustomIs struct { + Key string + Err error +} + +func (ewci errorWithCustomIs) Error() string { + return "["+ewci.Key+"]: " + ewci.Err.Error() +} + +func (ewci errorWithCustomIs) Is(target error) bool { + matched, ok := target.(errorWithCustomIs) + return ok && matched.Key == ewci.Key +} \ No newline at end of file diff --git a/error_backward.go b/error_backward.go new file mode 100644 index 0000000..d7e09a8 --- /dev/null +++ b/error_backward.go @@ -0,0 +1,22 @@ +// +build !go1.13 + +package errors + +// Is detects whether the error is equal to a given error. Errors +// are considered equal by this function if they are the same object, +// or if they both contain the same error inside an errors.Error. +func Is(e error, original error) bool { + if e == original { + return true + } + + if e, ok := e.(*Error); ok { + return Is(e.Err, original) + } + + if original, ok := original.(*Error); ok { + return Is(e, original.Err) + } + + return false +} \ No newline at end of file diff --git a/error_test.go b/error_test.go index 76f2f2e..d7b11f9 100644 --- a/error_test.go +++ b/error_test.go @@ -107,8 +107,8 @@ func TestNew(t *testing.T) { } } +// This test should work for any go version func TestIs(t *testing.T) { - if Is(nil, io.EOF) { t.Errorf("nil is an error") } @@ -128,51 +128,6 @@ func TestIs(t *testing.T) { if Is(io.EOF, fmt.Errorf("io.EOF")) { t.Errorf("io.EOF is fmt.Errorf") } - - custErr := errorWithCustomIs{ - Key: "TestForFun", - Err: io.EOF, - } - - shouldMatch := errorWithCustomIs{ - Key: "TestForFun", - } - - shouldNotMatch := errorWithCustomIs{Key: "notOk"} - - if !Is(custErr, shouldMatch) { - t.Errorf("custErr is not a TestForFun customError") - } - - if Is(custErr, shouldNotMatch) { - t.Errorf("custErr is a notOk customError") - } - - if !Is(custErr, New(shouldMatch)) { - t.Errorf("custErr is not a New(TestForFun customError)") - } - - if Is(custErr, New(shouldNotMatch)) { - t.Errorf("custErr is a New(notOk customError)") - } - - if !Is(New(custErr), shouldMatch) { - t.Errorf("New(custErr) is not a TestForFun customError") - } - - if Is(New(custErr), shouldNotMatch) { - t.Errorf("New(custErr) is a notOk customError") - } - - if !Is(New(custErr), New(shouldMatch)) { - t.Errorf("New(custErr) is not a New(TestForFun customError)") - } - - if Is(New(custErr), New(shouldNotMatch)) { - t.Errorf("New(custErr) is a New(notOk customError)") - } - - } func TestWrapError(t *testing.T) { @@ -361,17 +316,3 @@ func callersToFrames(callers []uintptr) []runtime.Frame { } } } - -type errorWithCustomIs struct { - Key string - Err error -} - -func (ewci errorWithCustomIs) Error() string { - return "["+ewci.Key+"]: " + ewci.Err.Error() -} - -func (ewci errorWithCustomIs) Is(target error) bool { - matched, ok := target.(errorWithCustomIs) - return ok && matched.Key == ewci.Key -} \ No newline at end of file