/* 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 ( "testing" "time" "github.com/cgrates/cgrates/utils" ) var rLS *ResourceLimiterService /* func TestRLsIndexStringFilters(t *testing.T) { rls := []*ResourceLimit{ &ResourceLimit{ ID: "RL1", Weight: 20, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Account", Values: []string{"1001", "1002"}}, &RequestFilter{Type: MetaRSRFields, Values: []string{"Subject(~^1.*1$)", "Destination(1002)"}, rsrFields: utils.ParseRSRFieldsMustCompile("Subject(~^1.*1$);Destination(1002)", utils.INFIELD_SEP), }}, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 2, Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ ID: "RL2", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Account", Values: []string{"dan", "1002"}}, &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"dan"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, UsageTTL: time.Duration(1 * time.Millisecond), Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ ID: "RL4", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaStringPrefix, FieldName: "Destination", Values: []string{"+49"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ ID: "RL5", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaStringPrefix, FieldName: "Destination", Values: []string{"+40"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, UsageTTL: time.Duration(10 * time.Millisecond), Usage: make(map[string]*ResourceUsage), }, } for _, rl := range rls { cache.Set(utils.ResourceLimitsPrefix+rl.ID, rl, true, "") } rLS = new(ResourceLimiterService) eIndexes := map[string]map[string]utils.StringMap{ "Account": map[string]utils.StringMap{ "1001": utils.StringMap{ "RL1": true, }, "1002": utils.StringMap{ "RL1": true, "RL2": true, }, "dan": utils.StringMap{ "RL2": true, }, }, "Subject": map[string]utils.StringMap{ "dan": utils.StringMap{ "RL2": true, }, }, utils.NOT_AVAILABLE: map[string]utils.StringMap{ utils.NOT_AVAILABLE: utils.StringMap{ "RL4": true, "RL5": true, }, }, } if err := rLS.indexStringFilters(nil); err != nil { t.Error(err) } else if !reflect.DeepEqual(eIndexes, rLS.stringIndexes) { t.Errorf("Expecting: %+v, received: %+v", eIndexes, rLS.stringIndexes) } rl3 := &ResourceLimit{ ID: "RL3", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"dan"}}, &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"1003"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, Usage: make(map[string]*ResourceUsage), } cache.Set(utils.ResourceLimitsPrefix+rl3.ID, rl3, true, "") rl6 := &ResourceLimit{ // Add it so we can test expiryTime ID: "RL6", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"dan"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), ExpiryTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, Usage: make(map[string]*ResourceUsage), } cache.Set(utils.ResourceLimitsPrefix+rl6.ID, rl6, true, "") eIndexes = map[string]map[string]utils.StringMap{ "Account": map[string]utils.StringMap{ "1001": utils.StringMap{ "RL1": true, }, "1002": utils.StringMap{ "RL1": true, "RL2": true, }, "dan": utils.StringMap{ "RL2": true, }, }, "Subject": map[string]utils.StringMap{ "dan": utils.StringMap{ "RL2": true, "RL3": true, "RL6": true, }, "1003": utils.StringMap{ "RL3": true, }, }, utils.NOT_AVAILABLE: map[string]utils.StringMap{ utils.NOT_AVAILABLE: utils.StringMap{ "RL4": true, "RL5": true, }, }, } // Test index update if err := rLS.indexStringFilters([]string{rl3.ID, rl6.ID}); err != nil { t.Error(err) } else if !reflect.DeepEqual(eIndexes, rLS.stringIndexes) { t.Errorf("Expecting: %+v, received: %+v", eIndexes, rLS.stringIndexes) } } */ func TestRLsLoadRLs(t *testing.T) { rls := []*ResourceLimit{ &ResourceLimit{ ID: "RL1", Weight: 20, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Account", Values: []string{"1001", "1002"}}, &RequestFilter{Type: MetaRSRFields, Values: []string{"Subject(~^1.*1$)", "Destination(1002)"}, rsrFields: utils.ParseRSRFieldsMustCompile("Subject(~^1.*1$);Destination(1002)", utils.INFIELD_SEP), }}, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 2, Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ ID: "RL2", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Account", Values: []string{"dan", "1002"}}, &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"dan"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, UsageTTL: time.Duration(1 * time.Millisecond), Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ ID: "RL3", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"dan"}}, &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"1003"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ ID: "RL4", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaStringPrefix, FieldName: "Destination", Values: []string{"+49"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ ID: "RL5", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaStringPrefix, FieldName: "Destination", Values: []string{"+40"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, UsageTTL: time.Duration(10 * time.Millisecond), Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ // Add it so we can test expiryTime ID: "RL6", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"dan"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), ExpiryTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, Usage: make(map[string]*ResourceUsage), }, } rlIdxr, err := NewReqFilterIndexer(accountingStorage, utils.ResourceLimitsIndex) if err != nil { t.Error(err) } for _, rl := range rls { if err := accountingStorage.SetResourceLimit(rl, utils.NonTransactional); err != nil { t.Error(err) } rlIdxr.IndexFilters(rl.ID, rl.Filters) } if err := rlIdxr.StoreIndexes(); err != nil { t.Error(err) } } func TestRLsMatchingResourceLimitsForEvent(t *testing.T) { rLS = &ResourceLimiterService{dataDB: accountingStorage, cdrStatS: nil} eResLimits := map[string]*ResourceLimit{ "RL1": &ResourceLimit{ ID: "RL1", Weight: 20, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Account", Values: []string{"1001", "1002"}}, &RequestFilter{Type: MetaRSRFields, Values: []string{"Subject(~^1.*1$)", "Destination(1002)"}, rsrFields: utils.ParseRSRFieldsMustCompile("Subject(~^1.*1$);Destination(1002)", utils.INFIELD_SEP), }}, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 2, Usage: make(map[string]*ResourceUsage), }, "RL2": &ResourceLimit{ ID: "RL2", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Account", Values: []string{"dan", "1002"}}, &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"dan"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, UsageTTL: time.Duration(1 * time.Millisecond), Usage: make(map[string]*ResourceUsage), }, } if resLimits, err := rLS.matchingResourceLimitsForEvent(map[string]interface{}{"Account": "1002", "Subject": "dan", "Destination": "1002"}); err != nil { t.Error(err) } else if len(eResLimits) != len(resLimits) { t.Errorf("Expecting: %+v, received: %+v", eResLimits, resLimits) } else { for rlID := range eResLimits { if _, hasID := resLimits[rlID]; !hasID { t.Errorf("Expecting: %+v, received: %+v", eResLimits, resLimits) } } // Make sure the filters are what we expect to be after retrieving from cache: fltr := resLimits["RL1"].Filters[1] if pass, _ := fltr.Pass(map[string]interface{}{"Subject": "10000001"}, "", nil); !pass { t.Errorf("Expecting RL: %+v, received: %+v", eResLimits["RL1"], resLimits["RL1"]) } if pass, _ := fltr.Pass(map[string]interface{}{"Account": "1002"}, "", nil); pass { t.Errorf("Expecting RL: %+v, received: %+v", eResLimits["RL1"], resLimits["RL1"]) } } } func TestRLsV1ResourceLimitsForEvent(t *testing.T) { eLimits := []*ResourceLimit{ &ResourceLimit{ ID: "RL1", Weight: 20, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Account", Values: []string{"1001", "1002"}}, &RequestFilter{Type: MetaRSRFields, Values: []string{"Subject(~^1.*1$)", "Destination(1002)"}, rsrFields: utils.ParseRSRFieldsMustCompile("Subject(~^1.*1$);Destination(1002)", utils.INFIELD_SEP), }}, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 2, Usage: make(map[string]*ResourceUsage), }, &ResourceLimit{ ID: "RL2", Weight: 10, Filters: []*RequestFilter{ &RequestFilter{Type: MetaString, FieldName: "Account", Values: []string{"dan", "1002"}}, &RequestFilter{Type: MetaString, FieldName: "Subject", Values: []string{"dan"}}, }, ActivationTime: time.Date(2014, 7, 3, 13, 43, 0, 1, time.UTC), Limit: 1, UsageTTL: time.Duration(1 * time.Millisecond), Usage: make(map[string]*ResourceUsage), }, } var rcvLmts []*ResourceLimit if err := rLS.V1ResourceLimitsForEvent(map[string]interface{}{"Account": "1002", "Subject": "dan", "Destination": "1002"}, &rcvLmts); err != nil { t.Error(err) } else if len(eLimits) != len(rcvLmts) { t.Errorf("Expecting: %+v, received: %+v", eLimits, rcvLmts) } } func TestRLsV1InitiateResourceUsage(t *testing.T) { attrRU := utils.AttrRLsResourceUsage{ ResourceUsageID: "651a8db2-4f67-4cf8-b622-169e8a482e50", Event: map[string]interface{}{"Account": "1002", "Subject": "dan", "Destination": "1002"}, RequestedUnits: 1, } var reply string if err := rLS.V1InitiateResourceUsage(attrRU, &reply); err != nil { t.Error(err) } else if reply != utils.OK { t.Error("Received reply: ", reply) } resLimits, err := rLS.matchingResourceLimitsForEvent(attrRU.Event) if err != nil { t.Error(err) } else if len(resLimits) != 2 { t.Errorf("Received: %+v", resLimits) } else if resLimits["RL1"].UsedUnits() != 1 { t.Errorf("RL1: %+v", resLimits["RL1"]) } else if _, hasKey := resLimits["RL1"].Usage[attrRU.ResourceUsageID]; !hasKey { t.Errorf("RL1: %+v", resLimits["RL1"]) } } func TestRLsV1TerminateResourceUsage(t *testing.T) { attrRU := utils.AttrRLsResourceUsage{ ResourceUsageID: "651a8db2-4f67-4cf8-b622-169e8a482e50", Event: map[string]interface{}{"Account": "1002", "Subject": "dan", "Destination": "1002"}, RequestedUnits: 1, } var reply string if err := rLS.V1TerminateResourceUsage(attrRU, &reply); err != nil { t.Error(err) } else if reply != utils.OK { t.Error("Received reply: ", reply) } resLimits, err := rLS.matchingResourceLimitsForEvent(attrRU.Event) if err != nil { t.Error(err) } else if len(resLimits) != 2 { t.Errorf("Received: %+v", resLimits) } else if resLimits["RL1"].UsedUnits() != 0 { t.Errorf("RL1: %+v", resLimits["RL1"]) } else if _, hasKey := resLimits["RL1"].Usage[attrRU.ResourceUsageID]; hasKey { t.Errorf("RL1: %+v", resLimits["RL1"]) } }