Add log redacting, controlled by the new EnableLogRedacting config option (default true)

Imported redacting code from https://github.com/whuang8/redactrus (thanks William Huang)
Didn't use it as a dependency as it was too small and we want to keep dependencies at a minimum
This commit is contained in:
Deluan 2021-05-02 16:39:25 -04:00
parent 2372f1d12b
commit cfbc39fb7f
5 changed files with 238 additions and 3 deletions

141
log/redactrus_test.go Executable file
View file

@ -0,0 +1,141 @@
package log
import (
"testing"
"github.com/sirupsen/logrus"
"github.com/stretchr/testify/assert"
)
var h = &Hook{}
type levelsTest struct {
name string
acceptedLevels []logrus.Level
expected []logrus.Level
description string
}
func TestLevels(t *testing.T) {
tests := []levelsTest{
{
name: "undefinedAcceptedLevels",
acceptedLevels: []logrus.Level{},
expected: logrus.AllLevels,
description: "All logrus levels expected, but did not receive them",
},
{
name: "definedAcceptedLevels",
acceptedLevels: []logrus.Level{logrus.InfoLevel},
expected: []logrus.Level{logrus.InfoLevel},
description: "Logrus Info level expected, but did not receive that.",
},
}
for _, test := range tests {
fn := func(t *testing.T) {
h.AcceptedLevels = test.acceptedLevels
levels := h.Levels()
assert.Equal(t, test.expected, levels, test.description)
}
t.Run(test.name, fn)
}
}
type levelThresholdTest struct {
name string
level logrus.Level
expected []logrus.Level
description string
}
func TestLevelThreshold(t *testing.T) {
tests := []levelThresholdTest{
{
name: "unknownLogLevel",
level: logrus.Level(100),
expected: []logrus.Level{},
description: "An empty Level slice was expected but was not returned",
},
{
name: "errorLogLevel",
level: logrus.ErrorLevel,
expected: []logrus.Level{logrus.PanicLevel, logrus.FatalLevel, logrus.ErrorLevel},
description: "The panic, fatal, and error levels were expected but were not returned",
},
}
for _, test := range tests {
fn := func(t *testing.T) {
levels := LevelThreshold(test.level)
assert.Equal(t, test.expected, levels, test.description)
}
t.Run(test.name, fn)
}
}
func TestInvalidRegex(t *testing.T) {
e := &logrus.Entry{}
h = &Hook{RedactionList: []string{"\\"}}
err := h.Fire(e)
assert.NotNil(t, err)
}
type EntryDataValuesTest struct {
name string
redactionList []string
logFields logrus.Fields
expected logrus.Fields
description string //nolint
}
// Test that any occurrence of a redaction pattern
// in the values of the entry's data fields is redacted.
func TestEntryDataValues(t *testing.T) {
tests := []EntryDataValuesTest{
{
name: "match on key",
redactionList: []string{"Password"},
logFields: logrus.Fields{"Password": "password123!"},
expected: logrus.Fields{"Password": "[REDACTED]"},
description: "Password value should have been redacted, but was not.",
},
{
name: "string value",
redactionList: []string{"William"},
logFields: logrus.Fields{"Description": "His name is William"},
expected: logrus.Fields{"Description": "His name is [REDACTED]"},
description: "William should have been redacted, but was not.",
},
}
for _, test := range tests {
fn := func(t *testing.T) {
logEntry := &logrus.Entry{
Data: test.logFields,
}
h = &Hook{RedactionList: test.redactionList}
err := h.Fire(logEntry)
assert.Nil(t, err)
assert.Equal(t, test.expected, logEntry.Data)
}
t.Run(test.name, fn)
}
}
// Test that any occurrence of a redaction pattern
// in the entry's Message field is redacted.
func TestEntryMessage(t *testing.T) {
logEntry := &logrus.Entry{
Message: "Secret Password: password123!",
}
h = &Hook{RedactionList: []string{`(Password: ).*`}}
err := h.Fire(logEntry)
assert.Nil(t, err)
assert.Equal(t, "Secret Password: [REDACTED]", logEntry.Message)
}