-
Notifications
You must be signed in to change notification settings - Fork 188
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[Access] Add draft of access api error framework #7106
base: master
Are you sure you want to change the base?
Changes from 9 commits
e176fe3
049bc5f
e4c2ea3
e4a33d9
4a7ba95
f99e889
e269179
6b039eb
df07206
56845e7
cb319c2
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,237 @@ | ||||||||||||||||||
package access | ||||||||||||||||||
|
||||||||||||||||||
import ( | ||||||||||||||||||
"context" | ||||||||||||||||||
"errors" | ||||||||||||||||||
"fmt" | ||||||||||||||||||
|
||||||||||||||||||
"github.com/onflow/flow-go/module/irrecoverable" | ||||||||||||||||||
) | ||||||||||||||||||
|
||||||||||||||||||
// RequireNoError returns nil if error is nil, otherwise throws an irrecoverable exception | ||||||||||||||||||
func RequireNoError(ctx context.Context, err error) error { | ||||||||||||||||||
if err == nil { | ||||||||||||||||||
return nil | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
irrecoverable.Throw(ctx, err) | ||||||||||||||||||
return irrecoverable.NewException(err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// RequireErrorIs returns the error if it unwraps to any of the provided target error types | ||||||||||||||||||
// Otherwise, it throws an irrecoverable exception | ||||||||||||||||||
func RequireErrorIs(ctx context.Context, err error, targetErrs ...error) error { | ||||||||||||||||||
if err == nil { | ||||||||||||||||||
return nil | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
for _, targetErr := range targetErrs { | ||||||||||||||||||
if errors.Is(err, targetErr) { | ||||||||||||||||||
return err | ||||||||||||||||||
} | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
irrecoverable.Throw(ctx, err) | ||||||||||||||||||
return irrecoverable.NewException(err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// RequireAccessError returns the error if it is an Access sentinel error | ||||||||||||||||||
// Otherwise, it throws an irrecoverable exception | ||||||||||||||||||
// Note: this method will not unwrap the error. the passed error must be an instance of a sentinel | ||||||||||||||||||
// error. | ||||||||||||||||||
func RequireAccessError(ctx context.Context, err error) error { | ||||||||||||||||||
if err == nil { | ||||||||||||||||||
return nil | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
if _, ok := err.(accessSentinel); ok { | ||||||||||||||||||
return err | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
irrecoverable.Throw(ctx, err) | ||||||||||||||||||
return irrecoverable.NewException(err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// accessSentinel is a marker interface for errors returned by the Access API | ||||||||||||||||||
// This is used to differentiate unexpected errors returned by endpoints | ||||||||||||||||||
// Implement this for all new Access sentinel errors. Any that do not will be considered unexpected | ||||||||||||||||||
// exceptions. | ||||||||||||||||||
type accessSentinel interface { | ||||||||||||||||||
accessSentinel() | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// InvalidRequestError indicates that the client's request was malformed or invalid | ||||||||||||||||||
type InvalidRequestError struct { | ||||||||||||||||||
err error | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func NewInvalidRequestError(err error) InvalidRequestError { | ||||||||||||||||||
return InvalidRequestError{err: err} | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e InvalidRequestError) Error() string { | ||||||||||||||||||
return fmt.Sprintf("invalid argument: %v", e.err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e InvalidRequestError) Unwrap() error { | ||||||||||||||||||
return e.err | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e InvalidRequestError) accessSentinel() {} | ||||||||||||||||||
|
||||||||||||||||||
func IsInvalidRequestError(err error) bool { | ||||||||||||||||||
var errInvalidRequest InvalidRequestError | ||||||||||||||||||
return errors.As(err, &errInvalidRequest) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// DataNotFoundError indicates that the requested data was not found on the system | ||||||||||||||||||
type DataNotFoundError struct { | ||||||||||||||||||
dataType string | ||||||||||||||||||
err error | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func NewDataNotFoundError(dataType string, err error) DataNotFoundError { | ||||||||||||||||||
return DataNotFoundError{dataType: dataType, err: err} | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e DataNotFoundError) Error() string { | ||||||||||||||||||
return fmt.Sprintf("data not found for %s: %v", e.dataType, e.err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e DataNotFoundError) Unwrap() error { | ||||||||||||||||||
return e.err | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e DataNotFoundError) accessSentinel() {} | ||||||||||||||||||
|
||||||||||||||||||
func IsDataNotFoundError(err error) bool { | ||||||||||||||||||
var errDataNotFound DataNotFoundError | ||||||||||||||||||
return errors.As(err, &errDataNotFound) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// InternalError indicates that a non-fatal internal error occurred | ||||||||||||||||||
// IMPORTANT: this should only be used for benign internal errors. Fatal or irrecoverable system | ||||||||||||||||||
// errors must be handled explicitly. | ||||||||||||||||||
type InternalError struct { | ||||||||||||||||||
err error | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func NewInternalError(err error) InternalError { | ||||||||||||||||||
return InternalError{err: err} | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e InternalError) Error() string { | ||||||||||||||||||
return fmt.Sprintf("internal error: %v", e.err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e InternalError) Unwrap() error { | ||||||||||||||||||
return e.err | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e InternalError) accessSentinel() {} | ||||||||||||||||||
|
||||||||||||||||||
func IsInternalError(err error) bool { | ||||||||||||||||||
var errInternalError InternalError | ||||||||||||||||||
return errors.As(err, &errInternalError) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// OutOfRangeError indicates that the request was for data that is outside of the available range. | ||||||||||||||||||
// This is a more specific version of DataNotFoundError, where the data is known to eventually exist, but | ||||||||||||||||||
// currently is not known. | ||||||||||||||||||
// For example, querying data for a height above the current finalized height. | ||||||||||||||||||
type OutOfRangeError struct { | ||||||||||||||||||
err error | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func NewOutOfRangeError(err error) OutOfRangeError { | ||||||||||||||||||
return OutOfRangeError{err: err} | ||||||||||||||||||
} | ||||||||||||||||||
Comment on lines
+146
to
+148
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. [optional] since you are mentioning it: Lines 139 to 140 in df07206
we could also implement it accordingly
Suggested change
similarly for the |
||||||||||||||||||
|
||||||||||||||||||
func (e OutOfRangeError) Error() string { | ||||||||||||||||||
return fmt.Sprintf("out of range: %v", e.err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e OutOfRangeError) Unwrap() error { | ||||||||||||||||||
return e.err | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e OutOfRangeError) accessSentinel() {} | ||||||||||||||||||
|
||||||||||||||||||
func IsOutOfRangeError(err error) bool { | ||||||||||||||||||
var errOutOfRangeError OutOfRangeError | ||||||||||||||||||
return errors.As(err, &errOutOfRangeError) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// PreconditionFailedError indicates that a precondition for the operation was not met | ||||||||||||||||||
// This is a more specific version of InvalidRequestError, where the request is valid, but the system | ||||||||||||||||||
// is not currently in a state to fulfill the request (but may be in the future). | ||||||||||||||||||
type PreconditionFailedError struct { | ||||||||||||||||||
err error | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func NewPreconditionFailedError(err error) PreconditionFailedError { | ||||||||||||||||||
return PreconditionFailedError{err: err} | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e PreconditionFailedError) Error() string { | ||||||||||||||||||
return fmt.Sprintf("precondition failed: %v", e.err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e PreconditionFailedError) Unwrap() error { | ||||||||||||||||||
return e.err | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e PreconditionFailedError) accessSentinel() {} | ||||||||||||||||||
|
||||||||||||||||||
func IsPreconditionFailedError(err error) bool { | ||||||||||||||||||
var errPreconditionFailed PreconditionFailedError | ||||||||||||||||||
return errors.As(err, &errPreconditionFailed) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// RequestCanceledError indicates that the request was canceled before the server finished processing it | ||||||||||||||||||
type RequestCanceledError struct { | ||||||||||||||||||
err error | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func NewRequestCanceledError(err error) RequestCanceledError { | ||||||||||||||||||
return RequestCanceledError{err: err} | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e RequestCanceledError) Error() string { | ||||||||||||||||||
return fmt.Sprintf("request canceled: %v", e.err) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e RequestCanceledError) Unwrap() error { | ||||||||||||||||||
return e.err | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e RequestCanceledError) accessSentinel() {} | ||||||||||||||||||
|
||||||||||||||||||
func IsRequestCanceledError(err error) bool { | ||||||||||||||||||
var requestCanceledError RequestCanceledError | ||||||||||||||||||
return errors.As(err, &requestCanceledError) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
// RequestTimedOutError indicates that the request timed out before the server finished processing it | ||||||||||||||||||
type RequestTimedOutError struct { | ||||||||||||||||||
err error | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func NewRequestTimedOutError(err error) RequestTimedOutError { | ||||||||||||||||||
return RequestTimedOutError{err: err} | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e RequestTimedOutError) Error() string { | ||||||||||||||||||
return fmt.Sprintf("request timedout: %v", e.err) | ||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Typo in 'timedout': should be 'timed out'.
Suggested change
Copilot is powered by AI, so mistakes are possible. Review output carefully before use. Positive FeedbackNegative Feedback |
||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e RequestTimedOutError) Unwrap() error { | ||||||||||||||||||
return e.err | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (e RequestTimedOutError) accessSentinel() {} | ||||||||||||||||||
|
||||||||||||||||||
func IsRequestTimedOutError(err error) bool { | ||||||||||||||||||
var requestTimedOutError RequestTimedOutError | ||||||||||||||||||
return errors.As(err, &requestTimedOutError) | ||||||||||||||||||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
package access_test | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
"testing" | ||
|
||
"github.com/stretchr/testify/require" | ||
|
||
"github.com/onflow/flow-go/access" | ||
"github.com/onflow/flow-go/module/irrecoverable" | ||
) | ||
|
||
func TestRequireNoError(t *testing.T) { | ||
t.Parallel() | ||
|
||
t.Run("no error", func(t *testing.T) { | ||
t.Parallel() | ||
|
||
signalerCtx := irrecoverable.NewMockSignalerContext(t, context.Background()) | ||
ctx := irrecoverable.WithSignalerContext(context.Background(), signalerCtx) | ||
|
||
err := access.RequireNoError(ctx, nil) | ||
require.NoError(t, err) | ||
}) | ||
|
||
t.Run("with error", func(t *testing.T) { | ||
t.Parallel() | ||
|
||
expectedErr := fmt.Errorf("expected error") | ||
|
||
signalerCtx := irrecoverable.NewMockSignalerContextExpectError(t, context.Background(), expectedErr) | ||
ctx := irrecoverable.WithSignalerContext(context.Background(), signalerCtx) | ||
|
||
err := access.RequireNoError(ctx, expectedErr) | ||
require.NotErrorIs(t, err, expectedErr, "expected error should be overridden and explicitly not wrapped") | ||
require.Containsf(t, err.Error(), expectedErr.Error(), "expected returned error message to contain original error message") | ||
}) | ||
} | ||
|
||
func TestRequireErrorIs(t *testing.T) { | ||
t.Parallel() | ||
|
||
targetErr := fmt.Errorf("target error") | ||
|
||
t.Run("no error", func(t *testing.T) { | ||
t.Parallel() | ||
|
||
signalerCtx := irrecoverable.NewMockSignalerContext(t, context.Background()) | ||
ctx := irrecoverable.WithSignalerContext(context.Background(), signalerCtx) | ||
|
||
err := access.RequireErrorIs(ctx, nil, targetErr) | ||
require.NoError(t, err) | ||
}) | ||
|
||
t.Run("with expected error", func(t *testing.T) { | ||
t.Parallel() | ||
|
||
expectedErr := fmt.Errorf("got err: %w", targetErr) | ||
|
||
signalerCtx := irrecoverable.NewMockSignalerContext(t, context.Background()) | ||
ctx := irrecoverable.WithSignalerContext(context.Background(), signalerCtx) | ||
|
||
err := access.RequireErrorIs(ctx, expectedErr, targetErr) | ||
require.ErrorIs(t, err, expectedErr) | ||
}) | ||
|
||
t.Run("with multiple expected error", func(t *testing.T) { | ||
t.Parallel() | ||
|
||
expectedErr := fmt.Errorf("got err: %w", targetErr) | ||
|
||
signalerCtx := irrecoverable.NewMockSignalerContext(t, context.Background()) | ||
ctx := irrecoverable.WithSignalerContext(context.Background(), signalerCtx) | ||
|
||
err := access.RequireErrorIs(ctx, expectedErr, fmt.Errorf("target error2"), targetErr) | ||
require.ErrorIs(t, err, expectedErr) | ||
}) | ||
|
||
t.Run("with unexpected error", func(t *testing.T) { | ||
t.Parallel() | ||
|
||
expectedErr := fmt.Errorf("expected error") | ||
|
||
signalerCtx := irrecoverable.NewMockSignalerContextExpectError(t, context.Background(), expectedErr) | ||
ctx := irrecoverable.WithSignalerContext(context.Background(), signalerCtx) | ||
|
||
err := access.RequireErrorIs(ctx, expectedErr, targetErr) | ||
require.NotErrorIs(t, err, expectedErr, "expected error should be overridden and explicitly not wrapped") | ||
require.Containsf(t, err.Error(), expectedErr.Error(), "expected returned error message to contain original error message") | ||
}) | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,11 @@ | ||
package common | ||
|
||
import "net/http" | ||
import ( | ||
"errors" | ||
"net/http" | ||
|
||
"github.com/onflow/flow-go/access" | ||
) | ||
|
||
// StatusError provides custom error with http status. | ||
type StatusError interface { | ||
|
@@ -56,3 +61,36 @@ func (e *Error) Status() int { | |
func (e *Error) Error() string { | ||
return e.err.Error() | ||
} | ||
|
||
// ErrorToStatusError converts an Access API error into a StatusError. | ||
// The input may either be a StatusError already, or an access sentinel error. | ||
// All generic errors are classified as `500 Internal Server Error` | ||
func ErrorToStatusError(err error) StatusError { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. each API has unique error reporting protocols (grpc vs rest vs websockets). Create API specific converters that return the appropriate mapping from access sentinels. |
||
if err == nil { | ||
return nil | ||
} | ||
|
||
var converted StatusError | ||
if errors.As(err, &converted) { | ||
return converted | ||
} | ||
|
||
switch { | ||
case access.IsInvalidRequestError(err): | ||
return NewBadRequestError(err) | ||
case access.IsDataNotFoundError(err): | ||
return NewNotFoundError(err.Error(), err) | ||
case access.IsPreconditionFailedError(err): | ||
return NewRestError(http.StatusPreconditionFailed, err.Error(), err) | ||
case access.IsOutOfRangeError(err): | ||
return NewNotFoundError(err.Error(), err) | ||
case access.IsInternalError(err): | ||
return NewRestError(http.StatusInternalServerError, err.Error(), err) | ||
case access.IsRequestCanceledError(err): | ||
return NewRestError(http.StatusRequestTimeout, "Request canceled", err) | ||
case access.IsRequestTimedOutError(err): | ||
return NewRestError(http.StatusRequestTimeout, "Request deadline exceeded", err) | ||
default: | ||
return NewRestError(http.StatusInternalServerError, err.Error(), err) | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This can be used for more complex endpoints that call into other methods to ensure all unexpected errors are handled.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
you could add your comment as context to the goDoc.