2020-05-15 09:39:16 +00:00
|
|
|
package leakybucket
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/antonmedv/expr"
|
|
|
|
"github.com/antonmedv/expr/vm"
|
|
|
|
|
|
|
|
"github.com/crowdsecurity/crowdsec/pkg/exprhelpers"
|
|
|
|
"github.com/crowdsecurity/crowdsec/pkg/types"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Uniq creates three new functions that share the same initialisation and the same scope.
|
|
|
|
// They are triggered respectively:
|
|
|
|
// on pour
|
|
|
|
// on overflow
|
|
|
|
// on leak
|
|
|
|
|
|
|
|
type Uniq struct {
|
|
|
|
DistinctCompiled *vm.Program
|
|
|
|
}
|
|
|
|
|
2020-11-30 09:37:17 +00:00
|
|
|
func (u *Uniq) OnBucketPour(bucketFactory *BucketFactory) func(types.Event, *Leaky) *types.Event {
|
|
|
|
return func(msg types.Event, leaky *Leaky) *types.Event {
|
2020-05-15 09:39:16 +00:00
|
|
|
element, err := getElement(msg, u.DistinctCompiled)
|
|
|
|
if err != nil {
|
2020-11-30 09:37:17 +00:00
|
|
|
leaky.logger.Errorf("Uniq filter exec failed : %v", err)
|
2020-05-15 09:39:16 +00:00
|
|
|
return &msg
|
|
|
|
}
|
2020-11-30 09:37:17 +00:00
|
|
|
leaky.logger.Tracef("Uniq '%s' -> '%s'", bucketFactory.Distinct, element)
|
|
|
|
for _, evt := range leaky.Queue.GetQueue() {
|
2020-05-15 09:39:16 +00:00
|
|
|
if val, err := getElement(evt, u.DistinctCompiled); err == nil && val == element {
|
2020-11-30 09:37:17 +00:00
|
|
|
leaky.logger.Debugf("Uniq(%s) : ko, discard event", element)
|
2020-05-15 09:39:16 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
2020-11-30 09:37:17 +00:00
|
|
|
leaky.logger.Errorf("Uniq filter exec failed : %v", err)
|
2020-05-15 09:39:16 +00:00
|
|
|
}
|
|
|
|
}
|
2020-11-30 09:37:17 +00:00
|
|
|
leaky.logger.Debugf("Uniq(%s) : ok", element)
|
2020-05-15 09:39:16 +00:00
|
|
|
return &msg
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-30 09:37:17 +00:00
|
|
|
func (u *Uniq) OnBucketOverflow(bucketFactory *BucketFactory) func(*Leaky, types.RuntimeAlert, *Queue) (types.RuntimeAlert, *Queue) {
|
|
|
|
return func(leaky *Leaky, alert types.RuntimeAlert, queue *Queue) (types.RuntimeAlert, *Queue) {
|
|
|
|
return alert, queue
|
2020-05-15 09:39:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-30 09:37:17 +00:00
|
|
|
func (u *Uniq) OnBucketInit(bucketFactory *BucketFactory) error {
|
2020-05-15 09:39:16 +00:00
|
|
|
var err error
|
|
|
|
|
2020-11-30 09:37:17 +00:00
|
|
|
u.DistinctCompiled, err = expr.Compile(bucketFactory.Distinct, expr.Env(exprhelpers.GetExprEnv(map[string]interface{}{"evt": &types.Event{}})))
|
2020-05-15 09:39:16 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// getElement computes a string from an event and a filter
|
|
|
|
func getElement(msg types.Event, cFilter *vm.Program) (string, error) {
|
|
|
|
el, err := expr.Run(cFilter, exprhelpers.GetExprEnv(map[string]interface{}{"evt": &msg}))
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
element, ok := el.(string)
|
|
|
|
if !ok {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return element, nil
|
|
|
|
}
|