/* 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 */ package engine import ( "fmt" "reflect" "strings" "testing" "time" "github.com/cgrates/birpc" "github.com/cgrates/birpc/context" "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]any{ "Account": "1001", } passEvent := map[string]any{ "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]any{ "Account": "2001", } passEvent = map[string]any{ "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]any{ "Tenant": "anotherTenant.org", } passEvent = map[string]any{ "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]any{ utils.Destination: "+5086517174963", } passEvent = map[string]any{ 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]any{ utils.Weight: 10, } passEvent = map[string]any{ 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]any{ "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]any{ "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]any{ "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]any{ "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]any{ utils.Tenant: "cgrates.org", utils.Account: "1010", utils.Destination: "+49", utils.Weight: 10, } passEvent2 := map[string]any{ 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]any{ "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]any{ "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]any{ "Test": map[string]any{ "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]any{ "Test": map[string]any{ "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]any{ 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]any{ "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]any{ "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]any{ "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]any{ "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]any{ "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]any{ "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{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) } expIndexes := map[string]utils.StringMap{ "*prefix:~*req.Account:1001": {"TH_1": true}, "*none:*any:*any": {"TH_2": true}, } if fltrIndexer, err := ComputeThresholdIndexes(dm, "cgrates.org", nil, "ID"); err != nil { t.Error(err) } else if !reflect.DeepEqual(fltrIndexer.indexes, expIndexes) { t.Errorf("Expected %v,Received %v", utils.ToJSON(expIndexes), utils.ToJSON(fltrIndexer.indexes)) } } 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: "~*req.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) } expIndexes := map[string]utils.StringMap{ "*string:~*req.Account:1001": {"CHRG_1": true}, "*string:~*req.Destination:10": {"CHRG_1": true}, "*string:~*req.Destination:20": {"CHRG_1": true}, "*none:*any:*any": {"CHRG_1": true}, } if fltrIndexer, err := ComputeChargerIndexes(dm, "cgrates.org", nil, "ID"); err != nil { t.Error(err) } else if !reflect.DeepEqual(fltrIndexer.indexes, expIndexes) { t.Errorf("Expected %v,Received %v", utils.ToJSON(expIndexes), utils.ToJSON(fltrIndexer.indexes)) } } 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_RES_1", Rules: []*FilterRule{ { Type: utils.MetaString, Element: "~*req.Resources", Values: []string{"ResourceProfile2"}, }, }, } if err := dm.SetFilter(fltr); err != nil { t.Error(err) } rs := &ResourceProfile{ Tenant: "cgrates.org", ID: "RES_GR_TEST", FilterIDs: []string{"FLTR_RES_1", 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) } expIndexes := map[string]utils.StringMap{ "*none:*any:*any": {"RES_GR_TEST": true}, "*string:~*req.Resources:ResourceProfile2": {"RES_GR_TEST": true}, } if fltrIndexer, err := ComputeResourceIndexes(dm, "cgrates.org", nil, "ID"); err != nil { t.Error(err) } else if !reflect.DeepEqual(fltrIndexer.indexes, expIndexes) { t.Errorf("Expected %v,Received %v", utils.ToJSON(fltrIndexer.indexes), utils.ToJSON(expIndexes)) } } 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) } expIndexes := map[string]utils.StringMap{ "*string:~*req.Supplier:SupplierProfile1": {"SPL_2": true}, } if fltrIndexes, err := ComputeSupplierIndexes(dm, "cgrates.org", nil, "ID"); err != nil { t.Error(err) } else if !reflect.DeepEqual(expIndexes, fltrIndexes.indexes) { t.Errorf("Expected %v,Received %v", utils.ToJSON(expIndexes), utils.ToJSON(fltrIndexes.indexes)) } } 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) } if err := dm.RemoveResourceProfile("cgrates.org", rs.ID, utils.NonTransactional, true); err != nil { t.Error(err) } } func TestRemoveItemFromIndexCHP(t *testing.T) { Cache.Clear(nil) 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) } } func TestComputeStatIndexes(t *testing.T) { cfg, err := config.NewDefaultCGRConfig() if err != nil { t.Error(err) } Cache.Clear(nil) db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) fltr := &Filter{ Tenant: config.CgrConfig().GeneralCfg().DefaultTenant, ID: "FLTR_STATS_1", Rules: []*FilterRule{ { Type: utils.MetaString, Element: "~*req.Stats", Values: []string{"StatQueueProfile1"}, }, }, } if err := fltr.Compile(); err != nil { t.Error(err) } if err := dm.SetFilter(fltr); err != nil { t.Error(err) } sq := &StatQueueProfile{ Tenant: "cgrates.org", ID: "TEST_PROFILE2", FilterIDs: []string{"FLTR_STATS_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), }, QueueLength: 10, TTL: time.Duration(10) * time.Second, Metrics: []*MetricWithFilters{ { MetricID: "*sum", }, { MetricID: utils.MetaACD, }, }, ThresholdIDs: []string{"Val1", "Val2"}, Blocker: true, Stored: true, Weight: 20, MinItems: 1, } if err := dm.SetStatQueueProfile(sq, true); err != nil { t.Error(err) } stIDs := []string{"TEST_PROFILE2"} if _, err := ComputeStatIndexes(dm, "cgrates.org", &stIDs, ""); err != nil { t.Error(err) } expIndexes := map[string]utils.StringMap{ "*string:~*req.Stats:StatQueueProfile1": {"TEST_PROFILE2": true}, } if fltrIndexer, err := ComputeStatIndexes(dm, "cgrates.org", nil, "ID"); err != nil { t.Error(err) } else if !reflect.DeepEqual(expIndexes, fltrIndexer.indexes) { t.Errorf("Expected %v,Received %v", utils.ToJSON(expIndexes), utils.ToJSON(fltrIndexer)) } } func TestComputeAttributeIndexes(t *testing.T) { cfg, err := config.NewDefaultCGRConfig() if err != nil { t.Error(err) } Cache.Clear(nil) db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) fltrAttr1 := &Filter{ Tenant: config.CgrConfig().GeneralCfg().DefaultTenant, ID: "FLTR_ATTR_1", Rules: []*FilterRule{ { Type: utils.MetaString, Element: "~*req.Attribute", Values: []string{"AttributeProfile1"}, }, { Type: utils.MetaGreaterOrEqual, Element: "~*req.UsageInterval", Values: []string{(1 * time.Second).String()}, }, { Type: utils.MetaGreaterOrEqual, Element: "~*req." + utils.Weight, Values: []string{"9.0"}, }, { Type: utils.MetaPrefix, Element: utils.DynamicDataPrefix + utils.MetaReq + utils.NestingSep + utils.Account, Values: []string{"1001"}, }, }, } if err := fltrAttr1.Compile(); err != nil { t.Error(err) } if err := dm.SetFilter(fltrAttr1); err != nil { t.Error(err) } ap := &AttributeProfile{ Tenant: "cgrates.org", ID: "AttrPrf1", FilterIDs: []string{"FLTR_ATTR_1"}, ActivationInterval: &utils.ActivationInterval{ ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC), }, Contexts: []string{"con1"}, Attributes: []*Attribute{ { Path: utils.MetaReq + utils.NestingSep + "FN1", Value: config.NewRSRParsersMustCompile("Val1", true, utils.INFIELD_SEP), }, }, Weight: 20, } if err := dm.SetAttributeProfile(ap, true); err != nil { t.Error(err) } ids := []string{"AttrPrf1"} if _, err := ComputeAttributeIndexes(dm, "cgrates.org", "con1", &ids, utils.NonTransactional); err != nil { t.Error(err) } expIndexes := map[string]utils.StringMap{ "*string:~*req.Attribute:AttributeProfile1": { "AttrPrf1": true, }, "*prefix:~*req.Account:1001": { "AttrPrf1": true, }, } if fltrIndexer, err := ComputeAttributeIndexes(dm, "cgrates.org", "con1", nil, "ID"); err != nil { t.Error(err) } else if !reflect.DeepEqual(expIndexes, fltrIndexer.indexes) { t.Errorf("Expected %v , Receveid %v", utils.ToJSON(expIndexes), utils.ToJSON(fltrIndexer.indexes)) } } func TestComputeDispatcherIndexes(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() Cache.Clear(nil) db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) fltr := &Filter{ Tenant: "cgrates.org", ID: "DSP_FLT", Rules: []*FilterRule{ { Element: utils.DynamicDataPrefix + utils.MetaReq + utils.NestingSep + utils.Account, Type: utils.MetaString, Values: []string{"2009"}, }, }} if err := dm.SetFilter(fltr); err != nil { t.Error(err) } dpP := &DispatcherProfile{ Tenant: "cgrates.org", ID: "Dsp1", FilterIDs: []string{"DSP_FLT"}, ActivationInterval: &utils.ActivationInterval{ ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC), }, Strategy: utils.MetaFirst, Weight: 20, Subsystems: []string{utils.MetaAttributes, utils.MetaSessionS}, } if err := dm.SetDispatcherProfile(dpP, true); err != nil { t.Error(err) } ids := []string{"Dsp1"} if _, err := ComputeDispatcherIndexes(dm, "cgrates.org", utils.MetaAttributes, &ids, utils.NonTransactional); err != nil { t.Error(err) } expIndexes := map[string]utils.StringMap{ "*string:~*req.Account:2009": { "Dsp1": true, }, } if fltrIndexer, err := ComputeDispatcherIndexes(dm, "cgrates.org", utils.MetaSessionS, nil, "ID"); err != nil { t.Error(err) } else if !reflect.DeepEqual(expIndexes, fltrIndexer.indexes) { t.Errorf("Expected %v,Received %v", utils.ToJSON(expIndexes), utils.ToJSON(fltrIndexer.indexes)) } } func TestRemoveItemFromIndexSQP(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) fltr := &Filter{ Tenant: "cgrates.org", ID: "FLTR_1", Rules: []*FilterRule{ { Element: utils.DynamicDataPrefix + utils.MetaReq + utils.NestingSep + utils.Account, Type: utils.MetaString, Values: []string{"1001"}, }, }, 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), }, } if err := dm.SetFilter(fltr); err != nil { t.Error(err) } sqs := &StatQueueProfile{ Tenant: "cgrates.org", ID: "DistinctMetricProfile", QueueLength: 10, FilterIDs: []string{"FLTR_1"}, TTL: time.Duration(10) * time.Second, Metrics: []*MetricWithFilters{ { MetricID: utils.MetaDDC, }, }, ThresholdIDs: []string{utils.META_NONE}, Stored: true, Weight: 20, } if err := dm.SetStatQueueProfile(sqs, true); err != nil { t.Error(err) } fltrIndexer := NewFilterIndexer(dm, utils.StatQueueProfilePrefix, sqs.Tenant) if err := fltrIndexer.RemoveItemFromIndex(sqs.Tenant, sqs.ID, []string{}); err != nil { t.Error(err) } } func TestRemoveItemFromIndexSPP(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() 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_ACNT_1001", FilterIDs: []string{"FLTR_SUPP_1"}, ActivationInterval: &utils.ActivationInterval{ ActivationTime: time.Date(2017, 11, 27, 0, 0, 0, 0, time.UTC), }, Sorting: utils.MetaWeight, SortingParameters: []string{}, Suppliers: []*Supplier{ { ID: "supplier1", Weight: 10, }, }, Weight: 20, } if err := dm.SetSupplierProfile(spp, true); err != nil { t.Error(err) } fltrIndexer := NewFilterIndexer(dm, utils.SupplierProfilePrefix, spp.Tenant) if err := fltrIndexer.RemoveItemFromIndex(spp.Tenant, spp.ID, []string{}); err != nil { t.Error(err) } } func TestRemoveItemFromIndexDP(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) fltr := &Filter{ Tenant: "cgrates.org", ID: "DSP_FLT", Rules: []*FilterRule{ { Element: utils.DynamicDataPrefix + utils.MetaReq + utils.NestingSep + utils.Account, Type: utils.MetaString, Values: []string{"2009"}, }, }, } if err := dm.SetFilter(fltr); err != nil { t.Error(err) } dpp := &DispatcherProfile{ Tenant: "cgrates.org", ID: "DSP_Test1", FilterIDs: []string{"DSP_FLT"}, Strategy: utils.MetaFirst, Subsystems: []string{utils.MetaAttributes, utils.MetaSessionS}, Weight: 20, } if err := dm.SetDispatcherProfile(dpp, true); err != nil { t.Error(err) } fltrIndexer := NewFilterIndexer(dm, utils.DispatcherProfilePrefix, dpp.Tenant) if err := fltrIndexer.RemoveItemFromIndex(dpp.Tenant, dpp.ID, []string{}); err != nil { t.Error(err) } } func TestUpdateFilterIndexes(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) oldFlt := &Filter{ Tenant: "cgrates.org", ID: "FLTR_1", Rules: []*FilterRule{ { Type: utils.MetaString, Element: utils.DynamicDataPrefix + utils.MetaReq + utils.NestingSep + utils.Account, Values: []string{"1001", "1002"}, }, { Type: "*prefix", Element: utils.DynamicDataPrefix + utils.MetaReq + utils.NestingSep + utils.Destination, Values: []string{"10", "20"}, }, { Type: "*rsr", Element: "", Values: []string{"~*req.Subject(~^1.*1$)", "~*req.Destination(1002)"}, }, }, } if err := dm.SetFilter(oldFlt); err != nil { t.Error(err) } chg := &ChargerProfile{ Tenant: "cgrates.org", ID: "CPP_3", FilterIDs: []string{"FLTR_1"}, ActivationInterval: &utils.ActivationInterval{ ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC), }, RunID: "*rated", AttributeIDs: []string{"ATTR_1"}, Weight: 20, } if err := dm.SetChargerProfile(chg, true); err != nil { t.Error(err) } thP := &ThresholdProfile{ Tenant: "cgrates.org", ID: "THD_ACNT_1001", FilterIDs: []string{"FLTR_1"}, ActivationInterval: &utils.ActivationInterval{ ActivationTime: time.Date(2014, 7, 29, 15, 0, 0, 0, time.UTC), }, MaxHits: 1, MinHits: 1, MinSleep: time.Duration(1 * time.Second), Weight: 10.0, ActionIDs: []string{"ACT_LOG_WARNING"}, Async: true, } if err := dm.SetThresholdProfile(thP, true); err != nil { t.Error(err) } rcf := &ResourceProfile{ Tenant: "cgrates.org", ID: "RCFG1", FilterIDs: []string{"FLTR_1"}, ActivationInterval: &utils.ActivationInterval{ ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC), }, UsageTTL: time.Duration(10) * time.Microsecond, Limit: 10, AllocationMessage: "MessageAllocation", Blocker: true, Stored: true, Weight: 20, } if err := dm.SetResourceProfile(rcf, true); err != nil { t.Error(err) } supp := &SupplierProfile{ Tenant: "cgrates.org", ID: "SPL_DESTINATION", FilterIDs: []string{"FLTR_1"}, Sorting: utils.MetaLC, Suppliers: []*Supplier{ { ID: "local", RatingPlanIDs: []string{"RP_LOCAL"}, Weight: 10, }, { ID: "mobile", RatingPlanIDs: []string{"RP_MOBILE"}, FilterIDs: []string{"*destinations:~*req.Destination:DST_MOBILE"}, Weight: 10, }, }, Weight: 100, } if err := dm.SetSupplierProfile(supp, true); err != nil { t.Error(err) } stat := &StatQueueProfile{ Tenant: "cgrates.org", ID: "TEST_PROFILE1", 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), }, QueueLength: 10, TTL: time.Duration(10) * time.Second, Metrics: []*MetricWithFilters{ { MetricID: utils.MetaACD, }, }, ThresholdIDs: []string{"THD_ACNT_1001"}, Blocker: true, Stored: true, Weight: 20, MinItems: 1, } if err := dm.SetStatQueueProfile(stat, true); err != nil { t.Error(err) } attr := &AttributeProfile{ Tenant: "cgrates.org", ID: "ApierTest", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"FLTR_1"}, Attributes: []*Attribute{ { Path: utils.MetaReq + utils.NestingSep + utils.Subject, Value: config.NewRSRParsersMustCompile("1011", true, utils.INFIELD_SEP), }, }, Weight: 20, } if err := dm.SetAttributeProfile(attr, true); err != nil { t.Error(err) } dpp := &DispatcherProfile{ Tenant: "cgrates.org", ID: "DSP_Test1", FilterIDs: []string{"FLTR_1"}, Strategy: utils.MetaFirst, Subsystems: []string{utils.MetaAttributes, utils.MetaSessionS}, Weight: 20, } if err := dm.SetDispatcherProfile(dpp, true); err != nil { t.Error(err) } newFlt := &Filter{ Tenant: "cgrates.org", ID: "FLTR_1", Rules: []*FilterRule{ { Type: utils.MetaString, Element: utils.DynamicDataPrefix + utils.MetaReq + utils.NestingSep + utils.Account, Values: []string{"1001"}, }, { Type: utils.MetaString, Element: "~*req.Subject", Values: []string{"1001"}, }, { Type: utils.MetaRSR, Element: utils.EmptyString, Values: []string{"~*req.Tenant(~^cgr.*\\.org$)"}, }, }, } if err := UpdateFilterIndexes(dm, "cgrates.org", oldFlt, newFlt); err != nil { t.Error(err) } } func TestFilterSPass11(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() defer func() { Cache.Clear(nil) }() cfg.FilterSCfg().ResourceSConns = []string{utils.ConcatenatedKey(utils.MetaInternal, utils.MetaResources)} cfg.FilterSCfg().StatSConns = []string{utils.ConcatenatedKey(utils.MetaInternal, utils.MetaStatS)} db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) acc := &Account{ ID: "cgrates.org:1001", BalanceMap: map[string]Balances{ utils.VOICE: { &Balance{Value: 20 * float64(time.Second), DestinationIDs: utils.NewStringMap("DST1"), Weight: 10}, &Balance{Value: 100 * float64(time.Second), DestinationIDs: utils.NewStringMap("DST2"), Weight: 20}, }}, } rsr := &Resource{ Tenant: "cgrates.org", ID: "RL1", Usages: map[string]*ResourceUsage{ "RU1": { ID: "RU1", ExpiryTime: time.Date(2014, 7, 3, 13, 43, 0, 0, time.UTC), Units: 2, }, }, TTLIdx: []string{"RU1"}, } sq := &StatQueue{ Tenant: "cgrates.org", ID: "SQ_1", dirty: utils.BoolPointer(true), SQMetrics: map[string]StatMetric{ utils.MetaASR: &StatASR{ Answered: 1, Count: 1, Events: map[string]*StatWithCompress{ "cgrates.org:TestStatRemExpired_1": {Stat: 1, CompressFactor: 1}, }, }, }} dm.SetResource(rsr) dm.SetAccount(acc) dm.SetStatQueue(sq) clientConn := make(chan birpc.ClientConnector, 1) clientConn <- clMock(func(ctx *context.Context, serviceMethod string, args, reply any) error { if serviceMethod == utils.ResourceSv1GetResource { tntId, concat := args.(*utils.TenantID) if !concat { return utils.ErrNotConvertible } rpl, err := dm.GetResource(tntId.Tenant, tntId.ID, false, false, utils.NonTransactional) if err != nil { return err } *reply.(**Resource) = rpl return nil } else if serviceMethod == utils.StatSv1GetQueueFloatMetrics { rpl := map[string]float64{ utils.MetaACC: 100.0, } *reply.(*map[string]float64) = rpl return nil } return utils.ErrNotImplemented }) fltrs := []*Filter{ { Tenant: "cgrates.org", ID: "FLTR_ACC", Rules: []*FilterRule{{ Type: utils.MetaString, Element: "~*accounts.1001.BalanceMap.*voice[0].Value", Values: []string{"~*accounts.1001.BalanceMap.*voice[0].Value" + utils.IfaceAsString(20*float64(time.Second))}, }}, }, { Tenant: "cgrates.org", ID: "FLTR_RES", Rules: []*FilterRule{ { Type: "*lte", Element: "~*resources.RL1.Usage.RUI.Units", Values: []string{"~*resources.RL1.Usage.RUI.Units.2"}, }, }, }, { Tenant: "cgrates.org", ID: "FLTR_STAT", Rules: []*FilterRule{ { Type: "*gt", Element: "~*stats.SQ_1.*asr", Values: []string{"~*stats.SQ_1.*asr.10.0"}, }, }, }, } connMgr := NewConnManager(cfg, map[string]chan birpc.ClientConnector{ utils.ConcatenatedKey(utils.MetaInternal, utils.MetaResources): clientConn, utils.ConcatenatedKey(utils.MetaInternal, utils.MetaStatS): clientConn, }) ev := &utils.MapStorage{} fS := NewFilterS(cfg, connMgr, dm) fltriDs := make([]string, len(fltrs)) for i, fltr := range fltrs { dm.SetFilter(fltr) fltriDs[i] = fltr.ID } if _, err := fS.Pass("cgrates.org", fltriDs, ev); err != nil { t.Error(err) } } func TestValidateInlineFilters(t *testing.T) { cases := []struct { fltrs []string expectedErr string }{ {[]string{"FLTR", "*string:~*req.Account:1001"}, ""}, {[]string{"FLTR", "*string:~*req,Acoount1001"}, "inline parse error for string: <*string:~*req,Acoount1001>"}, {[]string{"*exists:~*req.Supplier:"}, ""}, {[]string{"*exists:*req.Supplier"}, "inline parse error for string: <*exists:*req.Supplier>"}, {[]string{"*rsr:~*req.Account:(10)"}, "invalid RSRFilter start rule in string: <(10)>"}, } computeTestName := func(idx int, params []string) string { return fmt.Sprintf("Test No %d with parameters: %v", idx, params) } for i, c := range cases { t.Run(computeTestName(i, c.fltrs), func(t *testing.T) { err := validateInlineFilters(c.fltrs) if err != nil { if c.expectedErr == "" { t.Errorf("did not expect error, received: %v", err) } } else if c.expectedErr != "" { t.Errorf("expected error: %v", err) } }) } } func TestComputeDispatcherIndexesErr(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) dm.SetDispatcherProfile(&DispatcherProfile{ Tenant: "cgrates.org", ID: "DSP1", FilterIDs: []string{"DSP_FLT"}, Strategy: utils.MetaFirst, Subsystems: []string{utils.MetaAttributes, utils.MetaSessionS}, Weight: 20, }, true) if _, err := ComputeDispatcherIndexes(dm, "cgrates.org", utils.MetaSessionS, &[]string{"DSP1"}, ""); err == nil { t.Error(err) } } func TestFIRemoveItemFromIndexErr(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() defer func() { cfg2, _ := config.NewDefaultCGRConfig() config.SetCgrConfig(cfg2) }() cfg.DataDbCfg().RmtConns = []string{utils.ConcatenatedKey(utils.MetaInternal, utils.ReplicatorSv1)} db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) clientConn := make(chan birpc.ClientConnector, 1) clientConn <- clMock(func(ctx *context.Context, serviceMethod string, _, _ any) error { return utils.ErrNotImplemented }) dm := NewDataManager(db, cfg.CacheCfg(), NewConnManager(cfg, map[string]chan birpc.ClientConnector{ utils.ConcatenatedKey(utils.MetaInternal, utils.ReplicatorSv1): clientConn, })) testCases := []struct { name string itemType string cachePart string itemID string oldFilters []string }{ { name: "Threshold Not Found", itemType: utils.ThresholdProfilePrefix, itemID: "TH1", cachePart: utils.MetaThresholdProfiles, }, { name: "Attributes Not Found", itemType: utils.AttributeProfilePrefix, itemID: "ATTR_1", cachePart: utils.MetaAttributeProfiles, }, { name: "Resource Not Found", itemType: utils.ResourceProfilesPrefix, itemID: "RSC_1", cachePart: utils.MetaResourceProfile, }, { name: "Stat Queue Not Found", itemType: utils.StatQueueProfilePrefix, itemID: "SQ_1", cachePart: utils.MetaStatQueueProfiles, }, { name: "Supplier Not Found", itemType: utils.SupplierProfilePrefix, itemID: "SPP_1", cachePart: utils.MetaSupplierProfiles, }, { name: "ChargerProfile Not Found", itemType: utils.ChargerProfilePrefix, itemID: "SQ_1", cachePart: utils.MetaChargerProfiles, }, { name: "DispatcherProfile Not Found", itemType: utils.DispatcherProfilePrefix, itemID: "SPP_1", cachePart: utils.MetaDispatcherProfiles, }, } config.SetCgrConfig(cfg) for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { cfg.DataDbCfg().Items[tc.cachePart].Remote = true if err := NewFilterIndexer(dm, tc.itemType, "cgrates.org").RemoveItemFromIndex("cgrates.org", tc.itemID, tc.oldFilters); err == nil { t.Error("expected error, received nil") } }) } } func TestComputeSupplierIndexesErrs(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) testCases := []struct { name string dm *DataManager tenant string sppIDs *[]string expErr bool }{ { name: "Supplier Not Found", dm: dm, expErr: true, sppIDs: &[]string{"SPP_1"}, }, { name: "Without Filter", dm: dm, expErr: false, sppIDs: &[]string{"SPP_2"}, }, { name: "Filter doesn't exist", dm: dm, expErr: true, sppIDs: &[]string{"SPP_3"}, }, } dm.SetSupplierProfile(&SupplierProfile{ Tenant: "cgrates.org", ID: "SPP_2", Sorting: utils.MetaQOS, }, true) dm.SetSupplierProfile(&SupplierProfile{ Tenant: "cgrates.org", ID: "SPP_3", Sorting: utils.MetaQOS, FilterIDs: []string{"FLT_1"}, }, true) for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { if _, err := ComputeSupplierIndexes(tc.dm, "cgrates.org", tc.sppIDs, ""); (err != nil) != tc.expErr { t.Errorf("expected error: %v, received: %v", tc.expErr, err) } }) } } func TestComputeStatIndexesErrs(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) testCases := []struct { name string dm *DataManager tenant string stIDs *[]string expErr bool }{ { name: "StatQueueProfile Not Found", dm: dm, expErr: true, stIDs: &[]string{"SQ_1"}, }, { name: "Without Filter", dm: dm, expErr: false, stIDs: &[]string{"TEST_PROFILE1"}, }, { name: "Filter dosnt exist ", dm: dm, expErr: true, }, } dm.SetStatQueueProfile(&StatQueueProfile{ Tenant: "cgrates.org", ID: "TEST_PROFILE1", QueueLength: 10, TTL: time.Duration(10) * time.Second, Metrics: []*MetricWithFilters{ { MetricID: "*sum:~Val", }, }, ThresholdIDs: []string{"*none"}, }, true) dm.SetStatQueueProfile(&StatQueueProfile{ Tenant: "cgrates.org", ID: "TestStats", FilterIDs: []string{"FLTR_STATS"}, QueueLength: 100, TTL: time.Duration(1 * time.Second), Metrics: []*MetricWithFilters{ { MetricID: "*sum:~Value", }, { MetricID: "*average:~Value", }, { MetricID: "*sum:~Usage", }, }, }, true) for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { if _, err := ComputeStatIndexes(tc.dm, "cgrates.org", tc.stIDs, ""); (err != nil) != tc.expErr { t.Errorf("expected error: %v, received: %v", tc.expErr, err) } }) } } func TestComputeAttributeIndexesErr(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) testCases := []struct { name string dm *DataManager tenant string attrIDs *[]string expErr bool }{ { name: "AttributeProfile Not Found", dm: dm, expErr: true, attrIDs: &[]string{"ATTR_1"}, }, { name: "Without Filter", dm: dm, expErr: false, attrIDs: &[]string{"ATTR_1002_SIMPLEAUTH"}, }, { name: "Filter dosnt exist ", dm: dm, expErr: true, }, } dm.SetAttributeProfile(&AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_1002_SIMPLEAUTH", Contexts: []string{"simpleauth"}, Attributes: []*Attribute{ { Path: utils.MetaReq + utils.NestingSep + "Password", Type: utils.META_CONSTANT, Value: config.NewRSRParsersMustCompile("CGRateS.org", true, utils.INFIELD_SEP), }, }, Weight: 20.0, }, true) dm.SetAttributeProfile(&AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_1001_AUTH", Contexts: []string{"simpleauth"}, FilterIDs: []string{"FLTR_Attr"}, Attributes: []*Attribute{ { Path: utils.MetaReq + utils.NestingSep + "Password", Type: utils.META_CONSTANT, Value: config.NewRSRParsersMustCompile("CGRateS.org", true, utils.INFIELD_SEP), }, }, Weight: 20.0, }, true) for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { if _, err := ComputeAttributeIndexes(tc.dm, "cgrates.org", "simpleauth", tc.attrIDs, ""); (err != nil) != tc.expErr { t.Errorf("expected error: %v, received: %v", tc.expErr, err) } }) } } func TestComputeDispatcherIndexesErrs(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) testCases := []struct { name string dm *DataManager tenant string dspIDs *[]string expErr bool }{ { name: "DispatcherProfile Not Found", dm: dm, tenant: "cgrates.org", dspIDs: &[]string{"DISP_1"}, expErr: true, }, { name: "Without Filter", dm: dm, tenant: "cgrates.org", dspIDs: &[]string{"DISP_2"}, expErr: false, }, } dm.SetDispatcherProfile(&DispatcherProfile{ Tenant: "cgrates.org", ID: "DISP_2", Subsystems: []string{utils.MetaSessionS}, }, true) for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { if _, err := ComputeDispatcherIndexes(tc.dm, "cgrates.org", utils.MetaSessionS, tc.dspIDs, ""); (err != nil) != tc.expErr { t.Errorf("expected error: %v, received: %v", tc.expErr, err) } }) } } func TestComputeResourceIndexesErrs(t *testing.T) { cfg, _ := config.NewDefaultCGRConfig() db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items) dm := NewDataManager(db, cfg.CacheCfg(), nil) testCases := []struct { name string dm *DataManager tenant string resIDs *[]string expErr bool }{ { name: "ResourceProfile Not Found", dm: dm, tenant: "cgrates.org", resIDs: &[]string{"RES_1"}, expErr: true, }, { name: "Without Filter", dm: dm, tenant: "cgrates.org", resIDs: &[]string{"RES_2"}, expErr: true, }, } dm.SetResourceProfile(&ResourceProfile{ Tenant: "cgrates.org", ID: "RES_2", FilterIDs: []string{"FLTR_RES"}, UsageTTL: time.Duration(-1), Limit: 2, AllocationMessage: "Account1Channels", Weight: 20, ThresholdIDs: []string{utils.META_NONE}, }, true) for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { if _, err := ComputeResourceIndexes(tc.dm, "cgrates.org", tc.resIDs, ""); (err != nil) != tc.expErr { t.Errorf("expected error: %v, received: %v", tc.expErr, err) } }) } }