Source file src/errors/wrap.go

     1  // Copyright 2018 The Go Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  package errors
     6  
     7  import (
     8  	"internal/reflectlite"
     9  )
    10  
    11  // Unwrap returns the result of calling the Unwrap method on err, if err's
    12  // type contains an Unwrap method returning error.
    13  // Otherwise, Unwrap returns nil.
    14  func Unwrap(err error) error {
    15  	u, ok := err.(interface {
    16  		Unwrap() error
    17  	})
    18  	if !ok {
    19  		return nil
    20  	}
    21  	return u.Unwrap()
    22  }
    23  
    24  // Is reports whether any error in err's chain matches target.
    25  //
    26  // The chain consists of err itself followed by the sequence of errors obtained by
    27  // repeatedly calling Unwrap.
    28  //
    29  // An error is considered to match a target if it is equal to that target or if
    30  // it implements a method Is(error) bool such that Is(target) returns true.
    31  //
    32  // An error type might provide an Is method so it can be treated as equivalent
    33  // to an existing error. For example, if MyError defines
    34  //
    35  //	func (m MyError) Is(target error) bool { return target == fs.ErrExist }
    36  //
    37  // then Is(MyError{}, fs.ErrExist) returns true. See syscall.Errno.Is for
    38  // an example in the standard library. An Is method should only shallowly
    39  // compare err and the target and not call Unwrap on either.
    40  func Is(err, target error) bool {
    41  	if target == nil {
    42  		return err == target
    43  	}
    44  
    45  	isComparable := reflectlite.TypeOf(target).Comparable()
    46  	for {
    47  		if isComparable && err == target {
    48  			return true
    49  		}
    50  		if x, ok := err.(interface{ Is(error) bool }); ok && x.Is(target) {
    51  			return true
    52  		}
    53  		// TODO: consider supporting target.Is(err). This would allow
    54  		// user-definable predicates, but also may allow for coping with sloppy
    55  		// APIs, thereby making it easier to get away with them.
    56  		if err = Unwrap(err); err == nil {
    57  			return false
    58  		}
    59  	}
    60  }
    61  
    62  // As finds the first error in err's chain that matches target, and if one is found, sets
    63  // target to that error value and returns true. Otherwise, it returns false.
    64  //
    65  // The chain consists of err itself followed by the sequence of errors obtained by
    66  // repeatedly calling Unwrap.
    67  //
    68  // An error matches target if the error's concrete value is assignable to the value
    69  // pointed to by target, or if the error has a method As(interface{}) bool such that
    70  // As(target) returns true. In the latter case, the As method is responsible for
    71  // setting target.
    72  //
    73  // An error type might provide an As method so it can be treated as if it were a
    74  // different error type.
    75  //
    76  // As panics if target is not a non-nil pointer to either a type that implements
    77  // error, or to any interface type.
    78  func As(err error, target any) bool {
    79  	if target == nil {
    80  		panic("errors: target cannot be nil")
    81  	}
    82  	val := reflectlite.ValueOf(target)
    83  	typ := val.Type()
    84  	if typ.Kind() != reflectlite.Ptr || val.IsNil() {
    85  		panic("errors: target must be a non-nil pointer")
    86  	}
    87  	targetType := typ.Elem()
    88  	if targetType.Kind() != reflectlite.Interface && !targetType.Implements(errorType) {
    89  		panic("errors: *target must be interface or implement error")
    90  	}
    91  	for err != nil {
    92  		if reflectlite.TypeOf(err).AssignableTo(targetType) {
    93  			val.Elem().Set(reflectlite.ValueOf(err))
    94  			return true
    95  		}
    96  		if x, ok := err.(interface{ As(any) bool }); ok && x.As(target) {
    97  			return true
    98  		}
    99  		err = Unwrap(err)
   100  	}
   101  	return false
   102  }
   103  
   104  var errorType = reflectlite.TypeOf((*error)(nil)).Elem()
   105  

View as plain text