123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109 |
- package database
- import (
- "context"
- "fmt"
- "os"
- "time"
- "github.com/crowdsecurity/crowdsec/pkg/csconfig"
- "github.com/crowdsecurity/crowdsec/pkg/database/ent"
- "github.com/crowdsecurity/crowdsec/pkg/types"
- "github.com/go-co-op/gocron"
- _ "github.com/go-sql-driver/mysql"
- _ "github.com/lib/pq"
- _ "github.com/mattn/go-sqlite3"
- "github.com/pkg/errors"
- log "github.com/sirupsen/logrus"
- )
- type Client struct {
- Ent *ent.Client
- CTX context.Context
- Log *log.Logger
- CanFlush bool
- }
- func NewClient(config *csconfig.DatabaseCfg) (*Client, error) {
- var client *ent.Client
- var err error
- if config == nil {
- return &Client{}, fmt.Errorf("DB config is empty")
- }
- /*The logger that will be used by db operations*/
- clog := log.New()
- if err := types.ConfigureLogger(clog); err != nil {
- return nil, errors.Wrap(err, "while configuring db logger")
- }
- if config.LogLevel != nil {
- clog.SetLevel(*config.LogLevel)
- }
- entLogger := clog.WithField("context", "ent")
- entOpt := ent.Log(entLogger.Debug)
- switch config.Type {
- case "sqlite":
- /*if it's the first startup, we want to touch and chmod file*/
- if _, err := os.Stat(config.DbPath); os.IsNotExist(err) {
- f, err := os.OpenFile(config.DbPath, os.O_CREATE|os.O_RDWR, 0600)
- if err != nil {
- return &Client{}, errors.Wrapf(err, "failed to create SQLite database file %q", config.DbPath)
- }
- if err := f.Close(); err != nil {
- return &Client{}, errors.Wrapf(err, "failed to create SQLite database file %q", config.DbPath)
- }
- } else { /*ensure file perms*/
- if err := os.Chmod(config.DbPath, 0660); err != nil {
- return &Client{}, fmt.Errorf("unable to set perms on %s: %v", config.DbPath, err)
- }
- }
- client, err = ent.Open("sqlite3", fmt.Sprintf("file:%s?_busy_timeout=100000&_fk=1", config.DbPath), entOpt)
- if err != nil {
- return &Client{}, fmt.Errorf("failed opening connection to sqlite: %v", err)
- }
- case "mysql":
- client, err = ent.Open("mysql", fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?parseTime=True", config.User, config.Password, config.Host, config.Port, config.DbName), entOpt)
- if err != nil {
- return &Client{}, fmt.Errorf("failed opening connection to mysql: %v", err)
- }
- case "postgres", "postgresql":
- client, err = ent.Open("postgres", fmt.Sprintf("host=%s port=%d user=%s dbname=%s password=%s sslmode=%s", config.Host, config.Port, config.User, config.DbName, config.Password, config.Sslmode), entOpt)
- if err != nil {
- return &Client{}, fmt.Errorf("failed opening connection to postgres: %v", err)
- }
- default:
- return &Client{}, fmt.Errorf("unknown database type")
- }
- if config.LogLevel != nil && *config.LogLevel >= log.DebugLevel {
- clog.Debugf("Enabling request debug")
- client = client.Debug()
- }
- if err = client.Schema.Create(context.Background()); err != nil {
- return nil, fmt.Errorf("failed creating schema resources: %v", err)
- }
- return &Client{Ent: client, CTX: context.Background(), Log: clog, CanFlush: true}, nil
- }
- func (c *Client) StartFlushScheduler(config *csconfig.FlushDBCfg) (*gocron.Scheduler, error) {
- maxItems := 0
- maxAge := ""
- if config.MaxItems != nil && *config.MaxItems <= 0 {
- return nil, fmt.Errorf("max_items can't be zero or negative number")
- }
- if config.MaxItems != nil {
- maxItems = *config.MaxItems
- }
- if config.MaxAge != nil && *config.MaxAge != "" {
- maxAge = *config.MaxAge
- }
- // Init & Start cronjob every minute
- scheduler := gocron.NewScheduler(time.UTC)
- job, _ := scheduler.Every(1).Minute().Do(c.FlushAlerts, maxAge, maxItems)
- job.SingletonMode()
- scheduler.StartAsync()
- return scheduler, nil
- }
|