123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302 |
- package jsonfilelog // import "github.com/docker/docker/daemon/logger/jsonfilelog"
- import (
- "bytes"
- "compress/gzip"
- "encoding/json"
- "io/ioutil"
- "os"
- "path/filepath"
- "reflect"
- "strconv"
- "testing"
- "time"
- "github.com/docker/docker/daemon/logger"
- "github.com/docker/docker/daemon/logger/jsonfilelog/jsonlog"
- "github.com/gotestyourself/gotestyourself/fs"
- "github.com/stretchr/testify/assert"
- "github.com/stretchr/testify/require"
- )
- func TestJSONFileLogger(t *testing.T) {
- cid := "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657"
- tmp, err := ioutil.TempDir("", "docker-logger-")
- if err != nil {
- t.Fatal(err)
- }
- defer os.RemoveAll(tmp)
- filename := filepath.Join(tmp, "container.log")
- l, err := New(logger.Info{
- ContainerID: cid,
- LogPath: filename,
- })
- if err != nil {
- t.Fatal(err)
- }
- defer l.Close()
- if err := l.Log(&logger.Message{Line: []byte("line1"), Source: "src1"}); err != nil {
- t.Fatal(err)
- }
- if err := l.Log(&logger.Message{Line: []byte("line2"), Source: "src2"}); err != nil {
- t.Fatal(err)
- }
- if err := l.Log(&logger.Message{Line: []byte("line3"), Source: "src3"}); err != nil {
- t.Fatal(err)
- }
- res, err := ioutil.ReadFile(filename)
- if err != nil {
- t.Fatal(err)
- }
- expected := `{"log":"line1\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line2\n","stream":"src2","time":"0001-01-01T00:00:00Z"}
- {"log":"line3\n","stream":"src3","time":"0001-01-01T00:00:00Z"}
- `
- if string(res) != expected {
- t.Fatalf("Wrong log content: %q, expected %q", res, expected)
- }
- }
- func TestJSONFileLoggerWithTags(t *testing.T) {
- cid := "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657"
- cname := "test-container"
- tmp, err := ioutil.TempDir("", "docker-logger-")
- require.NoError(t, err)
- defer os.RemoveAll(tmp)
- filename := filepath.Join(tmp, "container.log")
- l, err := New(logger.Info{
- Config: map[string]string{
- "tag": "{{.ID}}/{{.Name}}", // first 12 characters of ContainerID and full ContainerName
- },
- ContainerID: cid,
- ContainerName: cname,
- LogPath: filename,
- })
- require.NoError(t, err)
- defer l.Close()
- err = l.Log(&logger.Message{Line: []byte("line1"), Source: "src1"})
- require.NoError(t, err)
- err = l.Log(&logger.Message{Line: []byte("line2"), Source: "src2"})
- require.NoError(t, err)
- err = l.Log(&logger.Message{Line: []byte("line3"), Source: "src3"})
- require.NoError(t, err)
- res, err := ioutil.ReadFile(filename)
- require.NoError(t, err)
- expected := `{"log":"line1\n","stream":"src1","attrs":{"tag":"a7317399f3f8/test-container"},"time":"0001-01-01T00:00:00Z"}
- {"log":"line2\n","stream":"src2","attrs":{"tag":"a7317399f3f8/test-container"},"time":"0001-01-01T00:00:00Z"}
- {"log":"line3\n","stream":"src3","attrs":{"tag":"a7317399f3f8/test-container"},"time":"0001-01-01T00:00:00Z"}
- `
- assert.Equal(t, expected, string(res))
- }
- func BenchmarkJSONFileLoggerLog(b *testing.B) {
- tmp := fs.NewDir(b, "bench-jsonfilelog")
- defer tmp.Remove()
- jsonlogger, err := New(logger.Info{
- ContainerID: "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657",
- LogPath: tmp.Join("container.log"),
- Config: map[string]string{
- "labels": "first,second",
- },
- ContainerLabels: map[string]string{
- "first": "label_value",
- "second": "label_foo",
- },
- })
- require.NoError(b, err)
- defer jsonlogger.Close()
- msg := &logger.Message{
- Line: []byte("Line that thinks that it is log line from docker\n"),
- Source: "stderr",
- Timestamp: time.Now().UTC(),
- }
- buf := bytes.NewBuffer(nil)
- require.NoError(b, marshalMessage(msg, nil, buf))
- b.SetBytes(int64(buf.Len()))
- b.ResetTimer()
- for i := 0; i < b.N; i++ {
- if err := jsonlogger.Log(msg); err != nil {
- b.Fatal(err)
- }
- }
- }
- func TestJSONFileLoggerWithOpts(t *testing.T) {
- cid := "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657"
- tmp, err := ioutil.TempDir("", "docker-logger-")
- if err != nil {
- t.Fatal(err)
- }
- defer os.RemoveAll(tmp)
- filename := filepath.Join(tmp, "container.log")
- config := map[string]string{"max-file": "3", "max-size": "1k", "compress": "true"}
- l, err := New(logger.Info{
- ContainerID: cid,
- LogPath: filename,
- Config: config,
- })
- if err != nil {
- t.Fatal(err)
- }
- defer l.Close()
- for i := 0; i < 36; i++ {
- if err := l.Log(&logger.Message{Line: []byte("line" + strconv.Itoa(i)), Source: "src1"}); err != nil {
- t.Fatal(err)
- }
- }
- res, err := ioutil.ReadFile(filename)
- if err != nil {
- t.Fatal(err)
- }
- penUlt, err := ioutil.ReadFile(filename + ".1")
- if err != nil {
- if !os.IsNotExist(err) {
- t.Fatal(err)
- }
- file, err := os.Open(filename + ".1.gz")
- defer file.Close()
- if err != nil {
- t.Fatal(err)
- }
- zipReader, err := gzip.NewReader(file)
- defer zipReader.Close()
- if err != nil {
- t.Fatal(err)
- }
- penUlt, err = ioutil.ReadAll(zipReader)
- if err != nil {
- t.Fatal(err)
- }
- }
- file, err := os.Open(filename + ".2.gz")
- defer file.Close()
- if err != nil {
- t.Fatal(err)
- }
- zipReader, err := gzip.NewReader(file)
- defer zipReader.Close()
- if err != nil {
- t.Fatal(err)
- }
- antepenult, err := ioutil.ReadAll(zipReader)
- if err != nil {
- t.Fatal(err)
- }
- expectedAntepenultimate := `{"log":"line0\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line1\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line2\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line3\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line4\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line5\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line6\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line7\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line8\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line9\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line10\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line11\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line12\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line13\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line14\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line15\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- `
- expectedPenultimate := `{"log":"line16\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line17\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line18\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line19\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line20\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line21\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line22\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line23\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line24\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line25\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line26\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line27\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line28\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line29\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line30\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line31\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- `
- expected := `{"log":"line32\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line33\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line34\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- {"log":"line35\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
- `
- if string(res) != expected {
- t.Fatalf("Wrong log content: %q, expected %q", res, expected)
- }
- if string(penUlt) != expectedPenultimate {
- t.Fatalf("Wrong log content: %q, expected %q", penUlt, expectedPenultimate)
- }
- if string(antepenult) != expectedAntepenultimate {
- t.Fatalf("Wrong log content: %q, expected %q", antepenult, expectedAntepenultimate)
- }
- }
- func TestJSONFileLoggerWithLabelsEnv(t *testing.T) {
- cid := "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657"
- tmp, err := ioutil.TempDir("", "docker-logger-")
- if err != nil {
- t.Fatal(err)
- }
- defer os.RemoveAll(tmp)
- filename := filepath.Join(tmp, "container.log")
- config := map[string]string{"labels": "rack,dc", "env": "environ,debug,ssl", "env-regex": "^dc"}
- l, err := New(logger.Info{
- ContainerID: cid,
- LogPath: filename,
- Config: config,
- ContainerLabels: map[string]string{"rack": "101", "dc": "lhr"},
- ContainerEnv: []string{"environ=production", "debug=false", "port=10001", "ssl=true", "dc_region=west"},
- })
- if err != nil {
- t.Fatal(err)
- }
- defer l.Close()
- if err := l.Log(&logger.Message{Line: []byte("line"), Source: "src1"}); err != nil {
- t.Fatal(err)
- }
- res, err := ioutil.ReadFile(filename)
- if err != nil {
- t.Fatal(err)
- }
- var jsonLog jsonlog.JSONLogs
- if err := json.Unmarshal(res, &jsonLog); err != nil {
- t.Fatal(err)
- }
- extra := make(map[string]string)
- if err := json.Unmarshal(jsonLog.RawAttrs, &extra); err != nil {
- t.Fatal(err)
- }
- expected := map[string]string{
- "rack": "101",
- "dc": "lhr",
- "environ": "production",
- "debug": "false",
- "ssl": "true",
- "dc_region": "west",
- }
- if !reflect.DeepEqual(extra, expected) {
- t.Fatalf("Wrong log attrs: %q, expected %q", extra, expected)
- }
- }
|