syslog.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. package syslogacquisition
  2. import (
  3. "fmt"
  4. "net"
  5. "strconv"
  6. "time"
  7. "github.com/crowdsecurity/crowdsec/pkg/acquisition/configuration"
  8. syslogserver "github.com/crowdsecurity/crowdsec/pkg/acquisition/modules/syslog/internal"
  9. leaky "github.com/crowdsecurity/crowdsec/pkg/leakybucket"
  10. "github.com/crowdsecurity/crowdsec/pkg/types"
  11. "github.com/influxdata/go-syslog/v3/rfc3164"
  12. "github.com/influxdata/go-syslog/v3/rfc5424"
  13. "github.com/pkg/errors"
  14. "github.com/prometheus/client_golang/prometheus"
  15. log "github.com/sirupsen/logrus"
  16. "gopkg.in/tomb.v2"
  17. "gopkg.in/yaml.v2"
  18. )
  19. type SyslogConfiguration struct {
  20. Proto string `yaml:"protocol,omitempty"`
  21. Port int `yaml:"listen_port,omitempty"`
  22. Addr string `yaml:"listen_addr,omitempty"`
  23. MaxMessageLen int `yaml:"max_message_len,omitempty"`
  24. configuration.DataSourceCommonCfg `yaml:",inline"`
  25. }
  26. type SyslogSource struct {
  27. config SyslogConfiguration
  28. logger *log.Entry
  29. server *syslogserver.SyslogServer
  30. serverTomb *tomb.Tomb
  31. }
  32. var linesReceived = prometheus.NewCounterVec(
  33. prometheus.CounterOpts{
  34. Name: "cs_syslogsource_hits_total",
  35. Help: "Total lines that were received.",
  36. },
  37. []string{"source"})
  38. var linesParsed = prometheus.NewCounterVec(
  39. prometheus.CounterOpts{
  40. Name: "cs_syslogsource_parsed_total",
  41. Help: "Total lines that were successfully parsed",
  42. },
  43. []string{"source", "type"})
  44. func (s *SyslogSource) GetName() string {
  45. return "syslog"
  46. }
  47. func (s *SyslogSource) GetMode() string {
  48. return s.config.Mode
  49. }
  50. func (s *SyslogSource) Dump() interface{} {
  51. return s
  52. }
  53. func (s *SyslogSource) CanRun() error {
  54. return nil
  55. }
  56. func (s *SyslogSource) GetMetrics() []prometheus.Collector {
  57. return []prometheus.Collector{linesReceived, linesParsed}
  58. }
  59. func (s *SyslogSource) GetAggregMetrics() []prometheus.Collector {
  60. return []prometheus.Collector{linesReceived, linesParsed}
  61. }
  62. func (s *SyslogSource) ConfigureByDSN(dsn string, labelType string, logger *log.Entry) error {
  63. return fmt.Errorf("syslog datasource does not support one shot acquisition")
  64. }
  65. func (s *SyslogSource) OneShotAcquisition(out chan types.Event, t *tomb.Tomb) error {
  66. return fmt.Errorf("syslog datasource does not support one shot acquisition")
  67. }
  68. func validatePort(port int) bool {
  69. return port > 0 && port <= 65535
  70. }
  71. func validateAddr(addr string) bool {
  72. return net.ParseIP(addr) != nil
  73. }
  74. func (s *SyslogSource) Configure(yamlConfig []byte, logger *log.Entry) error {
  75. s.logger = logger
  76. s.logger.Infof("Starting syslog datasource configuration")
  77. syslogConfig := SyslogConfiguration{}
  78. syslogConfig.Mode = configuration.TAIL_MODE
  79. err := yaml.UnmarshalStrict(yamlConfig, &syslogConfig)
  80. if err != nil {
  81. return errors.Wrap(err, "Cannot parse syslog configuration")
  82. }
  83. if syslogConfig.Addr == "" {
  84. syslogConfig.Addr = "127.0.0.1" //do we want a usable or secure default ?
  85. }
  86. if syslogConfig.Port == 0 {
  87. syslogConfig.Port = 514
  88. }
  89. if syslogConfig.MaxMessageLen == 0 {
  90. syslogConfig.MaxMessageLen = 2048
  91. }
  92. if !validatePort(syslogConfig.Port) {
  93. return fmt.Errorf("invalid port %d", syslogConfig.Port)
  94. }
  95. if !validateAddr(syslogConfig.Addr) {
  96. return fmt.Errorf("invalid listen IP %s", syslogConfig.Addr)
  97. }
  98. s.config = syslogConfig
  99. return nil
  100. }
  101. func (s *SyslogSource) StreamingAcquisition(out chan types.Event, t *tomb.Tomb) error {
  102. c := make(chan syslogserver.SyslogMessage)
  103. s.server = &syslogserver.SyslogServer{Logger: s.logger.WithField("syslog", "internal"), MaxMessageLen: s.config.MaxMessageLen}
  104. s.server.SetChannel(c)
  105. err := s.server.Listen(s.config.Addr, s.config.Port)
  106. if err != nil {
  107. return errors.Wrap(err, "could not start syslog server")
  108. }
  109. s.serverTomb = s.server.StartServer()
  110. t.Go(func() error {
  111. defer types.CatchPanic("crowdsec/acquis/syslog/live")
  112. return s.handleSyslogMsg(out, t, c)
  113. })
  114. return nil
  115. }
  116. func (s *SyslogSource) buildLogFromSyslog(ts *time.Time, hostname *string,
  117. appname *string, pid *string, msg *string) (string, error) {
  118. ret := ""
  119. if ts != nil {
  120. ret += ts.Format("Jan 2 15:04:05")
  121. } else {
  122. ret += time.Now().Format("Jan 2 15:04:05")
  123. }
  124. if hostname != nil {
  125. ret += " " + *hostname
  126. } else {
  127. ret += " unknownhost"
  128. }
  129. if appname != nil {
  130. ret += " " + *appname
  131. } else {
  132. return "", errors.Errorf("missing appname field in syslog message")
  133. }
  134. if pid != nil {
  135. /*
  136. !!! ugly hack !!!
  137. Due to a bug in the syslog parser we use (https://github.com/influxdata/go-syslog/issues/31),
  138. the ProcID field will contain garbage if the message as a ] anywhere in it.
  139. Assume that a correctly formated ProcID only contains number, and if this is not the case, set it to an arbitrary value
  140. */
  141. _, err := strconv.Atoi(*pid)
  142. if err != nil {
  143. ret += "[1]: "
  144. } else {
  145. ret += "[" + *pid + "]: "
  146. }
  147. } else {
  148. ret += ": "
  149. }
  150. if msg != nil {
  151. ret += *msg
  152. } else {
  153. return "", errors.Errorf("missing message field in syslog message")
  154. }
  155. return ret, nil
  156. }
  157. func (s *SyslogSource) handleSyslogMsg(out chan types.Event, t *tomb.Tomb, c chan syslogserver.SyslogMessage) error {
  158. for {
  159. select {
  160. case <-t.Dying():
  161. s.logger.Info("Syslog datasource is dying")
  162. s.serverTomb.Kill(nil)
  163. return s.serverTomb.Wait()
  164. case <-s.serverTomb.Dying():
  165. s.logger.Info("Syslog server is dying, exiting")
  166. return nil
  167. case <-s.serverTomb.Dead():
  168. s.logger.Info("Syslog server has exited")
  169. return nil
  170. case syslogLine := <-c:
  171. var line string
  172. var ts time.Time
  173. logger := s.logger.WithField("client", syslogLine.Client)
  174. linesReceived.With(prometheus.Labels{"source": syslogLine.Client}).Inc()
  175. p := rfc5424.NewParser()
  176. m, err := p.Parse(syslogLine.Message)
  177. if err != nil {
  178. logger.Debugf("could not parse message as RFC5424, falling back to RFC3164 : %s", err)
  179. p = rfc3164.NewParser(rfc3164.WithYear(rfc3164.CurrentYear{}))
  180. m, err = p.Parse(syslogLine.Message)
  181. if err != nil {
  182. logger.Errorf("could not parse message: %s", err)
  183. continue
  184. }
  185. msg := m.(*rfc3164.SyslogMessage)
  186. line, err = s.buildLogFromSyslog(msg.Timestamp, msg.Hostname, msg.Appname, msg.ProcID, msg.Message)
  187. if err != nil {
  188. logger.Error(err)
  189. continue
  190. }
  191. linesParsed.With(prometheus.Labels{"source": syslogLine.Client,
  192. "type": "RFC3164"}).Inc()
  193. } else {
  194. msg := m.(*rfc5424.SyslogMessage)
  195. line, err = s.buildLogFromSyslog(msg.Timestamp, msg.Hostname, msg.Appname, msg.ProcID, msg.Message)
  196. if err != nil {
  197. logger.Error(err)
  198. continue
  199. }
  200. linesParsed.With(prometheus.Labels{"source": syslogLine.Client,
  201. "type": "RFC5424"}).Inc()
  202. }
  203. l := types.Line{}
  204. l.Raw = line
  205. l.Module = s.GetName()
  206. l.Labels = s.config.Labels
  207. l.Time = ts
  208. l.Src = syslogLine.Client
  209. l.Process = true
  210. out <- types.Event{Line: l, Process: true, Type: types.LOG, ExpectMode: leaky.LIVE}
  211. }
  212. }
  213. }