velero/pkg/util/logging/error_location_hook_test.go

177 lines
5.1 KiB
Go

/*
Copyright 2017, 2019 the Velero contributors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package logging
import (
"errors"
"testing"
pkgerrs "github.com/pkg/errors"
"github.com/sirupsen/logrus"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
func TestFire(t *testing.T) {
tests := []struct {
name string
preEntryFields map[string]interface{}
expectedEntryFields map[string]interface{}
expectedErr bool
}{
{
name: "no error",
preEntryFields: map[string]interface{}{"foo": "bar"},
expectedEntryFields: map[string]interface{}{"foo": "bar"},
},
{
name: "basic (non-pkg/errors) error",
preEntryFields: map[string]interface{}{logrus.ErrorKey: errors.New("a normal error")},
expectedEntryFields: map[string]interface{}{logrus.ErrorKey: errors.New("a normal error")},
},
{
name: "non-error logged in error field",
preEntryFields: map[string]interface{}{logrus.ErrorKey: "not an error"},
expectedEntryFields: map[string]interface{}{logrus.ErrorKey: "not an error"},
expectedErr: false,
},
{
name: "pkg/errors error",
preEntryFields: map[string]interface{}{logrus.ErrorKey: pkgerrs.New("a pkg/errors error")},
expectedEntryFields: map[string]interface{}{
logrus.ErrorKey: pkgerrs.New("a pkg/errors error"),
errorFileField: "",
errorFunctionField: "github.com/vmware-tanzu/velero/pkg/util/logging.TestFire",
},
},
{
name: "already have error file and function fields",
preEntryFields: map[string]interface{}{
logrus.ErrorKey: pkgerrs.New("a pkg/errors error"),
errorFileField: "some_file.go:123",
errorFunctionField: "SomeFunction",
},
expectedEntryFields: map[string]interface{}{
logrus.ErrorKey: pkgerrs.New("a pkg/errors error"),
errorFileField: "some_file.go:123",
errorFunctionField: "SomeFunction",
},
},
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
hook := &ErrorLocationHook{}
entry := &logrus.Entry{
Data: logrus.Fields(test.preEntryFields),
}
// method under test
err := hook.Fire(entry)
require.Equal(t, test.expectedErr, err != nil)
require.Equal(t, len(test.expectedEntryFields), len(entry.Data))
for key, expectedValue := range test.expectedEntryFields {
actualValue, found := entry.Data[key]
assert.True(t, found, "expected key not found: %s", key)
switch key {
// test existence of this field only since testing the value
// is fragile
case errorFileField:
case logrus.ErrorKey:
if err, ok := expectedValue.(error); ok {
assert.Equal(t, err.Error(), actualValue.(error).Error())
} else {
assert.Equal(t, expectedValue, actualValue)
}
default:
assert.Equal(t, expectedValue, actualValue)
}
}
})
}
}
func TestGetInnermostTrace(t *testing.T) {
newError := func() error {
return errors.New("a normal error")
}
tests := []struct {
name string
err error
expectedRes error
}{
{
name: "normal error",
err: newError(),
expectedRes: nil,
},
{
name: "pkg/errs error",
err: pkgerrs.New("a pkg/errs error"),
expectedRes: pkgerrs.New("a pkg/errs error"),
},
{
name: "one level of stack-ing a normal error",
err: pkgerrs.WithStack(newError()),
expectedRes: pkgerrs.WithStack(newError()),
},
{
name: "two levels of stack-ing a normal error",
err: pkgerrs.WithStack(pkgerrs.WithStack(newError())),
expectedRes: pkgerrs.WithStack(newError()),
},
{
name: "one level of stack-ing a pkg/errors error",
err: pkgerrs.WithStack(pkgerrs.New("a pkg/errs error")),
expectedRes: pkgerrs.New("a pkg/errs error"),
},
{
name: "two levels of stack-ing a pkg/errors error",
err: pkgerrs.WithStack(pkgerrs.WithStack(pkgerrs.New("a pkg/errs error"))),
expectedRes: pkgerrs.New("a pkg/errs error"),
},
{
name: "two levels of wrapping a normal error",
err: pkgerrs.Wrap(pkgerrs.Wrap(newError(), "wrap 1"), "wrap 2"),
expectedRes: pkgerrs.Wrap(newError(), "wrap 1"),
},
{
name: "two levels of wrapping a pkg/errors error",
err: pkgerrs.Wrap(pkgerrs.Wrap(pkgerrs.New("a pkg/errs error"), "wrap 1"), "wrap 2"),
expectedRes: pkgerrs.New("a pkg/errs error"),
},
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
res := getInnermostTrace(test.err)
if test.expectedRes == nil {
assert.Nil(t, res)
return
}
assert.Equal(t, test.expectedRes.Error(), res.Error())
})
}
}