mirror of
https://github.com/kaspanet/kaspad.git
synced 2025-06-07 22:56:41 +00:00

* [NOD-375] Move to pkg/errors * [NOD-375] Fix tests * [NOD-375] Make AreErrorsEqual a shared function
113 lines
3.5 KiB
Go
113 lines
3.5 KiB
Go
package httpserverutils
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"github.com/jinzhu/gorm"
|
|
"github.com/pkg/errors"
|
|
"net/http"
|
|
"strings"
|
|
)
|
|
|
|
// HandlerError is an error returned from
|
|
// a rest route handler or a middleware.
|
|
type HandlerError struct {
|
|
Code int
|
|
Message string
|
|
ClientMessage string
|
|
}
|
|
|
|
func (hErr *HandlerError) Error() string {
|
|
return hErr.Message
|
|
}
|
|
|
|
// NewHandlerError returns a HandlerError with the given code and message.
|
|
func NewHandlerError(code int, message string) *HandlerError {
|
|
return &HandlerError{
|
|
Code: code,
|
|
Message: message,
|
|
ClientMessage: message,
|
|
}
|
|
}
|
|
|
|
// NewHandlerErrorWithCustomClientMessage returns a HandlerError with
|
|
// the given code, message and client error message.
|
|
func NewHandlerErrorWithCustomClientMessage(code int, message, clientMessage string) *HandlerError {
|
|
return &HandlerError{
|
|
Code: code,
|
|
Message: message,
|
|
ClientMessage: clientMessage,
|
|
}
|
|
}
|
|
|
|
// NewInternalServerHandlerError returns a HandlerError with
|
|
// the given message, and the http.StatusInternalServerError
|
|
// status text as client message.
|
|
func NewInternalServerHandlerError(message string) *HandlerError {
|
|
return NewHandlerErrorWithCustomClientMessage(http.StatusInternalServerError, message, http.StatusText(http.StatusInternalServerError))
|
|
}
|
|
|
|
// NewErrorFromDBErrors takes a slice of database errors and a prefix, and
|
|
// returns an error with all of the database errors formatted to one string with
|
|
// the given prefix
|
|
func NewErrorFromDBErrors(prefix string, dbErrors []error) error {
|
|
dbErrorsStrings := make([]string, len(dbErrors))
|
|
for i, dbErr := range dbErrors {
|
|
dbErrorsStrings[i] = fmt.Sprintf("\"%s\"", dbErr)
|
|
}
|
|
return errors.Errorf("%s [%s]", prefix, strings.Join(dbErrorsStrings, ","))
|
|
}
|
|
|
|
// NewHandlerErrorFromDBErrors takes a slice of database errors and a prefix, and
|
|
// returns an HandlerError with error code http.StatusInternalServerError with
|
|
// all of the database errors formatted to one string with the given prefix
|
|
func NewHandlerErrorFromDBErrors(prefix string, dbErrors []error) *HandlerError {
|
|
return NewInternalServerHandlerError(NewErrorFromDBErrors(prefix, dbErrors).Error())
|
|
}
|
|
|
|
// IsDBRecordNotFoundError returns true if the given dbErrors contains only a RecordNotFound error
|
|
func IsDBRecordNotFoundError(dbErrors []error) bool {
|
|
return len(dbErrors) == 1 && gorm.IsRecordNotFoundError(dbErrors[0])
|
|
}
|
|
|
|
// HasDBError returns true if the given dbErrors contain any errors that aren't RecordNotFound
|
|
func HasDBError(dbErrors []error) bool {
|
|
return !IsDBRecordNotFoundError(dbErrors) && len(dbErrors) > 0
|
|
}
|
|
|
|
// ClientError is the http response that is sent to the
|
|
// client in case of an error.
|
|
type ClientError struct {
|
|
ErrorCode int `json:"errorCode"`
|
|
ErrorMessage string `json:"errorMessage"`
|
|
}
|
|
|
|
func (err *ClientError) Error() string {
|
|
return fmt.Sprintf("%s (Code: %d)", err.ErrorMessage, err.ErrorCode)
|
|
}
|
|
|
|
// SendErr takes a HandlerError and create a ClientError out of it that is sent
|
|
// to the http client.
|
|
func SendErr(ctx *ServerContext, w http.ResponseWriter, hErr *HandlerError) {
|
|
errMsg := fmt.Sprintf("got error: %s", hErr)
|
|
ctx.Warnf(errMsg)
|
|
w.WriteHeader(hErr.Code)
|
|
SendJSONResponse(w, &ClientError{
|
|
ErrorCode: hErr.Code,
|
|
ErrorMessage: hErr.ClientMessage,
|
|
})
|
|
}
|
|
|
|
// SendJSONResponse encodes the given response to JSON format and
|
|
// sends it to the client
|
|
func SendJSONResponse(w http.ResponseWriter, response interface{}) {
|
|
b, err := json.Marshal(response)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
_, err = fmt.Fprintf(w, string(b))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|