gelf.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. // +build linux
  2. // Package gelf provides the log driver for forwarding server logs to
  3. // endpoints that support the Graylog Extended Log Format.
  4. package gelf
  5. import (
  6. "compress/flate"
  7. "encoding/json"
  8. "fmt"
  9. "net"
  10. "net/url"
  11. "strconv"
  12. "time"
  13. "github.com/Graylog2/go-gelf/gelf"
  14. "github.com/Sirupsen/logrus"
  15. "github.com/docker/docker/daemon/logger"
  16. "github.com/docker/docker/daemon/logger/loggerutils"
  17. "github.com/docker/docker/pkg/urlutil"
  18. )
  19. const name = "gelf"
  20. type gelfLogger struct {
  21. writer *gelf.Writer
  22. ctx logger.Context
  23. hostname string
  24. rawExtra json.RawMessage
  25. }
  26. func init() {
  27. if err := logger.RegisterLogDriver(name, New); err != nil {
  28. logrus.Fatal(err)
  29. }
  30. if err := logger.RegisterLogOptValidator(name, ValidateLogOpt); err != nil {
  31. logrus.Fatal(err)
  32. }
  33. }
  34. // New creates a gelf logger using the configuration passed in on the
  35. // context. The supported context configuration variable is gelf-address.
  36. func New(ctx logger.Context) (logger.Logger, error) {
  37. // parse gelf address
  38. address, err := parseAddress(ctx.Config["gelf-address"])
  39. if err != nil {
  40. return nil, err
  41. }
  42. // collect extra data for GELF message
  43. hostname, err := ctx.Hostname()
  44. if err != nil {
  45. return nil, fmt.Errorf("gelf: cannot access hostname to set source field")
  46. }
  47. // parse log tag
  48. tag, err := loggerutils.ParseLogTag(ctx, loggerutils.DefaultTemplate)
  49. if err != nil {
  50. return nil, err
  51. }
  52. extra := map[string]interface{}{
  53. "_container_id": ctx.ContainerID,
  54. "_container_name": ctx.Name(),
  55. "_image_id": ctx.ContainerImageID,
  56. "_image_name": ctx.ContainerImageName,
  57. "_command": ctx.Command(),
  58. "_tag": tag,
  59. "_created": ctx.ContainerCreated,
  60. }
  61. extraAttrs := ctx.ExtraAttributes(func(key string) string {
  62. if key[0] == '_' {
  63. return key
  64. }
  65. return "_" + key
  66. })
  67. for k, v := range extraAttrs {
  68. extra[k] = v
  69. }
  70. rawExtra, err := json.Marshal(extra)
  71. if err != nil {
  72. return nil, err
  73. }
  74. // create new gelfWriter
  75. gelfWriter, err := gelf.NewWriter(address)
  76. if err != nil {
  77. return nil, fmt.Errorf("gelf: cannot connect to GELF endpoint: %s %v", address, err)
  78. }
  79. if v, ok := ctx.Config["gelf-compression-type"]; ok {
  80. switch v {
  81. case "gzip":
  82. gelfWriter.CompressionType = gelf.CompressGzip
  83. case "zlib":
  84. gelfWriter.CompressionType = gelf.CompressZlib
  85. case "none":
  86. gelfWriter.CompressionType = gelf.CompressNone
  87. default:
  88. return nil, fmt.Errorf("gelf: invalid compression type %q", v)
  89. }
  90. }
  91. if v, ok := ctx.Config["gelf-compression-level"]; ok {
  92. val, err := strconv.Atoi(v)
  93. if err != nil {
  94. return nil, fmt.Errorf("gelf: invalid compression level %s, err %v", v, err)
  95. }
  96. gelfWriter.CompressionLevel = val
  97. }
  98. return &gelfLogger{
  99. writer: gelfWriter,
  100. ctx: ctx,
  101. hostname: hostname,
  102. rawExtra: rawExtra,
  103. }, nil
  104. }
  105. func (s *gelfLogger) Log(msg *logger.Message) error {
  106. level := gelf.LOG_INFO
  107. if msg.Source == "stderr" {
  108. level = gelf.LOG_ERR
  109. }
  110. m := gelf.Message{
  111. Version: "1.1",
  112. Host: s.hostname,
  113. Short: string(msg.Line),
  114. TimeUnix: float64(msg.Timestamp.UnixNano()/int64(time.Millisecond)) / 1000.0,
  115. Level: level,
  116. RawExtra: s.rawExtra,
  117. }
  118. if err := s.writer.WriteMessage(&m); err != nil {
  119. return fmt.Errorf("gelf: cannot send GELF message: %v", err)
  120. }
  121. return nil
  122. }
  123. func (s *gelfLogger) Close() error {
  124. return s.writer.Close()
  125. }
  126. func (s *gelfLogger) Name() string {
  127. return name
  128. }
  129. // ValidateLogOpt looks for gelf specific log option gelf-address.
  130. func ValidateLogOpt(cfg map[string]string) error {
  131. for key, val := range cfg {
  132. switch key {
  133. case "gelf-address":
  134. case "tag":
  135. case "labels":
  136. case "env":
  137. case "gelf-compression-level":
  138. i, err := strconv.Atoi(val)
  139. if err != nil || i < flate.DefaultCompression || i > flate.BestCompression {
  140. return fmt.Errorf("unknown value %q for log opt %q for gelf log driver", val, key)
  141. }
  142. case "gelf-compression-type":
  143. switch val {
  144. case "gzip", "zlib", "none":
  145. default:
  146. return fmt.Errorf("unknown value %q for log opt %q for gelf log driver", val, key)
  147. }
  148. default:
  149. return fmt.Errorf("unknown log opt %q for gelf log driver", key)
  150. }
  151. }
  152. if _, err := parseAddress(cfg["gelf-address"]); err != nil {
  153. return err
  154. }
  155. return nil
  156. }
  157. func parseAddress(address string) (string, error) {
  158. if address == "" {
  159. return "", nil
  160. }
  161. if !urlutil.IsTransportURL(address) {
  162. return "", fmt.Errorf("gelf-address should be in form proto://address, got %v", address)
  163. }
  164. url, err := url.Parse(address)
  165. if err != nil {
  166. return "", err
  167. }
  168. // we support only udp
  169. if url.Scheme != "udp" {
  170. return "", fmt.Errorf("gelf: endpoint needs to be UDP")
  171. }
  172. // get host and port
  173. if _, _, err = net.SplitHostPort(url.Host); err != nil {
  174. return "", fmt.Errorf("gelf: please provide gelf-address as udp://host:port")
  175. }
  176. return url.Host, nil
  177. }