2015-05-15 01:21:39 +00:00
|
|
|
package errcode
|
|
|
|
|
|
|
|
import (
|
2015-06-03 13:52:39 +00:00
|
|
|
"encoding/json"
|
2015-05-15 01:21:39 +00:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// ErrorCoder is the base interface for ErrorCode and Error allowing
|
|
|
|
// users of each to just call ErrorCode to get the real ID of each
|
|
|
|
type ErrorCoder interface {
|
|
|
|
ErrorCode() ErrorCode
|
|
|
|
}
|
|
|
|
|
2015-05-15 01:21:39 +00:00
|
|
|
// ErrorCode represents the error type. The errors are serialized via strings
|
|
|
|
// and the integer format may change and should *never* be exported.
|
|
|
|
type ErrorCode int
|
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// ErrorCode just returns itself
|
|
|
|
func (ec ErrorCode) ErrorCode() ErrorCode {
|
|
|
|
return ec
|
2015-05-15 01:21:39 +00:00
|
|
|
}
|
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// Error returns the ID/Value
|
|
|
|
func (ec ErrorCode) Error() string {
|
|
|
|
return ec.Descriptor().Value
|
2015-05-15 01:21:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Descriptor returns the descriptor for the error code.
|
|
|
|
func (ec ErrorCode) Descriptor() ErrorDescriptor {
|
|
|
|
d, ok := errorCodeToDescriptors[ec]
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
return ErrorCodeUnknown.Descriptor()
|
|
|
|
}
|
|
|
|
|
|
|
|
return d
|
|
|
|
}
|
|
|
|
|
|
|
|
// String returns the canonical identifier for this error code.
|
|
|
|
func (ec ErrorCode) String() string {
|
|
|
|
return ec.Descriptor().Value
|
|
|
|
}
|
|
|
|
|
|
|
|
// Message returned the human-readable error message for this error code.
|
|
|
|
func (ec ErrorCode) Message() string {
|
|
|
|
return ec.Descriptor().Message
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalText encodes the receiver into UTF-8-encoded text and returns the
|
|
|
|
// result.
|
|
|
|
func (ec ErrorCode) MarshalText() (text []byte, err error) {
|
|
|
|
return []byte(ec.String()), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalText decodes the form generated by MarshalText.
|
|
|
|
func (ec *ErrorCode) UnmarshalText(text []byte) error {
|
|
|
|
desc, ok := idToDescriptors[string(text)]
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
desc = ErrorCodeUnknown.Descriptor()
|
|
|
|
}
|
|
|
|
|
|
|
|
*ec = desc.Code
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// WithDetail creates a new Error struct based on the passed-in info and
|
|
|
|
// set the Detail property appropriately
|
|
|
|
func (ec ErrorCode) WithDetail(detail interface{}) Error {
|
|
|
|
if err, ok := detail.(error); ok {
|
|
|
|
detail = err.Error()
|
|
|
|
}
|
|
|
|
|
|
|
|
return Error{
|
|
|
|
Code: ec,
|
|
|
|
Detail: detail,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-15 01:21:39 +00:00
|
|
|
// Error provides a wrapper around ErrorCode with extra Details provided.
|
|
|
|
type Error struct {
|
2015-05-27 00:18:32 +00:00
|
|
|
Code ErrorCode `json:"code"`
|
|
|
|
Detail interface{} `json:"detail,omitempty"`
|
2015-05-15 01:21:39 +00:00
|
|
|
}
|
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// ErrorCode returns the ID/Value of this Error
|
|
|
|
func (e Error) ErrorCode() ErrorCode {
|
|
|
|
return e.Code
|
|
|
|
}
|
|
|
|
|
2015-05-15 01:21:39 +00:00
|
|
|
// Error returns a human readable representation of the error.
|
|
|
|
func (e Error) Error() string {
|
|
|
|
return fmt.Sprintf("%s: %s",
|
|
|
|
strings.ToLower(strings.Replace(e.Code.String(), "_", " ", -1)),
|
2015-05-27 00:18:32 +00:00
|
|
|
e.Code.Message())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Message returned the human-readable error message for this Error
|
|
|
|
func (e Error) Message() string {
|
|
|
|
return e.Code.Message()
|
2015-05-15 01:21:39 +00:00
|
|
|
}
|
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// ErrorDescriptor provides relevant information about a given error code.
|
|
|
|
type ErrorDescriptor struct {
|
|
|
|
// Code is the error code that this descriptor describes.
|
|
|
|
Code ErrorCode
|
2015-05-15 01:21:39 +00:00
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// Value provides a unique, string key, often captilized with
|
|
|
|
// underscores, to identify the error code. This value is used as the
|
|
|
|
// keyed value when serializing api errors.
|
|
|
|
Value string
|
2015-05-15 01:21:39 +00:00
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// Message is a short, human readable decription of the error condition
|
|
|
|
// included in API responses.
|
|
|
|
Message string
|
2015-05-15 01:21:39 +00:00
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// Description provides a complete account of the errors purpose, suitable
|
|
|
|
// for use in documentation.
|
|
|
|
Description string
|
2015-05-15 01:21:39 +00:00
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// HTTPStatusCode provides the http status code that is associated with
|
|
|
|
// this error condition.
|
|
|
|
HTTPStatusCode int
|
|
|
|
}
|
|
|
|
|
|
|
|
// ParseErrorCode returns the value by the string error code.
|
|
|
|
// `ErrorCodeUnknown` will be returned if the error is not known.
|
|
|
|
func ParseErrorCode(value string) ErrorCode {
|
|
|
|
ed, ok := idToDescriptors[value]
|
|
|
|
if ok {
|
|
|
|
return ed.Code
|
2015-05-15 01:21:39 +00:00
|
|
|
}
|
2015-06-03 13:52:39 +00:00
|
|
|
|
|
|
|
return ErrorCodeUnknown
|
2015-05-15 01:21:39 +00:00
|
|
|
}
|
|
|
|
|
2015-06-03 13:52:39 +00:00
|
|
|
// Errors provides the envelope for multiple errors and a few sugar methods
|
|
|
|
// for use within the application.
|
|
|
|
type Errors []error
|
|
|
|
|
2015-05-27 00:18:32 +00:00
|
|
|
func (errs Errors) Error() string {
|
|
|
|
switch len(errs) {
|
2015-05-15 01:21:39 +00:00
|
|
|
case 0:
|
|
|
|
return "<nil>"
|
|
|
|
case 1:
|
2015-05-27 00:18:32 +00:00
|
|
|
return errs[0].Error()
|
2015-05-15 01:21:39 +00:00
|
|
|
default:
|
|
|
|
msg := "errors:\n"
|
2015-05-27 00:18:32 +00:00
|
|
|
for _, err := range errs {
|
2015-05-15 01:21:39 +00:00
|
|
|
msg += err.Error() + "\n"
|
|
|
|
}
|
|
|
|
return msg
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Len returns the current number of errors.
|
2015-05-27 00:18:32 +00:00
|
|
|
func (errs Errors) Len() int {
|
|
|
|
return len(errs)
|
2015-05-15 01:21:39 +00:00
|
|
|
}
|
2015-06-03 13:52:39 +00:00
|
|
|
|
|
|
|
// jsonError extends Error with 'Message' so that we can include the
|
|
|
|
// error text, just in case the receiver of the JSON doesn't have this
|
|
|
|
// particular ErrorCode registered
|
|
|
|
type jsonError struct {
|
|
|
|
Code ErrorCode `json:"code"`
|
|
|
|
Message string `json:"message"`
|
|
|
|
Detail interface{} `json:"detail,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON converts slice of error, ErrorCode or Error into a
|
|
|
|
// slice of Error - then serializes
|
|
|
|
func (errs Errors) MarshalJSON() ([]byte, error) {
|
|
|
|
var tmpErrs []jsonError
|
|
|
|
|
|
|
|
for _, daErr := range errs {
|
|
|
|
var err Error
|
|
|
|
|
|
|
|
switch daErr.(type) {
|
|
|
|
case ErrorCode:
|
|
|
|
err = daErr.(ErrorCode).WithDetail(nil)
|
|
|
|
case Error:
|
|
|
|
err = daErr.(Error)
|
|
|
|
default:
|
|
|
|
err = ErrorCodeUnknown.WithDetail(daErr)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
tmpErrs = append(tmpErrs, jsonError{
|
|
|
|
Code: err.Code,
|
|
|
|
Message: err.Message(),
|
|
|
|
Detail: err.Detail,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.Marshal(tmpErrs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON deserializes []Error and then converts it into slice of
|
|
|
|
// Error or ErrorCode
|
|
|
|
func (errs *Errors) UnmarshalJSON(data []byte) error {
|
|
|
|
var tmpErrs []jsonError
|
|
|
|
|
|
|
|
if err := json.Unmarshal(data, &tmpErrs); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var newErrs Errors
|
|
|
|
for _, daErr := range tmpErrs {
|
|
|
|
if daErr.Detail == nil {
|
|
|
|
// Error's w/o details get converted to ErrorCode
|
|
|
|
newErrs = append(newErrs, daErr.Code)
|
|
|
|
} else {
|
|
|
|
// Error's w/ details are untouched
|
|
|
|
newErrs = append(newErrs, Error{
|
|
|
|
Code: daErr.Code,
|
|
|
|
Detail: daErr.Detail,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*errs = newErrs
|
|
|
|
return nil
|
|
|
|
}
|