123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262 |
- // Copyright 2019 The etcd Authors
- //
- // 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 traceutil
- import (
- "bytes"
- "context"
- "fmt"
- "io/ioutil"
- "os"
- "path/filepath"
- "testing"
- "time"
- "go.uber.org/zap"
- )
- func TestGet(t *testing.T) {
- traceForTest := &Trace{operation: "test"}
- tests := []struct {
- name string
- inputCtx context.Context
- outputTrace *Trace
- }{
- {
- name: "When the context does not have trace",
- inputCtx: context.TODO(),
- outputTrace: TODO(),
- },
- {
- name: "When the context has trace",
- inputCtx: context.WithValue(context.Background(), TraceKey, traceForTest),
- outputTrace: traceForTest,
- },
- }
- for _, tt := range tests {
- t.Run(tt.name, func(t *testing.T) {
- trace := Get(tt.inputCtx)
- if trace == nil {
- t.Errorf("Expected %v; Got nil", tt.outputTrace)
- }
- if trace.operation != tt.outputTrace.operation {
- t.Errorf("Expected %v; Got %v", tt.outputTrace, trace)
- }
- })
- }
- }
- func TestCreate(t *testing.T) {
- var (
- op = "Test"
- steps = []string{"Step1, Step2"}
- fields = []Field{
- {"traceKey1", "traceValue1"},
- {"traceKey2", "traceValue2"},
- }
- stepFields = []Field{
- {"stepKey1", "stepValue2"},
- {"stepKey2", "stepValue2"},
- }
- )
- trace := New(op, nil, fields[0], fields[1])
- if trace.operation != op {
- t.Errorf("Expected %v; Got %v", op, trace.operation)
- }
- for i, f := range trace.fields {
- if f.Key != fields[i].Key {
- t.Errorf("Expected %v; Got %v", fields[i].Key, f.Key)
- }
- if f.Value != fields[i].Value {
- t.Errorf("Expected %v; Got %v", fields[i].Value, f.Value)
- }
- }
- for i, v := range steps {
- trace.Step(v, stepFields[i])
- }
- for i, v := range trace.steps {
- if steps[i] != v.msg {
- t.Errorf("Expected %v; Got %v", steps[i], v.msg)
- }
- if stepFields[i].Key != v.fields[0].Key {
- t.Errorf("Expected %v; Got %v", stepFields[i].Key, v.fields[0].Key)
- }
- if stepFields[i].Value != v.fields[0].Value {
- t.Errorf("Expected %v; Got %v", stepFields[i].Value, v.fields[0].Value)
- }
- }
- }
- func TestLog(t *testing.T) {
- tests := []struct {
- name string
- trace *Trace
- fields []Field
- expectedMsg []string
- }{
- {
- name: "When dump all logs",
- trace: &Trace{
- operation: "Test",
- startTime: time.Now().Add(-100 * time.Millisecond),
- steps: []step{
- {time: time.Now().Add(-80 * time.Millisecond), msg: "msg1"},
- {time: time.Now().Add(-50 * time.Millisecond), msg: "msg2"},
- },
- },
- expectedMsg: []string{
- "msg1", "msg2",
- },
- },
- {
- name: "When trace has fields",
- trace: &Trace{
- operation: "Test",
- startTime: time.Now().Add(-100 * time.Millisecond),
- steps: []step{
- {
- time: time.Now().Add(-80 * time.Millisecond),
- msg: "msg1",
- fields: []Field{{"stepKey1", "stepValue1"}},
- },
- {
- time: time.Now().Add(-50 * time.Millisecond),
- msg: "msg2",
- fields: []Field{{"stepKey2", "stepValue2"}},
- },
- },
- },
- fields: []Field{
- {"traceKey1", "traceValue1"},
- {"count", 1},
- },
- expectedMsg: []string{
- "Test",
- "msg1", "msg2",
- "traceKey1:traceValue1", "count:1",
- "stepKey1:stepValue1", "stepKey2:stepValue2",
- },
- },
- }
- for _, tt := range tests {
- t.Run(tt.name, func(t *testing.T) {
- logPath := filepath.Join(os.TempDir(), fmt.Sprintf("test-log-%d", time.Now().UnixNano()))
- defer os.RemoveAll(logPath)
- lcfg := zap.NewProductionConfig()
- lcfg.OutputPaths = []string{logPath}
- lcfg.ErrorOutputPaths = []string{logPath}
- lg, _ := lcfg.Build()
- for _, f := range tt.fields {
- tt.trace.AddField(f)
- }
- tt.trace.lg = lg
- tt.trace.Log()
- data, err := ioutil.ReadFile(logPath)
- if err != nil {
- t.Fatal(err)
- }
- for _, msg := range tt.expectedMsg {
- if !bytes.Contains(data, []byte(msg)) {
- t.Errorf("Expected to find %v in log", msg)
- }
- }
- })
- }
- }
- func TestLogIfLong(t *testing.T) {
- tests := []struct {
- name string
- threshold time.Duration
- trace *Trace
- expectedMsg []string
- }{
- {
- name: "When the duration is smaller than threshold",
- threshold: time.Duration(200 * time.Millisecond),
- trace: &Trace{
- operation: "Test",
- startTime: time.Now().Add(-100 * time.Millisecond),
- steps: []step{
- {time: time.Now().Add(-50 * time.Millisecond), msg: "msg1"},
- {time: time.Now(), msg: "msg2"},
- },
- },
- expectedMsg: []string{},
- },
- {
- name: "When the duration is longer than threshold",
- threshold: time.Duration(50 * time.Millisecond),
- trace: &Trace{
- operation: "Test",
- startTime: time.Now().Add(-100 * time.Millisecond),
- steps: []step{
- {time: time.Now().Add(-50 * time.Millisecond), msg: "msg1"},
- {time: time.Now(), msg: "msg2"},
- },
- },
- expectedMsg: []string{
- "msg1", "msg2",
- },
- },
- {
- name: "When not all steps are longer than step threshold",
- threshold: time.Duration(50 * time.Millisecond),
- trace: &Trace{
- operation: "Test",
- startTime: time.Now().Add(-100 * time.Millisecond),
- steps: []step{
- {time: time.Now(), msg: "msg1"},
- {time: time.Now(), msg: "msg2"},
- },
- },
- expectedMsg: []string{
- "msg1",
- },
- },
- }
- for _, tt := range tests {
- t.Run(tt.name, func(t *testing.T) {
- logPath := filepath.Join(os.TempDir(), fmt.Sprintf("test-log-%d", time.Now().UnixNano()))
- defer os.RemoveAll(logPath)
- lcfg := zap.NewProductionConfig()
- lcfg.OutputPaths = []string{logPath}
- lcfg.ErrorOutputPaths = []string{logPath}
- lg, _ := lcfg.Build()
- tt.trace.lg = lg
- tt.trace.LogIfLong(tt.threshold)
- data, err := ioutil.ReadFile(logPath)
- if err != nil {
- t.Fatal(err)
- }
- for _, msg := range tt.expectedMsg {
- if !bytes.Contains(data, []byte(msg)) {
- t.Errorf("Expected to find %v in log", msg)
- }
- }
- })
- }
- }
|