2023-08-03 22:24:23 +00:00
|
|
|
package shared
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2023-09-20 19:06:11 +00:00
|
|
|
"net/http"
|
2023-08-03 22:24:23 +00:00
|
|
|
"strings"
|
2023-09-20 19:06:11 +00:00
|
|
|
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/rpc/lookup"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/blocks"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
|
|
|
http2 "github.com/prysmaticlabs/prysm/v4/network/http"
|
2023-08-03 22:24:23 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// DecodeError represents an error resulting from trying to decode an HTTP request.
|
|
|
|
// It tracks the full field name for which decoding failed.
|
|
|
|
type DecodeError struct {
|
|
|
|
path []string
|
|
|
|
err error
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewDecodeError wraps an error (either the initial decoding error or another DecodeError).
|
|
|
|
// The current field that failed decoding must be passed in.
|
|
|
|
func NewDecodeError(err error, field string) *DecodeError {
|
|
|
|
de, ok := err.(*DecodeError)
|
|
|
|
if ok {
|
|
|
|
return &DecodeError{path: append([]string{field}, de.path...), err: de.err}
|
|
|
|
}
|
|
|
|
return &DecodeError{path: []string{field}, err: err}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Error returns the formatted error message which contains the full field name and the actual decoding error.
|
|
|
|
func (e *DecodeError) Error() string {
|
|
|
|
return fmt.Sprintf("could not decode %s: %s", strings.Join(e.path, "."), e.err.Error())
|
|
|
|
}
|
2023-08-18 14:29:40 +00:00
|
|
|
|
|
|
|
// IndexedVerificationFailureError wraps a collection of verification failures.
|
|
|
|
type IndexedVerificationFailureError struct {
|
|
|
|
Message string `json:"message"`
|
|
|
|
Code int `json:"code"`
|
|
|
|
Failures []*IndexedVerificationFailure `json:"failures"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *IndexedVerificationFailureError) StatusCode() int {
|
|
|
|
return e.Code
|
|
|
|
}
|
|
|
|
|
|
|
|
// IndexedVerificationFailure represents an issue when verifying a single indexed object e.g. an item in an array.
|
|
|
|
type IndexedVerificationFailure struct {
|
|
|
|
Index int `json:"index"`
|
|
|
|
Message string `json:"message"`
|
|
|
|
}
|
2023-09-20 19:06:11 +00:00
|
|
|
|
2023-09-21 18:16:59 +00:00
|
|
|
// WriteStateFetchError writes an appropriate error based on the supplied argument.
|
|
|
|
// The argument error should be a result of fetching state.
|
|
|
|
func WriteStateFetchError(w http.ResponseWriter, err error) {
|
2023-10-13 17:21:38 +00:00
|
|
|
if _, ok := err.(*lookup.StateNotFoundError); ok {
|
|
|
|
http2.HandleError(w, "State not found", http.StatusNotFound)
|
|
|
|
return
|
2023-09-21 18:16:59 +00:00
|
|
|
}
|
|
|
|
if parseErr, ok := err.(*lookup.StateIdParseError); ok {
|
|
|
|
http2.HandleError(w, "Invalid state ID: "+parseErr.Error(), http.StatusBadRequest)
|
2023-10-13 17:21:38 +00:00
|
|
|
return
|
2023-09-21 18:16:59 +00:00
|
|
|
}
|
|
|
|
http2.HandleError(w, "Could not get state: "+err.Error(), http.StatusInternalServerError)
|
|
|
|
}
|
|
|
|
|
2023-09-20 19:06:11 +00:00
|
|
|
// WriteBlockFetchError writes an appropriate error based on the supplied argument.
|
|
|
|
// The argument error should be a result of fetching block.
|
|
|
|
func WriteBlockFetchError(w http.ResponseWriter, blk interfaces.ReadOnlySignedBeaconBlock, err error) bool {
|
|
|
|
if invalidBlockIdErr, ok := err.(*lookup.BlockIdParseError); ok {
|
|
|
|
http2.HandleError(w, "Invalid block ID: "+invalidBlockIdErr.Error(), http.StatusBadRequest)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
http2.HandleError(w, "Could not get block from block ID: %s"+err.Error(), http.StatusInternalServerError)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if err = blocks.BeaconBlockIsNil(blk); err != nil {
|
|
|
|
http2.HandleError(w, "Could not find requested block: %s"+err.Error(), http.StatusNotFound)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|