Files
cgrates/engine/filters_test.go
2023-02-02 17:49:05 +01:00

1531 lines
46 KiB
Go

/*
Real-time Online/Offline Charging System (OCS) for Telecom & ISP environments
Copyright (C) ITsysCOM GmbH
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>
*/
package engine
import (
"reflect"
"strings"
"testing"
"time"
"github.com/cgrates/cgrates/config"
"github.com/cgrates/cgrates/utils"
)
func TestFilterPassString(t *testing.T) {
cd := &CallDescriptor{
Category: "call",
Tenant: "cgrates.org",
Subject: "dan",
Destination: "+4986517174963",
TimeStart: time.Date(2013, time.October, 7, 14, 50, 0, 0, time.UTC),
TimeEnd: time.Date(2013, time.October, 7, 14, 52, 12, 0, time.UTC),
DurationIndex: 132 * time.Second,
ExtraFields: map[string]string{"navigation": "off"},
}
rf := &FilterRule{Type: utils.MetaString,
Element: "~Category", Values: []string{"call"}}
if passes, err := rf.passString(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaString,
Element: "~Category", Values: []string{"cal"}}
if passes, err := rf.passString(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Filter passes")
}
//not
rf = &FilterRule{Type: utils.MetaNotString,
Element: "~Category", Values: []string{"call"}}
if passes, err := rf.Pass(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Filter passes")
}
rf = &FilterRule{Type: utils.MetaNotString,
Element: "~Category", Values: []string{"cal"}}
if passes, err := rf.Pass(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
}
func TestFilterPassEmpty(t *testing.T) {
cd := &CallDescriptor{
Category: "",
Tenant: "cgrates.org",
Subject: "dan",
Destination: "+4986517174963",
TimeStart: time.Date(2013, time.October, 7, 14, 50, 0, 0, time.UTC),
TimeEnd: time.Date(2013, time.October, 7, 14, 52, 12, 0, time.UTC),
DurationIndex: 132 * time.Second,
ExtraFields: map[string]string{"navigation": "off"},
}
rf := &FilterRule{Type: utils.MetaEmpty, Element: "~Category", Values: []string{}}
if passes, err := rf.passEmpty(cd); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaEmpty, Element: "~ExtraFields", Values: []string{}}
if passes, err := rf.passEmpty(cd); err != nil {
t.Error(err)
} else if passes {
t.Error("Filter passes")
}
cd.ExtraFields = map[string]string{}
rf = &FilterRule{Type: utils.MetaEmpty, Element: "~ExtraFields", Values: []string{}}
if passes, err := rf.passEmpty(cd); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
//not
rf = &FilterRule{Type: utils.MetaNotEmpty, Element: "~Category", Values: []string{}}
if passes, err := rf.Pass(cd, []utils.DataProvider{}); err != nil {
t.Error(err)
} else if passes {
t.Error("Filter passes")
}
}
func TestFilterPassExists(t *testing.T) {
cd := &CallDescriptor{
Category: "",
Tenant: "cgrates.org",
Subject: "dan",
Destination: "+4986517174963",
TimeStart: time.Date(2013, time.October, 7, 14, 50, 0, 0, time.UTC),
TimeEnd: time.Date(2013, time.October, 7, 14, 52, 12, 0, time.UTC),
DurationIndex: 132 * time.Second,
ExtraFields: map[string]string{"navigation": "off"},
}
rf := &FilterRule{Type: utils.MetaExists, Element: "~Category", Values: []string{}}
if passes, err := rf.passExists(cd); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaExists, Element: "~ExtraFields1", Values: []string{}}
if passes, err := rf.passExists(cd); err != nil {
t.Error(err)
} else if passes {
t.Error("Filter passes")
}
cd.ExtraFields = map[string]string{}
rf = &FilterRule{Type: utils.MetaExists, Element: "~ExtraFields", Values: []string{}}
if passes, err := rf.passExists(cd); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
//not
rf = &FilterRule{Type: utils.MetaNotExists, Element: "~Category1", Values: []string{}}
if passes, err := rf.Pass(cd, []utils.DataProvider{}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
}
func TestFilterPassStringPrefix(t *testing.T) {
cd := &CallDescriptor{
Category: "call",
Tenant: "cgrates.org",
Subject: "dan",
Destination: "+4986517174963",
TimeStart: time.Date(2013, time.October, 7, 14, 50, 0, 0, time.UTC),
TimeEnd: time.Date(2013, time.October, 7, 14, 52, 12, 0, time.UTC),
DurationIndex: 132 * time.Second,
ExtraFields: map[string]string{"navigation": "off"},
}
rf := &FilterRule{Type: utils.MetaPrefix, Element: "~Category", Values: []string{"call"}}
if passes, err := rf.passStringPrefix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaPrefix, Element: "~Category", Values: []string{"premium"}}
if passes, err := rf.passStringPrefix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passes filter")
}
rf = &FilterRule{Type: utils.MetaPrefix, Element: "~Destination", Values: []string{"+49"}}
if passes, err := rf.passStringPrefix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaPrefix, Element: "~Destination", Values: []string{"+499"}}
if passes, err := rf.passStringPrefix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passes filter")
}
rf = &FilterRule{Type: utils.MetaPrefix, Element: "~navigation", Values: []string{"off"}}
if passes, err := rf.passStringPrefix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaPrefix, Element: "~nonexisting", Values: []string{"off"}}
if passing, err := rf.passStringPrefix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passing {
t.Error("Passes filter")
}
//not
rf = &FilterRule{Type: utils.MetaNotPrefix, Element: "~Category", Values: []string{"premium"}}
if passes, err := rf.Pass(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
}
func TestFilterPassStringSuffix(t *testing.T) {
cd := &CallDescriptor{
Category: "call",
Tenant: "cgrates.org",
Subject: "dan",
Destination: "+4986517174963",
TimeStart: time.Date(2013, time.October, 7, 14, 50, 0, 0, time.UTC),
TimeEnd: time.Date(2013, time.October, 7, 14, 52, 12, 0, time.UTC),
DurationIndex: 132 * time.Second,
ExtraFields: map[string]string{"navigation": "off"},
}
rf := &FilterRule{Type: utils.MetaSuffix, Element: "~Category", Values: []string{"call"}}
if passes, err := rf.passStringSuffix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaSuffix, Element: "~Category", Values: []string{"premium"}}
if passes, err := rf.passStringSuffix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passes filter")
}
rf = &FilterRule{Type: utils.MetaSuffix, Element: "~Destination", Values: []string{"963"}}
if passes, err := rf.passStringSuffix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaSuffix, Element: "~Destination", Values: []string{"4966"}}
if passes, err := rf.passStringSuffix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passes filter")
}
rf = &FilterRule{Type: utils.MetaSuffix, Element: "~navigation", Values: []string{"off"}}
if passes, err := rf.passStringSuffix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passes filter")
}
rf = &FilterRule{Type: utils.MetaSuffix, Element: "~nonexisting", Values: []string{"off"}}
if passing, err := rf.passStringSuffix(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passing {
t.Error("Passes filter")
}
//not
rf = &FilterRule{Type: utils.MetaNotSuffix, Element: "~Destination", Values: []string{"963"}}
if passes, err := rf.Pass(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passes filter")
}
}
func TestFilterPassRSRFields(t *testing.T) {
cd := &CallDescriptor{
Category: "call",
Tenant: "cgrates.org",
Subject: "dan",
Destination: "+4986517174963",
TimeStart: time.Date(2013, time.October, 7, 14, 50, 0, 0, time.UTC),
TimeEnd: time.Date(2013, time.October, 7, 14, 52, 12, 0, time.UTC),
DurationIndex: 132 * time.Second,
ExtraFields: map[string]string{"navigation": "off"},
}
rf, err := NewFilterRule(utils.MetaRSR, "", []string{"~Tenant(~^cgr.*\\.org$)"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passRSR([]utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passing")
}
rf, err = NewFilterRule(utils.MetaRSR, "", []string{"~navigation(on)"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passRSR([]utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passing")
}
rf, err = NewFilterRule(utils.MetaRSR, "", []string{"~navigation(off)"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passRSR([]utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passing")
}
//not
rf, err = NewFilterRule(utils.MetaNotRSR, "", []string{"~navigation(off)"})
if err != nil {
t.Error(err)
}
if passes, err := rf.Pass(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passing")
}
}
func TestFilterPassDestinations(t *testing.T) {
Cache.Set(utils.CacheReverseDestinations, "+49",
[]string{"DE", "EU_LANDLINE"}, nil, true, "")
cd := &CallDescriptor{
Category: "call",
Tenant: "cgrates.org",
Subject: "dan",
Destination: "+4986517174963",
TimeStart: time.Date(2013, time.October, 7, 14, 50, 0, 0, time.UTC),
TimeEnd: time.Date(2013, time.October, 7, 14, 52, 12, 0, time.UTC),
DurationIndex: 132 * time.Second,
ExtraFields: map[string]string{"navigation": "off"},
}
rf, err := NewFilterRule(utils.MetaDestinations, "~Destination", []string{"DE"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passDestinations(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if !passes {
t.Error("Not passing")
}
rf, err = NewFilterRule(utils.MetaDestinations, "~Destination", []string{"RO"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passDestinations(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passing")
}
//not
rf, err = NewFilterRule(utils.MetaNotDestinations, "~Destination", []string{"DE"})
if err != nil {
t.Error(err)
}
if passes, err := rf.Pass(cd, []utils.DataProvider{cd}); err != nil {
t.Error(err)
} else if passes {
t.Error("Passing")
}
}
func TestFilterPassGreaterThan(t *testing.T) {
rf, err := NewFilterRule(utils.MetaLessThan, "~ASR", []string{"40"})
if err != nil {
t.Error(err)
}
ev := utils.MapStorage{}
ev.Set([]string{"ASR"}, 20)
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("not passing")
}
ev = utils.MapStorage{}
ev.Set([]string{"ASR"}, 40)
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if passes {
t.Error("equal should not be passing")
}
rf, err = NewFilterRule(utils.MetaLessOrEqual, "~ASR", []string{"40"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("not passing")
}
rf, err = NewFilterRule(utils.MetaGreaterOrEqual, "~ASR", []string{"40"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("not passing")
}
rf, err = NewFilterRule(utils.MetaGreaterOrEqual, "~ASR", []string{"35.5"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("not passing")
}
ev = utils.MapStorage{}
ev.Set([]string{"ASR"}, 20)
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if passes {
t.Error("should not pass")
}
rf, err = NewFilterRule(utils.MetaGreaterOrEqual, "~ACD", []string{"1m50s"})
if err != nil {
t.Error(err)
}
ev = utils.MapStorage{}
ev.Set([]string{"ACD"}, time.Duration(2*time.Minute))
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("not pass")
}
// Second
ev = utils.MapStorage{}
ev.Set([]string{"ASR"}, time.Duration(20*time.Second))
rf, err = NewFilterRule("*gte", "~ASR", []string{"10s"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("passing")
}
rf, err = NewFilterRule("*gte", "~ASR", []string{"10"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("passing")
}
ev = utils.MapStorage{}
ev.Set([]string{"ASR"}, float64(20*time.Second))
rf, err = NewFilterRule("*gte", "~ASR", []string{"10s"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("passing")
}
//Here converter will be consider part of path and will get error : NOT_FOUND
ev = utils.MapStorage{}
ev.Set([]string{"ASR"}, 20)
rf, err = NewFilterRule("*gte", "~ASR{*duration_seconds}", []string{"10"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if passes {
t.Error("passing")
}
//Here converter will be consider part of path and will get error : NOT_FOUND
ev = utils.MapStorage{}
ev.Set([]string{"ASR"}, 20)
rf, err = NewFilterRule("*gte", "~ASR{*duration_seconds}", []string{"10s"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passGreaterThan(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if passes {
t.Error("passing")
}
}
func TestFilterpassEqualTo(t *testing.T) {
rf, err := NewFilterRule(utils.MetaEqual, "~ASR", []string{"40"})
if err != nil {
t.Error(err)
}
ev := utils.MapStorage{}
ev.Set([]string{"ASR"}, 40.0)
if passes, err := rf.passEqualTo(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("not passing")
}
ev = utils.MapStorage{}
ev.Set([]string{"ASR"}, 39)
if passes, err := rf.passEqualTo(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if passes {
t.Error("equal should not be passing")
}
rf, err = NewFilterRule(utils.MetaNotEqual, "~ASR", []string{"40"})
if err != nil {
t.Error(err)
}
if passes, err := rf.Pass(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("not passing", passes)
}
ev = utils.MapStorage{}
ev.Set([]string{"ASR"}, "string1")
rf, err = NewFilterRule(utils.MetaEqual, "~ASR", []string{"string1"})
if err != nil {
t.Error(err)
}
if passes, err := rf.passEqualTo(ev, []utils.DataProvider{ev}); err != nil {
t.Error(err)
} else if !passes {
t.Error("not passing")
}
}
func TestFilterNewRequestFilter(t *testing.T) {
rf, err := NewFilterRule(utils.MetaString, "~MetaString", []string{"String"})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf := &FilterRule{Type: utils.MetaString, Element: "~MetaString", Values: []string{"String"}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaEmpty, "~MetaEmpty", []string{})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaEmpty, Element: "~MetaEmpty", Values: []string{}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaExists, "~MetaExists", []string{})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaExists, Element: "~MetaExists", Values: []string{}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaPrefix, "~MetaPrefix", []string{"stringPrefix"})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaPrefix, Element: "~MetaPrefix", Values: []string{"stringPrefix"}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaSuffix, "~MetaSuffix", []string{"stringSuffix"})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaSuffix, Element: "~MetaSuffix", Values: []string{"stringSuffix"}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaTimings, "~MetaTimings", []string{""})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaTimings, Element: "~MetaTimings", Values: []string{""}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaDestinations, "~MetaDestinations", []string{""})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaDestinations, Element: "~MetaDestinations", Values: []string{""}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaLessThan, "~MetaLessThan", []string{"20"})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaLessThan, Element: "~MetaLessThan", Values: []string{"20"}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaLessOrEqual, "~MetaLessOrEqual", []string{"20"})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaLessOrEqual, Element: "~MetaLessOrEqual", Values: []string{"20"}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaGreaterThan, "~MetaGreaterThan", []string{"20"})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaGreaterThan, Element: "~MetaGreaterThan", Values: []string{"20"}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
rf, err = NewFilterRule(utils.MetaGreaterOrEqual, "~MetaGreaterOrEqual", []string{"20"})
if err != nil {
t.Errorf("Error: %+v", err)
}
erf = &FilterRule{Type: utils.MetaGreaterOrEqual, Element: "~MetaGreaterOrEqual", Values: []string{"20"}, negative: utils.BoolPointer(false)}
if !reflect.DeepEqual(erf, rf) {
t.Errorf("Expecting: %+v, received: %+v", erf, rf)
}
}
func TestInlineFilterPassFiltersForEvent(t *testing.T) {
cfg, _ := config.NewDefaultCGRConfig()
data := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dmFilterPass := NewDataManager(data, config.CgrConfig().CacheCfg(), nil)
filterS := FilterS{
cfg: cfg,
dm: dmFilterPass,
}
failEvent := map[string]interface{}{
"Account": "1001",
}
passEvent := map[string]interface{}{
"Account": "1007",
}
fEv := utils.MapStorage{}
fEv.Set([]string{utils.MetaReq}, failEvent)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*string:~*req.Account:1007:error"}, fEv); err != nil {
t.Error(err)
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*string:~*req.Account:1007"}, fEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
pEv := utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*string:~*req.Account:1007"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
//not
if pass, err := filterS.Pass("cgrates.org",
[]string{"*notstring:~*req.Account:1007"}, pEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
failEvent = map[string]interface{}{
"Account": "2001",
}
passEvent = map[string]interface{}{
"Account": "1007",
}
fEv = utils.MapStorage{}
fEv.Set([]string{utils.MetaReq}, failEvent)
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*prefix:~*req.Account:10"}, fEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*prefix:~*req.Account:10"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*suffix:~*req.Account:07"}, fEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*suffix:~*req.Account:07"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
//not
if pass, err := filterS.Pass("cgrates.org",
[]string{"*notsuffix:~*req.Account:07"}, pEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
failEvent = map[string]interface{}{
"Tenant": "anotherTenant.org",
}
passEvent = map[string]interface{}{
"Tenant": "cgrates.org",
}
fEv = utils.MapStorage{}
fEv.Set([]string{utils.MetaReq}, failEvent)
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~Tenant(~^cgr.*\\.org$)"}, fEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.Tenant(~^cgr.*\\.org$)"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
//not
if pass, err := filterS.Pass("cgrates.org",
[]string{"*notrsr::~*req.Tenant(~^cgr.*\\.org$)"}, pEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
Cache.Set(utils.CacheReverseDestinations, "+49",
[]string{"DE", "EU_LANDLINE"}, nil, true, "")
failEvent = map[string]interface{}{
utils.Destination: "+5086517174963",
}
passEvent = map[string]interface{}{
utils.Destination: "+4986517174963",
}
fEv = utils.MapStorage{}
fEv.Set([]string{utils.MetaReq}, failEvent)
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*destinations:~*req.Destination:EU"}, fEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*destinations:~*req.Destination:EU_LANDLINE"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
failEvent = map[string]interface{}{
utils.Weight: 10,
}
passEvent = map[string]interface{}{
utils.Weight: 20,
}
fEv = utils.MapStorage{}
fEv.Set([]string{utils.MetaReq}, failEvent)
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*gte:~*req.Weight:20"}, fEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*gte:~*req.Weight:10"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
failEvent = map[string]interface{}{
"EmptyString": "nonEmpty",
"EmptySlice": []string{""},
"EmptyMap": map[string]string{"": ""},
"EmptyPtr": &struct{}{},
"EmptyPtr2": &struct{}{},
"EmptyPtrSlice": &[]string{""},
"EmptyPtrMap": &map[string]string{"": ""},
}
var testnil *struct{}
testnil = nil
passEvent = map[string]interface{}{
"EmptyString": "",
"EmptySlice": []string{},
"EmptyMap": map[string]string{},
"EmptyPtr": testnil,
"EmptyPtr2": nil,
"EmptyPtrSlice": &[]string{},
"EmptyPtrMap": &map[string]string{},
}
fEv = utils.MapStorage{}
fEv.Set([]string{utils.MetaReq}, failEvent)
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent)
for key := range failEvent {
if pass, err := filterS.Pass("cgrates.org", []string{"*empty:~*req." + key + ":"},
fEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("For %s expecting: %+v, received: %+v", key, false, pass)
}
if pass, err := filterS.Pass("cgrates.org", []string{"*empty:~*req." + key + ":"},
pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("For %s expecting: %+v, received: %+v", key, true, pass)
}
}
if pass, err := filterS.Pass("cgrates.org", []string{"*exists:~*req.NewKey:"},
fEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("For NewKey expecting: %+v, received: %+v", false, pass)
}
if pass, err := filterS.Pass("cgrates.org", []string{"*notexists:~*req.NewKey:"},
fEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("For NewKey expecting: %+v, received: %+v", true, pass)
}
}
func TestPassRsr(t *testing.T) {
cfg, _ := config.NewDefaultCGRConfig()
data := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dmFilterPass := NewDataManager(data, config.CgrConfig().CacheCfg(), nil)
filterS := FilterS{
cfg: cfg,
dm: dmFilterPass,
}
passEvent1 := map[string]interface{}{
"8": "0045664",
}
pEv1 := utils.MapStorage{}
pEv1.Set([]string{utils.MetaReq}, passEvent1)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.8(~^004)"}, pEv1); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
passEvent1 = map[string]interface{}{
"5": "0",
}
pEv1 = utils.MapStorage{}
pEv1.Set([]string{utils.MetaReq}, passEvent1)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.5(~^0$)"}, pEv1); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
}
func TestPassRsr2(t *testing.T) {
cfg, _ := config.NewDefaultCGRConfig()
dmFilterPass := NewDataManager(NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items), config.CgrConfig().CacheCfg(), nil)
filterS := FilterS{
cfg: cfg,
dm: dmFilterPass,
}
pEv1 := utils.MapStorage{
utils.MetaReq: utils.MapStorage{
"8": "00545664",
},
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.8(~^004)"}, pEv1); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
}
func TestPassFiltersForEventWithEmptyFilter(t *testing.T) {
cfg, _ := config.NewDefaultCGRConfig()
data := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dmFilterPass := NewDataManager(data, config.CgrConfig().CacheCfg(), nil)
filterS := FilterS{
cfg: cfg,
dm: dmFilterPass,
}
passEvent1 := map[string]interface{}{
utils.Tenant: "cgrates.org",
utils.Account: "1010",
utils.Destination: "+49",
utils.Weight: 10,
}
passEvent2 := map[string]interface{}{
utils.Tenant: "itsyscom.com",
utils.Account: "dan",
utils.Destination: "+4986517174963",
utils.Weight: 20,
}
pEv1 := utils.MapStorage{}
pEv1.Set([]string{utils.MetaReq}, passEvent1)
pEv2 := utils.MapStorage{}
pEv2.Set([]string{utils.MetaReq}, passEvent2)
if pass, err := filterS.Pass("cgrates.org",
[]string{}, pEv1); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
if pass, err := filterS.Pass("itsyscom.com",
[]string{}, pEv2); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
ev := map[string]interface{}{
"Test": "MultipleCharacter",
}
pEv := utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, ev)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.Test(~^\\w{30,})"}, pEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
ev = map[string]interface{}{
"Test": "MultipleCharacter123456789MoreThan30Character",
}
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, ev)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.Test(~^\\w{30,})"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
ev = map[string]interface{}{
"Test": map[string]interface{}{
"Test2": "MultipleCharacter",
},
}
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, ev)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.Test.Test2(~^\\w{30,})"}, pEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
ev = map[string]interface{}{
"Test": map[string]interface{}{
"Test2": "MultipleCharacter123456789MoreThan30Character",
},
}
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, ev)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.Test.Test2(~^\\w{30,})"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", false, pass)
}
ev = map[string]interface{}{
utils.Account: "1003",
utils.Subject: "1003",
utils.Destination: "1002",
utils.SetupTime: time.Date(2017, 12, 1, 14, 25, 0, 0, time.UTC),
utils.Usage: "1m20s",
}
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, ev)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*string:~*req.Account:1003", "*prefix:~*req.Destination:10",
"*suffix:~*req.Subject:03", "*rsr::~*req.Destination(1002)"},
pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: %+v, received: %+v", true, pass)
}
}
func TestPassFilterMaxCost(t *testing.T) {
cfg, _ := config.NewDefaultCGRConfig()
data := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dmFilterPass := NewDataManager(data, config.CgrConfig().CacheCfg(), nil)
filterS := FilterS{
cfg: cfg,
dm: dmFilterPass,
}
//check with max usage -1 should fail
passEvent1 := map[string]interface{}{
"MaxUsage": time.Duration(-1),
}
pEv := utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent1)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.MaxUsage{*duration_nanoseconds}(>0)"}, pEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: false , received: %+v", pass)
}
//check with max usage 0 should fail
passEvent2 := map[string]interface{}{
"MaxUsage": time.Duration(0),
}
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent2)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.MaxUsage{*duration_nanoseconds}(>0)"}, pEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: false, received: %+v", pass)
}
//check with max usage 123 should pass
passEvent3 := map[string]interface{}{
"MaxUsage": time.Duration(123),
}
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent3)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.MaxUsage{*duration_nanoseconds}(>0)"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: true, received: %+v", pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*gt:~*req.MaxUsage:0"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: true, received: %+v", pass)
}
}
func TestPassFilterMissingField(t *testing.T) {
cfg, _ := config.NewDefaultCGRConfig()
data := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dmFilterPass := NewDataManager(data, config.CgrConfig().CacheCfg(), nil)
filterS := FilterS{
cfg: cfg,
dm: dmFilterPass,
}
passEvent1 := map[string]interface{}{
"test": "call",
}
pEv := utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent1)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.Category(^$)"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: true , received: %+v", pass)
}
passEvent2 := map[string]interface{}{
"Category": "",
}
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent2)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~Category(^$)"}, pEv); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: true , received: %+v", pass)
}
passEvent3 := map[string]interface{}{
"Category": "call",
}
pEv = utils.MapStorage{}
pEv.Set([]string{utils.MetaReq}, passEvent3)
if pass, err := filterS.Pass("cgrates.org",
[]string{"*rsr::~*req.Category(^$)"}, pEv); err != nil {
t.Errorf(err.Error())
} else if pass {
t.Errorf("Expecting: false , received: %+v", pass)
}
}
func TestEventCostFilter(t *testing.T) {
cfg, _ := config.NewDefaultCGRConfig()
data := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dmFilterPass := NewDataManager(data, config.CgrConfig().CacheCfg(), nil)
filterS := FilterS{
cfg: cfg,
dm: dmFilterPass,
}
cd := &EventCost{
Cost: utils.Float64Pointer(0.264933),
CGRID: "d8534def2b7067f4f5ad4f7ec7bbcc94bb46111a",
Rates: ChargedRates{
"3db483c": RateGroups{
{
Value: 0.1574,
RateUnit: 60000000000,
RateIncrement: 30000000000,
GroupIntervalStart: 0,
},
{
Value: 0.1574,
RateUnit: 60000000000,
RateIncrement: 1000000000,
GroupIntervalStart: 30000000000,
},
},
},
RunID: "*default",
Usage: utils.DurationPointer(101 * time.Second),
Rating: Rating{
"7f3d423": &RatingUnit{
MaxCost: 40,
RatesID: "3db483c",
TimingID: "128e970",
ConnectFee: 0,
RoundingMethod: "*up",
MaxCostStrategy: "*disconnect",
RatingFiltersID: "f8e95f2",
RoundingDecimals: 4,
},
},
Charges: []*ChargingInterval{
{
RatingID: "7f3d423",
Increments: []*ChargingIncrement{
{
Cost: 0.0787,
Usage: 30000000000,
AccountingID: "fee8a3a",
CompressFactor: 1,
},
},
CompressFactor: 1,
},
{
RatingID: "7f3d423",
Increments: []*ChargingIncrement{
{
Cost: 0.002623,
Usage: 1000000000,
AccountingID: "3463957",
CompressFactor: 71,
},
},
CompressFactor: 1,
},
},
Timings: ChargedTimings{
"128e970": &ChargedTiming{
StartTime: "00:00:00",
},
},
StartTime: time.Date(2019, 12, 06, 11, 57, 32, 0, time.UTC),
Accounting: Accounting{
"3463957": &BalanceCharge{
Units: 0.002623,
RatingID: "",
AccountID: "cgrates.org:1001",
BalanceUUID: "154419f2-45e0-4629-a203-06034ccb493f",
ExtraChargeID: "",
},
"fee8a3a": &BalanceCharge{
Units: 0.0787,
RatingID: "",
AccountID: "cgrates.org:1001",
BalanceUUID: "154419f2-45e0-4629-a203-06034ccb493f",
ExtraChargeID: "",
},
},
RatingFilters: RatingFilters{
"f8e95f2": RatingMatchedFilters{
"Subject": "*out:cgrates.org:mo_call_UK_Mobile_O2_GBRCN:*any",
"RatingPlanID": "RP_MO_CALL_44800",
"DestinationID": "DST_44800",
"DestinationPrefix": "44800",
},
},
AccountSummary: &AccountSummary{
ID: "234189200129930",
Tenant: "cgrates.org",
Disabled: false,
AllowNegative: false,
BalanceSummaries: BalanceSummaries{
&BalanceSummary{
ID: "MOBILE_DATA",
Type: "*data",
UUID: "08a05723-5849-41b9-b6a9-8ee362539280",
Value: 3221225472,
Disabled: false,
},
&BalanceSummary{
ID: "MOBILE_SMS",
Type: "*sms",
UUID: "06a87f20-3774-4eeb-826e-a79c5f175fd3",
Value: 247,
Disabled: false,
},
&BalanceSummary{
ID: "MOBILE_VOICE",
Type: "*voice",
UUID: "4ad16621-6e22-4e35-958e-5e1ff93ad7b7",
Value: 14270000000000,
Disabled: false,
},
&BalanceSummary{
ID: "MONETARY_POSTPAID",
Type: "*monetary",
UUID: "154419f2-45e0-4629-a203-06034ccb493f",
Value: 50,
Disabled: false,
},
},
},
}
cd.initCache()
cgrDp := utils.MapStorage{utils.MetaEC: cd}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*string:~*ec.Charges[0].Increments[0].Accounting.Balance.Value:50"}, cgrDp); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: true , received: %+v", pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*string:~*ec.Charges[0].Increments[0].Accounting.AccountID:cgrates.org:1001"}, cgrDp); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: true , received: %+v", pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*string:~*ec.Charges[0].Rating.Rates[0].Value:0.1574"}, cgrDp); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: true , received: %+v", pass)
}
if pass, err := filterS.Pass("cgrates.org",
[]string{"*string:~*ec.Charges[0].Increments[0].Accounting.Balance.ID:MONETARY_POSTPAID"}, cgrDp); err != nil {
t.Errorf(err.Error())
} else if !pass {
t.Errorf("Expecting: true , received: %+v", pass)
}
}
func TestComputeThresholdIndexes(t *testing.T) {
cfg, err := config.NewDefaultCGRConfig()
if err != nil {
t.Error(err)
}
db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dm := NewDataManager(db, cfg.CacheCfg(), nil)
fltr := &Filter{
Tenant: "cgrates.org",
ID: "FLTR_2",
Rules: []*FilterRule{
{
Type: utils.MetaPrefix,
Element: utils.DynamicDataPrefix + utils.MetaReq + utils.NestingSep + utils.Account,
Values: []string{"1001"},
},
},
}
if err := dm.SetFilter(fltr); err != nil {
t.Error(err)
}
thd1 := &ThresholdProfile{
Tenant: "cgrates.org",
ID: "TH_1",
FilterIDs: []string{"FLTR_2"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 35, 0, 0, time.UTC),
ExpiryTime: time.Date(2014, 7, 14, 14, 35, 0, 0, time.UTC),
},
}
if err := dm.SetThresholdProfile(thd1, true); err != nil {
t.Error(err)
}
thd2 := &ThresholdProfile{
Tenant: "cgrates.org",
ID: "TH_2",
FilterIDs: []string{"FLTR_2", utils.META_NONE},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 35, 0, 0, time.UTC),
},
}
if err := dm.SetThresholdProfile(thd2, true); err != nil {
t.Error(err)
}
thIDs := []string{"TH_1", "TH_2"}
if _, err := ComputeThresholdIndexes(dm, "cgrates.org", &thIDs, utils.NonTransactional); err != nil {
t.Error(err)
}
if _, err := ComputeThresholdIndexes(dm, "cgrates.org", nil, "ID"); err != nil {
t.Error(err)
}
}
func TestComputeChargerIndexes(t *testing.T) {
cfg, err := config.NewDefaultCGRConfig()
if err != nil {
t.Error(err)
}
db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dm := NewDataManager(db, cfg.CacheCfg(), nil)
fltr := &Filter{
Tenant: "cgrates.org",
ID: "Filter3",
Rules: []*FilterRule{
{
Element: "Destination",
Type: utils.MetaString,
Values: []string{"10", "20"},
},
},
}
if err := dm.SetFilter(fltr); err != nil {
t.Error(err)
}
chP := &ChargerProfile{
Tenant: "cgrates.org",
ID: "CHRG_1",
FilterIDs: []string{"Filter3", "*string:~*req.Account:1001", utils.META_NONE},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
ExpiryTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
AttributeIDs: []string{"ATTR_1"},
Weight: 20,
}
if err := dm.SetChargerProfile(chP, true); err != nil {
t.Error(err)
}
chIDs := []string{"CHRG_1"}
if _, err := ComputeChargerIndexes(dm, "cgrates.org", &chIDs, utils.NonTransactional); err != nil {
t.Error(err)
}
if _, err := ComputeChargerIndexes(dm, "cgrates.org", nil, "ID"); err != nil {
t.Error(err)
}
}
func TestComputeResourceIndexes(t *testing.T) {
cfg, err := config.NewDefaultCGRConfig()
if err != nil {
t.Error(err)
}
db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dm := NewDataManager(db, cfg.CacheCfg(), nil)
fltr := &Filter{
Tenant: "cgrates.org",
ID: "FLTR_TH_Resource",
Rules: []*FilterRule{
{
Type: "*lt",
Element: "~*resources.ResTest.TotalUsage",
Values: []string{"2.0"},
},
},
}
if err := dm.SetFilter(fltr); err != nil {
t.Error(err)
}
rs := &ResourceProfile{
Tenant: "cgrates.org",
ID: "RES_GR_TEST",
FilterIDs: []string{"FLTR_TH_Resource", utils.META_NONE},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
ExpiryTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
UsageTTL: time.Duration(-1),
Limit: 2,
AllocationMessage: "Account1Channels",
Weight: 20,
ThresholdIDs: []string{utils.META_NONE},
}
if err := dm.SetResourceProfile(rs, true); err != nil {
t.Error(err)
}
chIDs := []string{"RES_GR_TEST"}
if _, err := ComputeResourceIndexes(dm, "cgrates.org", &chIDs, utils.NonTransactional); err != nil {
t.Error(err)
}
if _, err := ComputeResourceIndexes(dm, "cgrates.org", nil, "ID"); err != nil {
t.Error(err)
}
}
func TestComputeSupplierIndexes(t *testing.T) {
cfg, err := config.NewDefaultCGRConfig()
if err != nil {
t.Error(err)
}
db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dm := NewDataManager(db, cfg.CacheCfg(), nil)
fltr := &Filter{
Tenant: "cgrates.org",
ID: "FLTR_SUPP_1",
Rules: []*FilterRule{
{
Type: utils.MetaString,
Element: "~*req.Supplier",
Values: []string{"SupplierProfile1"},
},
},
}
if err := dm.SetFilter(fltr); err != nil {
t.Error(err)
}
spp := &SupplierProfile{
Tenant: "cgrates.org",
ID: "SPL_2",
Sorting: utils.MetaLC,
FilterIDs: []string{"FLTR_SUPP_1"},
Suppliers: []*Supplier{
{
ID: "SPL1",
FilterIDs: []string{"FLTR_1"},
AccountIDs: []string{"accc"},
Weight: 20,
Blocker: false,
},
},
Weight: 10,
}
if err := dm.SetSupplierProfile(spp, true); err != nil {
t.Error(err)
}
chIDs := []string{"SPL_2"}
if _, err := ComputeSupplierIndexes(dm, "cgrates.org", &chIDs, utils.NonTransactional); err != nil {
t.Error(err)
}
if _, err := ComputeSupplierIndexes(dm, "cgrates.org", nil, "ID"); err != nil {
t.Error(err)
}
}
func TestRemoveItemFromIndexRP(t *testing.T) {
cfg, err := config.NewDefaultCGRConfig()
if err != nil {
t.Error(err)
}
db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dm := NewDataManager(db, cfg.CacheCfg(), nil)
fltr := &Filter{
Tenant: "cgrates.org",
ID: "FLTR_1",
Rules: []*FilterRule{
{
Type: utils.MetaString,
Element: "~*req.Account",
Values: []string{"1001"},
},
},
}
if err := dm.SetFilter(fltr); err != nil {
t.Error(err)
}
fltr.Compile()
rs := &ResourceProfile{
Tenant: "cgrates.org",
ID: "RES_GR_TEST",
FilterIDs: []string{"FLTR_1"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
ExpiryTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
UsageTTL: time.Duration(-1),
Limit: 2,
AllocationMessage: "Account1Channels",
Weight: 20,
ThresholdIDs: []string{utils.META_NONE},
}
if err := dm.SetResourceProfile(rs, true); err != nil {
t.Error(err)
}
fltInd := NewFilterIndexer(dm, utils.ResourceProfilesPrefix, rs.Tenant)
if err := fltInd.RemoveItemFromIndex("cgrates.org", rs.ID, []string{}); err != nil {
t.Error(err)
}
}
func TestRemoveItemFromIndexCHP(t *testing.T) {
cfg, err := config.NewDefaultCGRConfig()
if err != nil {
t.Error(err)
}
db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
dm := NewDataManager(db, cfg.CacheCfg(), nil)
fltr := &Filter{
Tenant: "cgrates.org",
ID: "FLTR_CP_2",
Rules: []*FilterRule{
{
Type: utils.MetaString,
Element: "~*req.Charger",
Values: []string{"ChargerProfile2"},
},
}}
if err := fltr.Compile(); err != nil {
t.Error(err)
}
if err := dm.SetFilter(fltr); err != nil {
t.Error(err)
}
chp := &ChargerProfile{
Tenant: "cgrates.org",
ID: "Raw",
FilterIDs: []string{"FLTR_CP_2"},
RunID: utils.MetaRaw,
AttributeIDs: []string{"*constant:*req.RequestType:*none"},
Weight: 0,
}
if err := dm.SetChargerProfile(chp, false); err != nil {
t.Error(err)
}
fltInd := NewFilterIndexer(dm, utils.ChargerProfilePrefix, chp.Tenant)
if err := fltInd.RemoveItemFromIndex("cgrates.org", chp.ID, []string{}); err != nil {
t.Error(err)
}
if err := dm.RemoveFilter("cgrates.org", fltr.ID, utils.NonTransactional); err != nil {
t.Error(err)
}
if err := fltInd.RemoveItemFromIndex("cgrates.org", chp.ID, []string{}); err == nil {
t.Error(err)
}
if err := dm.SetChargerProfile(chp, true); err == nil || !strings.HasPrefix(err.Error(), "broken reference to filter:") {
t.Error(err)
}
}