jsonfilelog_test.go 9.1 KB

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