gelf.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  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. info logger.Info
  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(info logger.Info) (logger.Logger, error) {
  37. // parse gelf address
  38. address, err := parseAddress(info.Config["gelf-address"])
  39. if err != nil {
  40. return nil, err
  41. }
  42. // collect extra data for GELF message
  43. hostname, err := info.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(info, loggerutils.DefaultTemplate)
  49. if err != nil {
  50. return nil, err
  51. }
  52. extra := map[string]interface{}{
  53. "_container_id": info.ContainerID,
  54. "_container_name": info.Name(),
  55. "_image_id": info.ContainerImageID,
  56. "_image_name": info.ContainerImageName,
  57. "_command": info.Command(),
  58. "_tag": tag,
  59. "_created": info.ContainerCreated,
  60. }
  61. extraAttrs, err := info.ExtraAttributes(func(key string) string {
  62. if key[0] == '_' {
  63. return key
  64. }
  65. return "_" + key
  66. })
  67. if err != nil {
  68. return nil, err
  69. }
  70. for k, v := range extraAttrs {
  71. extra[k] = v
  72. }
  73. rawExtra, err := json.Marshal(extra)
  74. if err != nil {
  75. return nil, err
  76. }
  77. // create new gelfWriter
  78. gelfWriter, err := gelf.NewWriter(address)
  79. if err != nil {
  80. return nil, fmt.Errorf("gelf: cannot connect to GELF endpoint: %s %v", address, err)
  81. }
  82. if v, ok := info.Config["gelf-compression-type"]; ok {
  83. switch v {
  84. case "gzip":
  85. gelfWriter.CompressionType = gelf.CompressGzip
  86. case "zlib":
  87. gelfWriter.CompressionType = gelf.CompressZlib
  88. case "none":
  89. gelfWriter.CompressionType = gelf.CompressNone
  90. default:
  91. return nil, fmt.Errorf("gelf: invalid compression type %q", v)
  92. }
  93. }
  94. if v, ok := info.Config["gelf-compression-level"]; ok {
  95. val, err := strconv.Atoi(v)
  96. if err != nil {
  97. return nil, fmt.Errorf("gelf: invalid compression level %s, err %v", v, err)
  98. }
  99. gelfWriter.CompressionLevel = val
  100. }
  101. return &gelfLogger{
  102. writer: gelfWriter,
  103. info: info,
  104. hostname: hostname,
  105. rawExtra: rawExtra,
  106. }, nil
  107. }
  108. func (s *gelfLogger) Log(msg *logger.Message) error {
  109. level := gelf.LOG_INFO
  110. if msg.Source == "stderr" {
  111. level = gelf.LOG_ERR
  112. }
  113. m := gelf.Message{
  114. Version: "1.1",
  115. Host: s.hostname,
  116. Short: string(msg.Line),
  117. TimeUnix: float64(msg.Timestamp.UnixNano()/int64(time.Millisecond)) / 1000.0,
  118. Level: level,
  119. RawExtra: s.rawExtra,
  120. }
  121. logger.PutMessage(msg)
  122. if err := s.writer.WriteMessage(&m); err != nil {
  123. return fmt.Errorf("gelf: cannot send GELF message: %v", err)
  124. }
  125. return nil
  126. }
  127. func (s *gelfLogger) Close() error {
  128. return s.writer.Close()
  129. }
  130. func (s *gelfLogger) Name() string {
  131. return name
  132. }
  133. // ValidateLogOpt looks for gelf specific log option gelf-address.
  134. func ValidateLogOpt(cfg map[string]string) error {
  135. for key, val := range cfg {
  136. switch key {
  137. case "gelf-address":
  138. case "tag":
  139. case "labels":
  140. case "env":
  141. case "env-regex":
  142. case "gelf-compression-level":
  143. i, err := strconv.Atoi(val)
  144. if err != nil || i < flate.DefaultCompression || i > flate.BestCompression {
  145. return fmt.Errorf("unknown value %q for log opt %q for gelf log driver", val, key)
  146. }
  147. case "gelf-compression-type":
  148. switch val {
  149. case "gzip", "zlib", "none":
  150. default:
  151. return fmt.Errorf("unknown value %q for log opt %q for gelf log driver", val, key)
  152. }
  153. default:
  154. return fmt.Errorf("unknown log opt %q for gelf log driver", key)
  155. }
  156. }
  157. _, err := parseAddress(cfg["gelf-address"])
  158. return err
  159. }
  160. func parseAddress(address string) (string, error) {
  161. if address == "" {
  162. return "", nil
  163. }
  164. if !urlutil.IsTransportURL(address) {
  165. return "", fmt.Errorf("gelf-address should be in form proto://address, got %v", address)
  166. }
  167. url, err := url.Parse(address)
  168. if err != nil {
  169. return "", err
  170. }
  171. // we support only udp
  172. if url.Scheme != "udp" {
  173. return "", fmt.Errorf("gelf: endpoint needs to be UDP")
  174. }
  175. // get host and port
  176. if _, _, err = net.SplitHostPort(url.Host); err != nil {
  177. return "", fmt.Errorf("gelf: please provide gelf-address as udp://host:port")
  178. }
  179. return url.Host, nil
  180. }