123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 |
- package logrus
- import (
- "bytes"
- "errors"
- "fmt"
- "strings"
- "testing"
- "time"
- "github.com/stretchr/testify/assert"
- )
- func TestFormatting(t *testing.T) {
- tf := &TextFormatter{DisableColors: true}
- testCases := []struct {
- value string
- expected string
- }{
- {`foo`, "time=\"0001-01-01T00:00:00Z\" level=panic test=foo\n"},
- }
- for _, tc := range testCases {
- b, _ := tf.Format(WithField("test", tc.value))
- if string(b) != tc.expected {
- t.Errorf("formatting expected for %q (result was %q instead of %q)", tc.value, string(b), tc.expected)
- }
- }
- }
- func TestQuoting(t *testing.T) {
- tf := &TextFormatter{DisableColors: true}
- checkQuoting := func(q bool, value interface{}) {
- b, _ := tf.Format(WithField("test", value))
- idx := bytes.Index(b, ([]byte)("test="))
- cont := bytes.Contains(b[idx+5:], []byte("\""))
- if cont != q {
- if q {
- t.Errorf("quoting expected for: %#v", value)
- } else {
- t.Errorf("quoting not expected for: %#v", value)
- }
- }
- }
- checkQuoting(false, "")
- checkQuoting(false, "abcd")
- checkQuoting(false, "v1.0")
- checkQuoting(false, "1234567890")
- checkQuoting(false, "/foobar")
- checkQuoting(false, "foo_bar")
- checkQuoting(false, "foo@bar")
- checkQuoting(false, "foobar^")
- checkQuoting(false, "+/-_^@f.oobar")
- checkQuoting(true, "foobar$")
- checkQuoting(true, "&foobar")
- checkQuoting(true, "x y")
- checkQuoting(true, "x,y")
- checkQuoting(false, errors.New("invalid"))
- checkQuoting(true, errors.New("invalid argument"))
- // Test for quoting empty fields.
- tf.QuoteEmptyFields = true
- checkQuoting(true, "")
- checkQuoting(false, "abcd")
- checkQuoting(true, errors.New("invalid argument"))
- }
- func TestEscaping(t *testing.T) {
- tf := &TextFormatter{DisableColors: true}
- testCases := []struct {
- value string
- expected string
- }{
- {`ba"r`, `ba\"r`},
- {`ba'r`, `ba'r`},
- }
- for _, tc := range testCases {
- b, _ := tf.Format(WithField("test", tc.value))
- if !bytes.Contains(b, []byte(tc.expected)) {
- t.Errorf("escaping expected for %q (result was %q instead of %q)", tc.value, string(b), tc.expected)
- }
- }
- }
- func TestEscaping_Interface(t *testing.T) {
- tf := &TextFormatter{DisableColors: true}
- ts := time.Now()
- testCases := []struct {
- value interface{}
- expected string
- }{
- {ts, fmt.Sprintf("\"%s\"", ts.String())},
- {errors.New("error: something went wrong"), "\"error: something went wrong\""},
- }
- for _, tc := range testCases {
- b, _ := tf.Format(WithField("test", tc.value))
- if !bytes.Contains(b, []byte(tc.expected)) {
- t.Errorf("escaping expected for %q (result was %q instead of %q)", tc.value, string(b), tc.expected)
- }
- }
- }
- func TestTimestampFormat(t *testing.T) {
- checkTimeStr := func(format string) {
- customFormatter := &TextFormatter{DisableColors: true, TimestampFormat: format}
- customStr, _ := customFormatter.Format(WithField("test", "test"))
- timeStart := bytes.Index(customStr, ([]byte)("time="))
- timeEnd := bytes.Index(customStr, ([]byte)("level="))
- timeStr := customStr[timeStart+5+len("\"") : timeEnd-1-len("\"")]
- if format == "" {
- format = time.RFC3339
- }
- _, e := time.Parse(format, (string)(timeStr))
- if e != nil {
- t.Errorf("time string \"%s\" did not match provided time format \"%s\": %s", timeStr, format, e)
- }
- }
- checkTimeStr("2006-01-02T15:04:05.000000000Z07:00")
- checkTimeStr("Mon Jan _2 15:04:05 2006")
- checkTimeStr("")
- }
- func TestDisableLevelTruncation(t *testing.T) {
- entry := &Entry{
- Time: time.Now(),
- Message: "testing",
- }
- keys := []string{}
- timestampFormat := "Mon Jan 2 15:04:05 -0700 MST 2006"
- checkDisableTruncation := func(disabled bool, level Level) {
- tf := &TextFormatter{DisableLevelTruncation: disabled}
- var b bytes.Buffer
- entry.Level = level
- tf.printColored(&b, entry, keys, timestampFormat)
- logLine := (&b).String()
- if disabled {
- expected := strings.ToUpper(level.String())
- if !strings.Contains(logLine, expected) {
- t.Errorf("level string expected to be %s when truncation disabled", expected)
- }
- } else {
- expected := strings.ToUpper(level.String())
- if len(level.String()) > 4 {
- if strings.Contains(logLine, expected) {
- t.Errorf("level string %s expected to be truncated to %s when truncation is enabled", expected, expected[0:4])
- }
- } else {
- if !strings.Contains(logLine, expected) {
- t.Errorf("level string expected to be %s when truncation is enabled and level string is below truncation threshold", expected)
- }
- }
- }
- }
- checkDisableTruncation(true, DebugLevel)
- checkDisableTruncation(true, InfoLevel)
- checkDisableTruncation(false, ErrorLevel)
- checkDisableTruncation(false, InfoLevel)
- }
- func TestDisableTimestampWithColoredOutput(t *testing.T) {
- tf := &TextFormatter{DisableTimestamp: true, ForceColors: true}
- b, _ := tf.Format(WithField("test", "test"))
- if strings.Contains(string(b), "[0000]") {
- t.Error("timestamp not expected when DisableTimestamp is true")
- }
- }
- func TestTextFormatterFieldMap(t *testing.T) {
- formatter := &TextFormatter{
- DisableColors: true,
- FieldMap: FieldMap{
- FieldKeyMsg: "message",
- FieldKeyLevel: "somelevel",
- FieldKeyTime: "timeywimey",
- },
- }
- entry := &Entry{
- Message: "oh hi",
- Level: WarnLevel,
- Time: time.Date(1981, time.February, 24, 4, 28, 3, 100, time.UTC),
- Data: Fields{
- "field1": "f1",
- "message": "messagefield",
- "somelevel": "levelfield",
- "timeywimey": "timeywimeyfield",
- },
- }
- b, err := formatter.Format(entry)
- if err != nil {
- t.Fatal("Unable to format entry: ", err)
- }
- assert.Equal(t,
- `timeywimey="1981-02-24T04:28:03Z" `+
- `somelevel=warning `+
- `message="oh hi" `+
- `field1=f1 `+
- `fields.message=messagefield `+
- `fields.somelevel=levelfield `+
- `fields.timeywimey=timeywimeyfield`+"\n",
- string(b),
- "Formatted output doesn't respect FieldMap")
- }
- // TODO add tests for sorting etc., this requires a parser for the text
- // formatter output.
|