123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203 |
- // Copyright (c) 2016 Uber Technologies, Inc.
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to deal
- // in the Software without restriction, including without limitation the rights
- // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- // copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- // THE SOFTWARE.
- package zap
- import (
- "os"
- "path/filepath"
- "sync/atomic"
- "testing"
- "github.com/stretchr/testify/assert"
- "github.com/stretchr/testify/require"
- "go.uber.org/zap/zapcore"
- )
- func TestConfig(t *testing.T) {
- tests := []struct {
- desc string
- cfg Config
- expectN int64
- expectRe string
- }{
- {
- desc: "production",
- cfg: NewProductionConfig(),
- expectN: 2 + 100 + 1, // 2 from initial logs, 100 initial sampled logs, 1 from off-by-one in sampler
- expectRe: `{"level":"info","caller":"[a-z0-9_-]+/config_test.go:\d+","msg":"info","k":"v","z":"zz"}` + "\n" +
- `{"level":"warn","caller":"[a-z0-9_-]+/config_test.go:\d+","msg":"warn","k":"v","z":"zz"}` + "\n",
- },
- {
- desc: "development",
- cfg: NewDevelopmentConfig(),
- expectN: 3 + 200, // 3 initial logs, all 200 subsequent logs
- expectRe: "DEBUG\t[a-z0-9_-]+/config_test.go:" + `\d+` + "\tdebug\t" + `{"k": "v", "z": "zz"}` + "\n" +
- "INFO\t[a-z0-9_-]+/config_test.go:" + `\d+` + "\tinfo\t" + `{"k": "v", "z": "zz"}` + "\n" +
- "WARN\t[a-z0-9_-]+/config_test.go:" + `\d+` + "\twarn\t" + `{"k": "v", "z": "zz"}` + "\n" +
- `go.uber.org/zap.TestConfig.\w+`,
- },
- }
- for _, tt := range tests {
- t.Run(tt.desc, func(t *testing.T) {
- logOut := filepath.Join(t.TempDir(), "test.log")
- tt.cfg.OutputPaths = []string{logOut}
- tt.cfg.EncoderConfig.TimeKey = "" // no timestamps in tests
- tt.cfg.InitialFields = map[string]interface{}{"z": "zz", "k": "v"}
- hook, count := makeCountingHook()
- logger, err := tt.cfg.Build(Hooks(hook))
- require.NoError(t, err, "Unexpected error constructing logger.")
- logger.Debug("debug")
- logger.Info("info")
- logger.Warn("warn")
- byteContents, err := os.ReadFile(logOut)
- require.NoError(t, err, "Couldn't read log contents from temp file.")
- logs := string(byteContents)
- assert.Regexp(t, tt.expectRe, logs, "Unexpected log output.")
- for i := 0; i < 200; i++ {
- logger.Info("sampling")
- }
- assert.Equal(t, tt.expectN, count.Load(), "Hook called an unexpected number of times.")
- })
- }
- }
- func TestConfigWithInvalidPaths(t *testing.T) {
- tests := []struct {
- desc string
- output string
- errOutput string
- }{
- {"output directory doesn't exist", "/tmp/not-there/foo.log", "stderr"},
- {"error output directory doesn't exist", "stdout", "/tmp/not-there/foo-errors.log"},
- {"neither output directory exists", "/tmp/not-there/foo.log", "/tmp/not-there/foo-errors.log"},
- }
- for _, tt := range tests {
- t.Run(tt.desc, func(t *testing.T) {
- cfg := NewProductionConfig()
- cfg.OutputPaths = []string{tt.output}
- cfg.ErrorOutputPaths = []string{tt.errOutput}
- _, err := cfg.Build()
- assert.Error(t, err, "Expected an error opening a non-existent directory.")
- })
- }
- }
- func TestConfigWithMissingAttributes(t *testing.T) {
- tests := []struct {
- desc string
- cfg Config
- expectErr string
- }{
- {
- desc: "missing level",
- cfg: Config{
- Encoding: "json",
- },
- expectErr: "missing Level",
- },
- {
- desc: "missing encoder time in encoder config",
- cfg: Config{
- Level: NewAtomicLevelAt(zapcore.InfoLevel),
- Encoding: "json",
- EncoderConfig: zapcore.EncoderConfig{
- MessageKey: "msg",
- TimeKey: "ts",
- },
- },
- expectErr: "missing EncodeTime in EncoderConfig",
- },
- }
- for _, tt := range tests {
- t.Run(tt.desc, func(t *testing.T) {
- cfg := tt.cfg
- _, err := cfg.Build()
- assert.EqualError(t, err, tt.expectErr)
- })
- }
- }
- func makeSamplerCountingHook() (h func(zapcore.Entry, zapcore.SamplingDecision),
- dropped, sampled *atomic.Int64,
- ) {
- dropped = new(atomic.Int64)
- sampled = new(atomic.Int64)
- h = func(_ zapcore.Entry, dec zapcore.SamplingDecision) {
- if dec&zapcore.LogDropped > 0 {
- dropped.Add(1)
- } else if dec&zapcore.LogSampled > 0 {
- sampled.Add(1)
- }
- }
- return h, dropped, sampled
- }
- func TestConfigWithSamplingHook(t *testing.T) {
- shook, dcount, scount := makeSamplerCountingHook()
- cfg := Config{
- Level: NewAtomicLevelAt(InfoLevel),
- Development: false,
- Sampling: &SamplingConfig{
- Initial: 100,
- Thereafter: 100,
- Hook: shook,
- },
- Encoding: "json",
- EncoderConfig: NewProductionEncoderConfig(),
- OutputPaths: []string{"stderr"},
- ErrorOutputPaths: []string{"stderr"},
- }
- expectRe := `{"level":"info","caller":"[a-z0-9_-]+/config_test.go:\d+","msg":"info","k":"v","z":"zz"}` + "\n" +
- `{"level":"warn","caller":"[a-z0-9_-]+/config_test.go:\d+","msg":"warn","k":"v","z":"zz"}` + "\n"
- expectDropped := 99 // 200 - 100 initial - 1 thereafter
- expectSampled := 103 // 2 from initial + 100 + 1 thereafter
- logOut := filepath.Join(t.TempDir(), "test.log")
- cfg.OutputPaths = []string{logOut}
- cfg.EncoderConfig.TimeKey = "" // no timestamps in tests
- cfg.InitialFields = map[string]interface{}{"z": "zz", "k": "v"}
- logger, err := cfg.Build()
- require.NoError(t, err, "Unexpected error constructing logger.")
- logger.Debug("debug")
- logger.Info("info")
- logger.Warn("warn")
- byteContents, err := os.ReadFile(logOut)
- require.NoError(t, err, "Couldn't read log contents from temp file.")
- logs := string(byteContents)
- assert.Regexp(t, expectRe, logs, "Unexpected log output.")
- for i := 0; i < 200; i++ {
- logger.Info("sampling")
- }
- assert.Equal(t, int64(expectDropped), dcount.Load())
- assert.Equal(t, int64(expectSampled), scount.Load())
- }
|