package util import "fmt" // Err is the implementation of error that all goadb functions return. type Err struct { // Code is the high-level "type" of error. Code ErrCode // Message is a human-readable description of the error. Message string // Details is optional, and can be used to associate any auxiliary data with an error. Details interface{} // Cause is optional, and points to the more specific error that caused this one. Cause error } var _ error = &Err{} //go:generate stringer -type=ErrCode type ErrCode byte const ( AssertionError ErrCode = iota ParseError ErrCode = iota // The server was not available on the request port and could not be started. ServerNotAvailable ErrCode = iota // General network error communicating with the server. NetworkError ErrCode = iota // The server returned an error message, but we couldn't parse it. AdbError ErrCode = iota // The server returned a "device not found" error. DeviceNotFound ErrCode = iota // Tried to perform an operation on a path that doesn't exist on the device. FileNoExistError ErrCode = iota ) func Errorf(code ErrCode, format string, args ...interface{}) error { return &Err{ Code: code, Message: fmt.Sprintf(format, args...), } } /* WrapErrf returns an *Err that wraps another *Err and has the same ErrCode. Panics if cause is not an *Err. To wrap generic errors, use WrapErrorf. */ func WrapErrf(cause error, format string, args ...interface{}) error { if cause == nil { return nil } err := cause.(*Err) return &Err{ Code: err.Code, Message: fmt.Sprintf(format, args...), Cause: err, } } /* WrapErrorf returns an *Err that wraps another arbitrary error with an ErrCode and a message. If cause is nil, returns nil, so you can use it like return util.WrapErrorf(DoSomethingDangerous(), util.NetworkError, "well that didn't work") If cause is known to be of type *Err, use WrapErrf. */ func WrapErrorf(cause error, code ErrCode, format string, args ...interface{}) error { if cause == nil { return nil } return &Err{ Code: code, Message: fmt.Sprintf(format, args...), Cause: cause, } } func AssertionErrorf(format string, args ...interface{}) error { return &Err{ Code: AssertionError, Message: fmt.Sprintf(format, args...), } } func (err *Err) Error() string { msg := fmt.Sprintf("%s: %s", err.Code, err.Message) if err.Details != nil { msg = fmt.Sprintf("%s (%+v)", msg, err.Details) } return msg } // HasErrCode returns true if err is an *Err and err.Code == code. func HasErrCode(err error, code ErrCode) bool { switch err := err.(type) { case *Err: return err.Code == code default: return false } }