//go:build integration // +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 ( "path" "reflect" "testing" "time" "github.com/cgrates/birpc" "github.com/cgrates/birpc/context" "github.com/cgrates/cgrates/config" "github.com/cgrates/cgrates/engine" "github.com/cgrates/cgrates/sessions" "github.com/cgrates/cgrates/utils" ) var ( ses3CfgDir string ses3CfgPath string ses3Cfg *config.CGRConfig ses3RPC *birpc.Client ses3Tests = []func(t *testing.T){ testSes3ItLoadConfig, testSes3ItResetDataDB, testSes3ItResetStorDb, testSes3ItStartEngine, testSes3ItRPCConn, testSes3ItLoadFromFolder, testSes3ItProcessEvent, testSes3ItThreshold1002After, testSes3ItStatMetricsAfter, testSes3ItProcessEvent, testSes3ItThreshold1002After2, testSes3ItStatMetricsAfter2, testSes3ItAddVoiceBalance, testSes3ItTerminatWithoutInit, testSes3ItBalance, testSes3ItCDRs, testSes3ItStopCgrEngine, } ) func TestSes3ItSessions(t *testing.T) { switch *utils.DBType { case utils.MetaInternal: ses3CfgDir = "sessions_internal" case utils.MetaMySQL: ses3CfgDir = "sessions_mysql" case utils.MetaMongo: ses3CfgDir = "sessions_mongo" case utils.MetaPostgres: t.SkipNow() default: t.Fatal("Unknown Database type") } for _, stest := range ses3Tests { t.Run(ses3CfgDir, stest) } } func testSes3ItLoadConfig(t *testing.T) { var err error ses3CfgPath = path.Join(*utils.DataDir, "conf", "samples", ses3CfgDir) if ses3Cfg, err = config.NewCGRConfigFromPath(ses3CfgPath); err != nil { t.Error(err) } } func testSes3ItResetDataDB(t *testing.T) { if err := engine.InitDataDB(ses3Cfg); err != nil { t.Fatal(err) } } func testSes3ItResetStorDb(t *testing.T) { if err := engine.InitStorDb(ses3Cfg); err != nil { t.Fatal(err) } } func testSes3ItStartEngine(t *testing.T) { if _, err := engine.StopStartEngine(ses3CfgPath, *utils.WaitRater); err != nil { t.Fatal(err) } } func testSes3ItRPCConn(t *testing.T) { ses3RPC = engine.NewRPCClient(t, ses3Cfg.ListenCfg()) } func testSes3ItLoadFromFolder(t *testing.T) { var reply string attrs := &utils.AttrLoadTpFromFolder{FolderPath: path.Join(*utils.DataDir, "tariffplans", "testit")} if err := ses3RPC.Call(context.Background(), utils.APIerSv1LoadTariffPlanFromFolder, attrs, &reply); err != nil { t.Error(err) } time.Sleep(100 * time.Millisecond) } func testSes3ItProcessEvent(t *testing.T) { initUsage := 5 * time.Minute args := sessions.V1ProcessMessageArgs{ AllocateResources: true, Debit: true, GetAttributes: true, ProcessThresholds: true, ProcessStats: true, CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "TestSSv1ItProcessEvent", Event: map[string]any{ utils.CGRID: "c87609aa1cb6e9529ab1836cfeeebaab7aa7ebaf", utils.Tenant: "cgrates.org", utils.Category: "call", utils.ToR: utils.MetaVoice, utils.OriginID: "TestSSv1It2", utils.RequestType: utils.MetaPrepaid, utils.AccountField: "1001", utils.Destination: "1002", utils.SetupTime: time.Date(2018, time.January, 7, 16, 60, 0, 0, time.UTC), utils.AnswerTime: time.Date(2018, time.January, 7, 16, 60, 10, 0, time.UTC), utils.Usage: initUsage, }, APIOpts: map[string]any{ utils.OptsAPIKey: "ses12345", }, }, } var rply sessions.V1ProcessMessageReply if err := ses3RPC.Call(context.Background(), utils.SessionSv1ProcessMessage, args, &rply); err != nil { t.Fatal(err) } if rply.MaxUsage == nil || *rply.MaxUsage != initUsage { t.Errorf("Unexpected MaxUsage: %v", rply.MaxUsage) } if *rply.ResourceAllocation != "RES_ACNT_1001" { t.Errorf("Unexpected ResourceAllocation: %s", *rply.ResourceAllocation) } eAttrs := &engine.AttrSProcessEventReply{ MatchedProfiles: []string{"cgrates.org:ATTR_ACNT_1001"}, AlteredFields: []string{"*req.OfficeGroup"}, CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "TestSSv1ItProcessEvent", Event: map[string]any{ utils.CGRID: "c87609aa1cb6e9529ab1836cfeeebaab7aa7ebaf", utils.Tenant: "cgrates.org", utils.Category: "call", utils.ToR: utils.MetaVoice, utils.AccountField: "1001", utils.Destination: "1002", "OfficeGroup": "Marketing", utils.OriginID: "TestSSv1It2", utils.RequestType: utils.MetaPrepaid, utils.SetupTime: "2018-01-07T17:00:00Z", utils.AnswerTime: "2018-01-07T17:00:10Z", utils.Usage: 300000000000.0, }, APIOpts: map[string]any{ utils.MetaSubsys: utils.MetaSessionS, utils.OptsAPIKey: "ses12345", utils.MetaEventType: utils.StatUpdate, "*attrProfileIDs": nil, "*rsUnits": 1., "*rsUsageID": "TestSSv1It2", }, }, } if *utils.Encoding == utils.MetaGOB { eAttrs.CGREvent.Event[utils.Usage] = 5 * time.Minute eAttrs.CGREvent.Event[utils.SetupTime], _ = utils.IfaceAsTime("2018-01-07T17:00:00Z", "") eAttrs.CGREvent.Event[utils.AnswerTime], _ = utils.IfaceAsTime("2018-01-07T17:00:10Z", "") } if !reflect.DeepEqual(*eAttrs, *rply.Attributes) { t.Errorf("expecting: %+v, received: %+v", utils.ToJSON(eAttrs), utils.ToJSON(rply.Attributes)) } } func testSes3ItThreshold1002After(t *testing.T) { var td engine.Threshold eTd := engine.Threshold{Tenant: "cgrates.org", ID: "THD_ACNT_1001", Hits: 1} if err := ses3RPC.Call(context.Background(), utils.ThresholdSv1GetThreshold, &utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "THD_ACNT_1001"}}, &td); err != nil { t.Error(err) } else if !reflect.DeepEqual(eTd.Tenant, td.Tenant) { t.Errorf("expecting: %+v, received: %+v", eTd.Tenant, td.Tenant) } else if !reflect.DeepEqual(eTd.ID, td.ID) { t.Errorf("expecting: %+v, received: %+v", eTd.ID, td.ID) } else if !reflect.DeepEqual(eTd.Hits, td.Hits) { t.Errorf("expecting: %+v, received: %+v", eTd.Hits, td.Hits) } } func testSes3ItStatMetricsAfter(t *testing.T) { var metrics map[string]string statMetrics := map[string]string{ utils.MetaACD: "5m0s", utils.MetaASR: "100%", utils.MetaTCD: "5m0s", } if err := ses3RPC.Call(context.Background(), utils.StatSv1GetQueueStringMetrics, &utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "Stat_1"}}, &metrics); err != nil { t.Error(err) } if !reflect.DeepEqual(statMetrics, metrics) { t.Errorf("expecting: %+v, received reply: %s", statMetrics, metrics) } } func testSes3ItThreshold1002After2(t *testing.T) { var td engine.Threshold eTd := engine.Threshold{Tenant: "cgrates.org", ID: "THD_ACNT_1001", Hits: 2} if err := ses3RPC.Call(context.Background(), utils.ThresholdSv1GetThreshold, &utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "THD_ACNT_1001"}}, &td); err != nil { t.Error(err) } else if !reflect.DeepEqual(eTd.Tenant, td.Tenant) { t.Errorf("expecting: %+v, received: %+v", eTd.Tenant, td.Tenant) } else if !reflect.DeepEqual(eTd.ID, td.ID) { t.Errorf("expecting: %+v, received: %+v", eTd.ID, td.ID) } else if !reflect.DeepEqual(eTd.Hits, td.Hits) { t.Errorf("expecting: %+v, received: %+v", eTd.Hits, td.Hits) } } func testSes3ItStatMetricsAfter2(t *testing.T) { var metrics map[string]string statMetrics := map[string]string{ utils.MetaACD: "5m0s", utils.MetaASR: "100%", utils.MetaTCD: "10m0s", } if err := ses3RPC.Call(context.Background(), utils.StatSv1GetQueueStringMetrics, &utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "Stat_1"}}, &metrics); err != nil { t.Error(err) } if !reflect.DeepEqual(statMetrics, metrics) { t.Errorf("expecting: %+v, received reply: %s", statMetrics, metrics) } } func testSes3ItAddVoiceBalance(t *testing.T) { attrSetBalance := utils.AttrSetBalance{ Tenant: "cgrates.org", Account: "1002", BalanceType: utils.MetaVoice, Value: 5 * float64(time.Second), Balance: map[string]any{ utils.ID: "TestDynamicDebitBalance", utils.RatingSubject: "*zero5ms", }, } var reply string if err := ses3RPC.Call(context.Background(), utils.APIerSv2SetBalance, attrSetBalance, &reply); err != nil { t.Error(err) } else if reply != utils.OK { t.Errorf("Received: %s", reply) } var acnt *engine.Account attrs := &utils.AttrGetAccount{ Tenant: "cgrates.org", Account: "1002", } if err := ses3RPC.Call(context.Background(), utils.APIerSv2GetAccount, attrs, &acnt); err != nil { t.Error(err) } else if rply := acnt.BalanceMap[utils.MetaVoice].GetTotalValue(); rply != float64(5*time.Second) { t.Errorf("Expecting: %v, received: %v", float64(5*time.Second), rply) } } func testSes3ItTerminatWithoutInit(t *testing.T) { go func() { // used in a gorutine to not block the test // because it needs to call initSession when the call for Teminate is still active args := &sessions.V1TerminateSessionArgs{ TerminateSession: true, CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "TestSesItUpdateSession", Event: map[string]any{ utils.Tenant: "cgrates.org", utils.Category: "call", utils.ToR: utils.MetaVoice, utils.OriginID: "TestTerminate", utils.RequestType: utils.MetaPrepaid, utils.AccountField: "1002", utils.Subject: "1001", utils.Destination: "1001", utils.SetupTime: time.Date(2018, time.January, 7, 16, 60, 0, 0, time.UTC), utils.AnswerTime: time.Date(2018, time.January, 7, 16, 60, 10, 0, time.UTC), utils.Usage: 2 * time.Second, }, }, } var rply string if err := ses3RPC.Call(context.Background(), utils.SessionSv1TerminateSession, args, &rply); err != nil { t.Error(err) } if rply != utils.OK { t.Errorf("Unexpected reply: %s", rply) } }() time.Sleep(time.Millisecond) args1 := &sessions.V1InitSessionArgs{ InitSession: true, CGREvent: &utils.CGREvent{ Tenant: "cgrates.org", ID: "TestSesItInitiateSession", Event: map[string]any{ utils.Tenant: "cgrates.org", utils.Category: "call", utils.ToR: utils.MetaVoice, utils.OriginID: "TestTerminate", utils.RequestType: utils.MetaPrepaid, utils.AccountField: "1002", utils.Subject: "1001", utils.Destination: "1001", utils.SetupTime: time.Date(2018, time.January, 7, 16, 60, 0, 0, time.UTC), utils.AnswerTime: time.Date(2018, time.January, 7, 16, 60, 10, 0, time.UTC), utils.Usage: 5 * time.Second, }, }, } var rply1 sessions.V1InitSessionReply if err := ses3RPC.Call(context.Background(), utils.SessionSv1InitiateSession, args1, &rply1); err != nil { t.Error(err) return } else if rply1.MaxUsage != nil && *rply1.MaxUsage != 0 { t.Errorf("Unexpected MaxUsage: %v", rply1.MaxUsage) } time.Sleep(5 * time.Millisecond) aSessions := make([]*sessions.ExternalSession, 0) if err := ses3RPC.Call(context.Background(), utils.SessionSv1GetActiveSessions, new(utils.SessionFilter), &aSessions); err == nil || err.Error() != utils.ErrNotFound.Error() { t.Error(err) } } func testSes3ItBalance(t *testing.T) { time.Sleep(10 * time.Millisecond) var acnt engine.Account attrs := &utils.AttrGetAccount{ Tenant: "cgrates.org", Account: "1002", } if err := ses3RPC.Call(context.Background(), utils.APIerSv2GetAccount, attrs, &acnt); err != nil { t.Error(err) } else if rply := acnt.BalanceMap[utils.MetaVoice].GetTotalValue(); rply != float64(3*time.Second) { t.Errorf("Expecting: %v, received: %v", 3*time.Second, rply) } } func testSes3ItCDRs(t *testing.T) { var reply string if err := ses3RPC.Call(context.Background(), utils.SessionSv1ProcessCDR, &utils.CGREvent{ Tenant: "cgrates.org", ID: "TestSesItProccesCDR", Event: map[string]any{ utils.Tenant: "cgrates.org", utils.Category: "call", utils.ToR: utils.MetaVoice, utils.OriginID: "TestTerminate", utils.RequestType: utils.MetaPrepaid, utils.AccountField: "1002", utils.Subject: "1001", utils.Destination: "1001", utils.SetupTime: time.Date(2018, time.January, 7, 16, 60, 0, 0, time.UTC), utils.AnswerTime: time.Date(2018, time.January, 7, 16, 60, 10, 0, time.UTC), utils.Usage: 2 * time.Second, }, }, &reply); err != nil { t.Error(err) } else if reply != utils.OK { t.Errorf("Received reply: %s", reply) } time.Sleep(20 * time.Millisecond) var cdrs []*engine.ExternalCDR req := utils.RPCCDRsFilter{RunIDs: []string{"CustomerCharges"}, Accounts: []string{"1002"}} if err := ses3RPC.Call(context.Background(), utils.APIerSv2GetCDRs, &req, &cdrs); err != nil { t.Error("Unexpected error: ", err.Error()) } else if len(cdrs) != 1 { t.Error("Unexpected number of CDRs returned: ", len(cdrs)) } else if cdrs[0].Usage != "2s" { t.Errorf("Unexpected CDR Usage received, cdr: %v %+v ", cdrs[0].Usage, cdrs[0]) } } func testSes3ItStopCgrEngine(t *testing.T) { if err := engine.KillEngine(100); err != nil { t.Error(err) } }