// +build integration /* 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 general_tests import ( "net/rpc" "path" "reflect" "sort" "strings" "testing" "time" v1 "github.com/cgrates/cgrates/apier/v1" "github.com/cgrates/cgrates/config" "github.com/cgrates/cgrates/engine" "github.com/cgrates/cgrates/utils" ) var ( attrCfgPath string attrCfg *config.CGRConfig attrRPC *rpc.Client attrDataDir = "/usr/share/cgrates" alsPrfConfigDIR string sTestsAlsPrf = []func(t *testing.T){ testAttributeSInitCfg, testAttributeSInitDataDb, testAttributeSResetStorDb, testAttributeSStartEngine, testAttributeSRPCConn, testAttributeSLoadFromFolder, testAttributeSProcessEvent, testAttributeSProcessEventWithAccount, testAttributeSProcessEventWithAccountFull, testAttributeSProcessEventWithStat, testAttributeSProcessEventWithStatFull, testAttributeSProcessEventWithResource, testAttributeSProcessEventWithResourceFull, testAttributeSProcessEventWithLibPhoneNumber, testAttributeSProcessEventWithLibPhoneNumberComposed, testAttributeSProcessEventWithLibPhoneNumberFull, testAttributeSStopEngine, } ) func TestAttributeSIT(t *testing.T) { attrsTests := sTestsAlsPrf switch *dbType { case utils.MetaInternal: alsPrfConfigDIR = "tutinternal" case utils.MetaMySQL: alsPrfConfigDIR = "tutmysql" case utils.MetaMongo: alsPrfConfigDIR = "tutmongo" case utils.MetaPostgres: t.SkipNow() default: t.Fatal("Unknown Database type") } for _, stest := range attrsTests { t.Run(alsPrfConfigDIR, stest) } } func testAttributeSInitCfg(t *testing.T) { var err error attrCfgPath = path.Join(attrDataDir, "conf", "samples", alsPrfConfigDIR) attrCfg, err = config.NewCGRConfigFromPath(attrCfgPath) if err != nil { t.Error(err) } attrCfg.DataFolderPath = attrDataDir // Share DataFolderPath through config towards StoreDb for Flush() } func testAttributeSInitDataDb(t *testing.T) { if err := engine.InitDataDb(attrCfg); err != nil { t.Fatal(err) } } // Wipe out the cdr database func testAttributeSResetStorDb(t *testing.T) { if err := engine.InitStorDb(attrCfg); err != nil { t.Fatal(err) } } // Start CGR Engine func testAttributeSStartEngine(t *testing.T) { if _, err := engine.StopStartEngine(attrCfgPath, *waitRater); err != nil { t.Fatal(err) } } // Connect rpc client to rater func testAttributeSRPCConn(t *testing.T) { var err error attrRPC, err = newRPCClient(attrCfg.ListenCfg()) // We connect over JSON so we can also troubleshoot if needed if err != nil { t.Fatal(err) } } func testAttributeSLoadFromFolder(t *testing.T) { var reply string attrs := &utils.AttrLoadTpFromFolder{FolderPath: path.Join(*dataDir, "tariffplans", "testit")} if err := attrRPC.Call(utils.APIerSv1LoadTariffPlanFromFolder, attrs, &reply); err != nil { t.Error(err) } time.Sleep(500 * time.Millisecond) } func testAttributeSProcessEvent(t *testing.T) { ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEvent", Event: map[string]interface{}{ utils.EVENT_NAME: "VariableTest", utils.ToR: utils.VOICE, }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_VARIABLE"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + utils.Category}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEvent", Event: map[string]interface{}{ utils.EVENT_NAME: "VariableTest", utils.Category: utils.VOICE, utils.ToR: utils.VOICE, }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } sort.Strings(rplyEv.AlteredFields) if !reflect.DeepEqual(eRply, rplyEv) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply), utils.ToJSON(rplyEv)) } } func testAttributeSProcessEventWithAccount(t *testing.T) { // add new attribute profile var result string alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_ACCOUNT", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddAccountInfo"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "Balance", Type: utils.MetaVariable, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*accounts.1001.BalanceMap.*monetary[0].Value", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_ACCOUNT"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithAccount", Event: map[string]interface{}{ "EventName": "AddAccountInfo", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_ACCOUNT"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "Balance"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithAccount", Event: map[string]interface{}{ "EventName": "AddAccountInfo", "Balance": "10", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } sort.Strings(rplyEv.AlteredFields) if !reflect.DeepEqual(eRply, rplyEv) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply), utils.ToJSON(rplyEv)) } } func testAttributeSProcessEventWithAccountFull(t *testing.T) { // add new attribute profile var result string alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_ACCOUNT2", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddFullAccount"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "FullAccount", Type: utils.MetaVariable, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*accounts.1001", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_ACCOUNT2"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithAccount2", Event: map[string]interface{}{ "EventName": "AddFullAccount", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_ACCOUNT2"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "FullAccount"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithAccount2", Event: map[string]interface{}{ "EventName": "AddFullAccount", "FullAccount": "{\"ID\":\"cgrates.org:1001\",\"BalanceMap\":{\"*monetary\":[{\"Uuid\":\"18160631-a4ae-4078-8048-b4c6b87a36c6\",\"ID\":\"\",\"Value\":10,\"ExpirationDate\":\"0001-01-01T00:00:00Z\",\"Weight\":10,\"DestinationIDs\":{},\"RatingSubject\":\"\",\"Categories\":{},\"SharedGroups\":{},\"Timings\":null,\"TimingIDs\":{},\"Disabled\":false,\"Factor\":null,\"Blocker\":false}]},\"UnitCounters\":null,\"ActionTriggers\":null,\"AllowNegative\":false,\"Disabled\":false,\"UpdateTime\":\"2020-10-06T12:43:51.805Z\"}", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } if !reflect.DeepEqual(rplyEv.MatchedProfiles, eRply.MatchedProfiles) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply.MatchedProfiles), utils.ToJSON(rplyEv.MatchedProfiles)) } else if !reflect.DeepEqual(rplyEv.AlteredFields, eRply.AlteredFields) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply.AlteredFields), utils.ToJSON(rplyEv.AlteredFields)) } // some fields are generated(e.g. BalanceID) and compare only some part of the string strAcc := utils.IfaceAsString(rplyEv.CGREvent.Event["FullAccount"]) if !strings.Contains(strAcc, "\"ID\":\"cgrates.org:1001\"") { t.Errorf("Expecting: %s, received: %s", "\"ID\":\"cgrates.org:1001\"", strAcc) } else if !strings.Contains(strAcc, "\"UnitCounters\":null,\"ActionTriggers\":null,\"AllowNegative\":false,\"Disabled\":false") { t.Errorf("Expecting: %s, received: %s", "\"UnitCounters\":null,\"ActionTriggers\":null,\"AllowNegative\":false,\"Disabled\":false", strAcc) } } func testAttributeSProcessEventWithStat(t *testing.T) { // simulate some stat event var reply []string expected := []string{"Stat_1"} ev1 := &engine.StatsArgsProcessEvent{ CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "event1", Event: map[string]interface{}{ utils.Account: "1001", utils.AnswerTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC), utils.Usage: 11 * time.Second, utils.COST: 10.0, }, }, }, } if err := attrRPC.Call(utils.StatSv1ProcessEvent, &ev1, &reply); err != nil { t.Error(err) } else if !reflect.DeepEqual(reply, expected) { t.Errorf("Expecting: %+v, received: %+v", expected, reply) } expected = []string{"Stat_1"} ev1.CGREvent = &utils.CGREvent{ Tenant: "cgrates.org", ID: "event2", Event: map[string]interface{}{ utils.Account: "1001", utils.AnswerTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC), utils.Usage: 11 * time.Second, utils.COST: 10.5, }, } if err := attrRPC.Call(utils.StatSv1ProcessEvent, &ev1, &reply); err != nil { t.Error(err) } else if !reflect.DeepEqual(reply, expected) { t.Errorf("Expecting: %+v, received: %+v", expected, reply) } // add new attribute profile alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_STATS", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddStatEvent"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "AcdMetric", Type: utils.MetaVariable, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*stats.Stat_1.*acd", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() var result string if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_STATS"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithStat", Event: map[string]interface{}{ "EventName": "AddStatEvent", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_STATS"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "AcdMetric"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithStat", Event: map[string]interface{}{ "EventName": "AddStatEvent", "AcdMetric": "11", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } sort.Strings(rplyEv.AlteredFields) if !reflect.DeepEqual(eRply, rplyEv) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply), utils.ToJSON(rplyEv)) } } func testAttributeSProcessEventWithStatFull(t *testing.T) { // add new attribute profile alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_STATS2", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddFullStats"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "AllMetrics", Type: utils.MetaVariable, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*stats.Stat_1", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() var result string if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_STATS2"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithStat", Event: map[string]interface{}{ "EventName": "AddFullStats", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_STATS2"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "AllMetrics"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithStat", Event: map[string]interface{}{ "EventName": "AddFullStats", "AllMetrics": "{\"*acd\":11,\"*asr\":100,\"*tcd\":22}", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } sort.Strings(rplyEv.AlteredFields) if !reflect.DeepEqual(eRply, rplyEv) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply), utils.ToJSON(rplyEv)) } } func testAttributeSProcessEventWithResource(t *testing.T) { //create a resourceProfile rlsConfig := &engine.ResourceProfile{ Tenant: "cgrates.org", ID: "ResTest", UsageTTL: time.Minute, Limit: 10, AllocationMessage: "MessageAllocation", Stored: true, Weight: 20, ThresholdIDs: []string{utils.META_NONE}, } var result string if err := attrRPC.Call(utils.APIerSv1SetResourceProfile, &v1.ResourceWithCache{ResourceProfile: rlsConfig}, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var reply *engine.ResourceProfile if err := attrRPC.Call(utils.APIerSv1GetResourceProfile, &utils.TenantID{Tenant: "cgrates.org", ID: rlsConfig.ID}, &reply); err != nil { t.Error(err) } else if !reflect.DeepEqual(reply, rlsConfig) { t.Errorf("Expecting: %+v, received: %+v", utils.ToJSON(rlsConfig), utils.ToJSON(reply)) } // Allocate 3 units for resource ResTest argsRU := utils.ArgRSv1ResourceUsage{ CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: utils.UUIDSha1Prefix(), Event: map[string]interface{}{ "Account": "3001", "Destination": "3002"}, }, }, UsageID: "651a8db2-4f67-4cf8-b622-169e8a482e21", Units: 3, } if err := attrRPC.Call(utils.ResourceSv1AllocateResources, argsRU, &result); err != nil { t.Error(err) } // Allocate 2 units for resource ResTest argsRU2 := utils.ArgRSv1ResourceUsage{ CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: utils.UUIDSha1Prefix(), Event: map[string]interface{}{ "Account": "3001", "Destination": "3002"}, }, }, UsageID: "651a8db2-4f67-4cf8-b622-169e8a482e22", Units: 2, } if err := attrRPC.Call(utils.ResourceSv1AllocateResources, argsRU2, &result); err != nil { t.Error(err) } // add new attribute profile alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_RESOURCE", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddResourceUsages"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "ResourceTotalUsages", Type: utils.MetaVariable, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*resources.ResTest.TotalUsage", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_RESOURCE"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithResource", Event: map[string]interface{}{ "EventName": "AddResourceUsages", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_RESOURCE"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "ResourceTotalUsages"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithResource", Event: map[string]interface{}{ "EventName": "AddResourceUsages", "ResourceTotalUsages": "5", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } sort.Strings(rplyEv.AlteredFields) if !reflect.DeepEqual(eRply, rplyEv) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply), utils.ToJSON(rplyEv)) } } func testAttributeSProcessEventWithResourceFull(t *testing.T) { // add new attribute profile var result string alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_RESOURCE2", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddFullResource"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "FullResource", Type: utils.MetaVariable, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*resources.ResTest", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_RESOURCE2"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithResource2", Event: map[string]interface{}{ "EventName": "AddFullResource", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_RESOURCE2"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "FullResource"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithResource2", Event: map[string]interface{}{ "EventName": "AddFullResource", "FullResource": "{\"Tenant\":\"cgrates.org\",\"ID\":\"ResTest\",\"Usages\":{\"651a8db2-4f67-4cf8-b622-169e8a482e21\":{\"Tenant\":\"cgrates.org\",\"ID\":\"651a8db2-4f67-4cf8-b622-169e8a482e21\",\"ExpiryTime\":\"2020-10-06T16:12:52.450804203+03:00\",\"Units\":3},\"651a8db2-4f67-4cf8-b622-169e8a482e22\":{\"Tenant\":\"cgrates.org\",\"ID\":\"651a8db2-4f67-4cf8-b622-169e8a482e22\",\"ExpiryTime\":\"2020-10-06T16:12:52.451034151+03:00\",\"Units\":2}},\"TTLIdx\":[\"651a8db2-4f67-4cf8-b622-169e8a482e21\",\"651a8db2-4f67-4cf8-b622-169e8a482e22\"]}", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } if !reflect.DeepEqual(rplyEv.MatchedProfiles, eRply.MatchedProfiles) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply.MatchedProfiles), utils.ToJSON(rplyEv.MatchedProfiles)) } else if !reflect.DeepEqual(rplyEv.AlteredFields, eRply.AlteredFields) { t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eRply.AlteredFields), utils.ToJSON(rplyEv.AlteredFields)) } // some fields are generated(e.g. Time) and compare only some part of the string strRes := utils.IfaceAsString(rplyEv.CGREvent.Event["FullResource"]) if !strings.Contains(strRes, "{\"Tenant\":\"cgrates.org\",\"ID\":\"ResTest\",\"Usages\":{") { t.Errorf("Expecting: %s, received: %s", "{\"Tenant\":\"cgrates.org\",\"ID\":\"ResTest\",\"Usages\":{", strRes) } else if !strings.Contains(strRes, ",\"TTLIdx\":[") { t.Errorf("Expecting: %s, received: %s", ",\"TTLIdx\":[", strRes) } } func testAttributeSProcessEventWithLibPhoneNumber(t *testing.T) { // add new attribute profile var result string alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_LIBPHONENUMBER2", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddDestinationCarrier"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "DestinationCarrier", Type: utils.MetaVariable, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*libphonenumber.<~*req.Destination>.Carrier", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_LIBPHONENUMBER2"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithLibPhoneNumber2", Event: map[string]interface{}{ "EventName": "AddDestinationCarrier", "Destination": "+447779330921", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_LIBPHONENUMBER2"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "DestinationCarrier"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithLibPhoneNumber2", Event: map[string]interface{}{ "EventName": "AddDestinationCarrier", "Destination": "+447779330921", "DestinationCarrier": "Orange", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } sort.Strings(rplyEv.AlteredFields) if !reflect.DeepEqual(eRply, rplyEv) { t.Errorf("Expecting: %+v, received: %+v", utils.ToJSON(eRply), utils.ToJSON(rplyEv)) } } func testAttributeSProcessEventWithLibPhoneNumberComposed(t *testing.T) { // add new attribute profile var result string alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_LIBPHONENUMBER_COMPOSED", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddComposedInfo"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "DestinationCarrier", Type: utils.META_COMPOSED, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*libphonenumber.<~*req.Destination>.Carrier", }, }, }, { Path: utils.MetaReq + utils.NestingSep + "DestinationCarrier", Type: utils.META_COMPOSED, Value: config.RSRParsers{ &config.RSRParser{ Rules: ";", }, }, }, { Path: utils.MetaReq + utils.NestingSep + "DestinationCarrier", Type: utils.META_COMPOSED, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*libphonenumber.<~*req.Destination>.CountryCode", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_LIBPHONENUMBER_COMPOSED"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithLibPhoneNumberComposed", Event: map[string]interface{}{ "EventName": "AddComposedInfo", "Destination": "+447779330921", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_LIBPHONENUMBER_COMPOSED"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "DestinationCarrier"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithLibPhoneNumberComposed", Event: map[string]interface{}{ "EventName": "AddComposedInfo", "Destination": "+447779330921", "DestinationCarrier": "Orange;44", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } sort.Strings(rplyEv.AlteredFields) if !reflect.DeepEqual(eRply, rplyEv) { t.Errorf("Expecting: %+v, received: %+v", utils.ToJSON(eRply), utils.ToJSON(rplyEv)) } } func testAttributeSProcessEventWithLibPhoneNumberFull(t *testing.T) { // add new attribute profile var result string alsPrf := &v1.AttributeWithCache{ AttributeProfile: &engine.AttributeProfile{ Tenant: "cgrates.org", ID: "ATTR_LIBPHONENUMBER", Contexts: []string{utils.META_ANY}, FilterIDs: []string{"*string:~*req.EventName:AddDestinationDetails"}, 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), }, Attributes: []*engine.Attribute{ { Path: utils.MetaReq + utils.NestingSep + "DestinationDetails", Type: utils.MetaVariable, Value: config.RSRParsers{ &config.RSRParser{ Rules: "~*libphonenumber.<~*req.Destination>", }, }, }, }, Blocker: false, Weight: 10, }, } alsPrf.Compile() if err := attrRPC.Call(utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil { t.Error(err) } else if result != utils.OK { t.Error("Unexpected reply returned", result) } var replyAttr *engine.AttributeProfile if err := attrRPC.Call(utils.APIerSv1GetAttributeProfile, utils.TenantIDWithOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_LIBPHONENUMBER"}}, &replyAttr); err != nil { t.Fatal(err) } replyAttr.Compile() if !reflect.DeepEqual(alsPrf.AttributeProfile, replyAttr) { t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, replyAttr) } ev := &engine.AttrArgsProcessEvent{ Context: utils.StringPointer(utils.MetaSessionS), CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithLibPhoneNumber", Event: map[string]interface{}{ "EventName": "AddDestinationDetails", "Destination": "+447779330921", }, }, }, } eRply := engine.AttrSProcessEventReply{ MatchedProfiles: []string{"ATTR_LIBPHONENUMBER"}, AlteredFields: []string{utils.MetaReq + utils.NestingSep + "DestinationDetails"}, CGREventWithOpts: &utils.CGREventWithOpts{ CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "testAttributeSProcessEventWithLibPhoneNumber", Event: map[string]interface{}{ "EventName": "AddDestinationDetails", "Destination": "+447779330921", "DestinationDetails": "{\"Carrier\":\"Orange\",\"CountryCode\":44,\"CountryCodeSource\":1,\"Extension\":\"\",\"GeoLocation\":\"\",\"ItalianLeadingZero\":false,\"LengthOfNationalDestinationCode\":0,\"NationalNumber\":7779330921,\"NumberOfLeadingZeros\":1,\"NumberType\":1,\"PreferredDomesticCarrierCode\":\"\",\"RawInput\":\"+447779330921\",\"Region\":\"GB\"}", }, }, }, } sort.Strings(eRply.AlteredFields) var rplyEv engine.AttrSProcessEventReply if err := attrRPC.Call(utils.AttributeSv1ProcessEvent, ev, &rplyEv); err != nil { t.Fatal(err) } sort.Strings(rplyEv.AlteredFields) if !reflect.DeepEqual(eRply, rplyEv) { t.Errorf("Expecting: %+v, received: %+v", utils.ToJSON(eRply), utils.ToJSON(rplyEv)) } } func testAttributeSStopEngine(t *testing.T) { if err := engine.KillEngine(accDelay); err != nil { t.Error(err) } }