jsonfilelog_test.go 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. package jsonfilelog
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "io/ioutil"
  6. "os"
  7. "path/filepath"
  8. "reflect"
  9. "strconv"
  10. "testing"
  11. "time"
  12. "github.com/docker/docker/daemon/logger"
  13. "github.com/docker/docker/daemon/logger/jsonfilelog/jsonlog"
  14. "github.com/gotestyourself/gotestyourself/fs"
  15. "github.com/stretchr/testify/assert"
  16. "github.com/stretchr/testify/require"
  17. )
  18. func TestJSONFileLogger(t *testing.T) {
  19. cid := "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657"
  20. tmp, err := ioutil.TempDir("", "docker-logger-")
  21. if err != nil {
  22. t.Fatal(err)
  23. }
  24. defer os.RemoveAll(tmp)
  25. filename := filepath.Join(tmp, "container.log")
  26. l, err := New(logger.Info{
  27. ContainerID: cid,
  28. LogPath: filename,
  29. })
  30. if err != nil {
  31. t.Fatal(err)
  32. }
  33. defer l.Close()
  34. if err := l.Log(&logger.Message{Line: []byte("line1"), Source: "src1"}); err != nil {
  35. t.Fatal(err)
  36. }
  37. if err := l.Log(&logger.Message{Line: []byte("line2"), Source: "src2"}); err != nil {
  38. t.Fatal(err)
  39. }
  40. if err := l.Log(&logger.Message{Line: []byte("line3"), Source: "src3"}); err != nil {
  41. t.Fatal(err)
  42. }
  43. res, err := ioutil.ReadFile(filename)
  44. if err != nil {
  45. t.Fatal(err)
  46. }
  47. expected := `{"log":"line1\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  48. {"log":"line2\n","stream":"src2","time":"0001-01-01T00:00:00Z"}
  49. {"log":"line3\n","stream":"src3","time":"0001-01-01T00:00:00Z"}
  50. `
  51. if string(res) != expected {
  52. t.Fatalf("Wrong log content: %q, expected %q", res, expected)
  53. }
  54. }
  55. func TestJSONFileLoggerWithTags(t *testing.T) {
  56. cid := "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657"
  57. cname := "test-container"
  58. tmp, err := ioutil.TempDir("", "docker-logger-")
  59. require.NoError(t, err)
  60. defer os.RemoveAll(tmp)
  61. filename := filepath.Join(tmp, "container.log")
  62. l, err := New(logger.Info{
  63. Config: map[string]string{
  64. "tag": "{{.ID}}/{{.Name}}", // first 12 characters of ContainerID and full ContainerName
  65. },
  66. ContainerID: cid,
  67. ContainerName: cname,
  68. LogPath: filename,
  69. })
  70. require.NoError(t, err)
  71. defer l.Close()
  72. err = l.Log(&logger.Message{Line: []byte("line1"), Source: "src1"})
  73. require.NoError(t, err)
  74. err = l.Log(&logger.Message{Line: []byte("line2"), Source: "src2"})
  75. require.NoError(t, err)
  76. err = l.Log(&logger.Message{Line: []byte("line3"), Source: "src3"})
  77. require.NoError(t, err)
  78. res, err := ioutil.ReadFile(filename)
  79. require.NoError(t, err)
  80. expected := `{"log":"line1\n","stream":"src1","attrs":{"tag":"a7317399f3f8/test-container"},"time":"0001-01-01T00:00:00Z"}
  81. {"log":"line2\n","stream":"src2","attrs":{"tag":"a7317399f3f8/test-container"},"time":"0001-01-01T00:00:00Z"}
  82. {"log":"line3\n","stream":"src3","attrs":{"tag":"a7317399f3f8/test-container"},"time":"0001-01-01T00:00:00Z"}
  83. `
  84. assert.Equal(t, expected, string(res))
  85. }
  86. func BenchmarkJSONFileLoggerLog(b *testing.B) {
  87. tmp := fs.NewDir(b, "bench-jsonfilelog")
  88. defer tmp.Remove()
  89. jsonlogger, err := New(logger.Info{
  90. ContainerID: "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657",
  91. LogPath: tmp.Join("container.log"),
  92. Config: map[string]string{
  93. "labels": "first,second",
  94. },
  95. ContainerLabels: map[string]string{
  96. "first": "label_value",
  97. "second": "label_foo",
  98. },
  99. })
  100. require.NoError(b, err)
  101. defer jsonlogger.Close()
  102. msg := &logger.Message{
  103. Line: []byte("Line that thinks that it is log line from docker\n"),
  104. Source: "stderr",
  105. Timestamp: time.Now().UTC(),
  106. }
  107. buf := bytes.NewBuffer(nil)
  108. require.NoError(b, marshalMessage(msg, nil, buf))
  109. b.SetBytes(int64(buf.Len()))
  110. b.ResetTimer()
  111. for i := 0; i < b.N; i++ {
  112. if err := jsonlogger.Log(msg); err != nil {
  113. b.Fatal(err)
  114. }
  115. }
  116. }
  117. func TestJSONFileLoggerWithOpts(t *testing.T) {
  118. cid := "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657"
  119. tmp, err := ioutil.TempDir("", "docker-logger-")
  120. if err != nil {
  121. t.Fatal(err)
  122. }
  123. defer os.RemoveAll(tmp)
  124. filename := filepath.Join(tmp, "container.log")
  125. config := map[string]string{"max-file": "2", "max-size": "1k"}
  126. l, err := New(logger.Info{
  127. ContainerID: cid,
  128. LogPath: filename,
  129. Config: config,
  130. })
  131. if err != nil {
  132. t.Fatal(err)
  133. }
  134. defer l.Close()
  135. for i := 0; i < 20; i++ {
  136. if err := l.Log(&logger.Message{Line: []byte("line" + strconv.Itoa(i)), Source: "src1"}); err != nil {
  137. t.Fatal(err)
  138. }
  139. }
  140. res, err := ioutil.ReadFile(filename)
  141. if err != nil {
  142. t.Fatal(err)
  143. }
  144. penUlt, err := ioutil.ReadFile(filename + ".1")
  145. if err != nil {
  146. t.Fatal(err)
  147. }
  148. expectedPenultimate := `{"log":"line0\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  149. {"log":"line1\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  150. {"log":"line2\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  151. {"log":"line3\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  152. {"log":"line4\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  153. {"log":"line5\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  154. {"log":"line6\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  155. {"log":"line7\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  156. {"log":"line8\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  157. {"log":"line9\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  158. {"log":"line10\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  159. {"log":"line11\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  160. {"log":"line12\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  161. {"log":"line13\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  162. {"log":"line14\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  163. {"log":"line15\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  164. `
  165. expected := `{"log":"line16\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  166. {"log":"line17\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  167. {"log":"line18\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  168. {"log":"line19\n","stream":"src1","time":"0001-01-01T00:00:00Z"}
  169. `
  170. if string(res) != expected {
  171. t.Fatalf("Wrong log content: %q, expected %q", res, expected)
  172. }
  173. if string(penUlt) != expectedPenultimate {
  174. t.Fatalf("Wrong log content: %q, expected %q", penUlt, expectedPenultimate)
  175. }
  176. }
  177. func TestJSONFileLoggerWithLabelsEnv(t *testing.T) {
  178. cid := "a7317399f3f857173c6179d44823594f8294678dea9999662e5c625b5a1c7657"
  179. tmp, err := ioutil.TempDir("", "docker-logger-")
  180. if err != nil {
  181. t.Fatal(err)
  182. }
  183. defer os.RemoveAll(tmp)
  184. filename := filepath.Join(tmp, "container.log")
  185. config := map[string]string{"labels": "rack,dc", "env": "environ,debug,ssl", "env-regex": "^dc"}
  186. l, err := New(logger.Info{
  187. ContainerID: cid,
  188. LogPath: filename,
  189. Config: config,
  190. ContainerLabels: map[string]string{"rack": "101", "dc": "lhr"},
  191. ContainerEnv: []string{"environ=production", "debug=false", "port=10001", "ssl=true", "dc_region=west"},
  192. })
  193. if err != nil {
  194. t.Fatal(err)
  195. }
  196. defer l.Close()
  197. if err := l.Log(&logger.Message{Line: []byte("line"), Source: "src1"}); err != nil {
  198. t.Fatal(err)
  199. }
  200. res, err := ioutil.ReadFile(filename)
  201. if err != nil {
  202. t.Fatal(err)
  203. }
  204. var jsonLog jsonlog.JSONLogs
  205. if err := json.Unmarshal(res, &jsonLog); err != nil {
  206. t.Fatal(err)
  207. }
  208. extra := make(map[string]string)
  209. if err := json.Unmarshal(jsonLog.RawAttrs, &extra); err != nil {
  210. t.Fatal(err)
  211. }
  212. expected := map[string]string{
  213. "rack": "101",
  214. "dc": "lhr",
  215. "environ": "production",
  216. "debug": "false",
  217. "ssl": "true",
  218. "dc_region": "west",
  219. }
  220. if !reflect.DeepEqual(extra, expected) {
  221. t.Fatalf("Wrong log attrs: %q, expected %q", extra, expected)
  222. }
  223. }