Skip to content

Commit

Permalink
Merge PR cosmos#4849: Result from error
Browse files Browse the repository at this point in the history
  • Loading branch information
ethanfrey authored and alexanderbez committed Aug 5, 2019
1 parent cf19802 commit a73b3f5
Show file tree
Hide file tree
Showing 2 changed files with 70 additions and 4 deletions.
32 changes: 28 additions & 4 deletions types/errors.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,11 @@ import (
"strings"

"github.com/pkg/errors"
cmn "github.com/tendermint/tendermint/libs/common"

abci "github.com/tendermint/tendermint/abci/types"
cmn "github.com/tendermint/tendermint/libs/common"

sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
)

// CodeType - ABCI code identifier within codespace
Expand Down Expand Up @@ -248,10 +250,14 @@ func (err *sdkError) Code() CodeType {
// Implements ABCIError.
func (err *sdkError) ABCILog() string {
errMsg := err.cmnError.Error()
return encodeErrorLog(err.codespace, err.code, errMsg)
}

func encodeErrorLog(codespace CodespaceType, code CodeType, msg string) string {
jsonErr := humanReadableError{
Codespace: err.codespace,
Code: err.code,
Message: errMsg,
Codespace: codespace,
Code: code,
Message: msg,
}

var buff bytes.Buffer
Expand Down Expand Up @@ -282,6 +288,24 @@ func (err *sdkError) QueryResult() abci.ResponseQuery {
}
}

// ResultFromError will return err.Result() if it implements sdk.Error
// Otherwise, it will use the reflecton from types/error to determine
// the code, codespace, and log.
//
// This is intended to provide a bridge to allow both error types
// to live side-by-side.
func ResultFromError(err error) Result {
if sdk, ok := err.(Error); ok {
return sdk.Result()
}
space, code, log := sdkerrors.ABCIInfo(err, false)
return Result{
Codespace: CodespaceType(space),
Code: CodeType(code),
Log: encodeErrorLog(CodespaceType(space), CodeType(code), log),
}
}

//----------------------------------------
// REST error utilities

Expand Down
42 changes: 42 additions & 0 deletions types/errors_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@ import (
"testing"

"github.com/stretchr/testify/require"

sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
)

var codeTypes = []CodeType{
Expand Down Expand Up @@ -93,3 +95,43 @@ func TestAppendMsgToErr(t *testing.T) {
fmt.Sprintf("Should have formatted the error message of ABCI Log. tc #%d", i))
}
}

func TestResultFromError(t *testing.T) {
cases := map[string]struct {
err error
expect Result
}{
"sdk.Error": {
err: ErrUnauthorized("not owner"),
expect: Result{
Codespace: CodespaceRoot,
Code: CodeUnauthorized,
Log: `{"codespace":"sdk","code":4,"message":"not owner"}`,
},
},
"types/errors": {
err: sdkerrors.Wrap(sdkerrors.ErrUnauthorized, "not owner"),
expect: Result{
Codespace: CodespaceRoot,
Code: CodeUnauthorized,
Log: `{"codespace":"sdk","code":4,"message":"not owner: unauthorized"}`,
},
},
"stdlib errors": {
err: fmt.Errorf("not owner"),
expect: Result{
Codespace: CodespaceType("undefined"),
Code: CodeInternal,
// note that we redact the internal errors in the new package to not leak eg. panics
Log: `{"codespace":"undefined","code":1,"message":"internal error"}`,
},
},
}

for name, tc := range cases {
t.Run(name, func(t *testing.T) {
res := ResultFromError(tc.err)
require.Equal(t, tc.expect, res)
})
}
}

0 comments on commit a73b3f5

Please sign in to comment.