Files
cgrates/apier/v1/attributes_it_test.go
2026-01-14 11:07:12 +01:00

2235 lines
75 KiB
Go

//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 Affero 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 Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>
*/
package v1
import (
"path"
"reflect"
"slices"
"sort"
"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/utils"
)
var (
alsPrfCfgPath string
alsPrfCfg *config.CGRConfig
attrSRPC *birpc.Client
alsPrf *engine.AttributeProfileWithAPIOpts
alsPrfConfigDIR string //run tests for specific configuration
sTestsAlsPrf = []func(t *testing.T){
testAttributeSInitCfg,
testAttributeSInitDataDb,
testAttributeSResetStorDb,
testAttributeSStartEngine,
testAttributeSRPCConn,
testAttributeSLoadFromFolder,
testAttributeSGetAttributeForEvent,
testAttributeSGetAttributeForEventNotFound,
testAttributeSGetAttributeForEventWithMetaAnyContext,
testAttributeSProcessEvent,
testAttributeSProcessEventNotFound,
testAttributeSProcessEventMissing,
testAttributeSProcessEventWithNoneSubstitute,
testAttributeSProcessEventWithNoneSubstitute2,
testAttributeSProcessEventWithNoneSubstitute3,
testAttributeSProcessEventWithHeader,
//testAttributeSProcessEventSentryPeer,
testAttributeSGetAttPrfIDs,
testAttributeSSetAlsPrfBrokenReference,
testAttributeSGetAlsPrfBeforeSet,
testAttributeSSetAlsPrf,
testAttributeSUpdateAlsPrf,
testAttributeSRemAlsPrf,
testAttributeSSetAlsPrf2,
testAttributeSSetAlsPrf3,
testAttributeSSetAlsPrf4,
testAttributeSPing,
testAttributeSProcessEventWithSearchAndReplace,
testAttributeSProcessWithMultipleRuns,
testAttributeSProcessWithMultipleRuns2,
testAttributeSGetAttributeProfileIDsCount,
testAttributeSSetAttributeWithEmptyPath,
testAttributeSSetAlsPrfWithoutTenant,
testAttributeSRmvAlsPrfWithoutTenant,
testAttributeSKillEngine,
//start test for cache options
testAttributeSInitCfg,
testAttributeSInitDataDb,
testAttributeSResetStorDb,
testAttributeSStartEngine,
testAttributeSRPCConn,
testAttributeSCachingMetaNone,
testAttributeSCachingMetaLoad,
testAttributeSCachingMetaReload1,
testAttributeSCachingMetaReload2,
testAttributeSCachingMetaRemove,
testAttributeSCacheOpts,
testAttributeSKillEngine,
//cache test
testAttributeSInitCfg,
testAttributeSInitDataDb,
testAttributeSResetStorDb,
testAttributeSStartEngine,
testAttributeSRPCConn,
testAttributeSCacheTestProcessEventNotFound,
testAttributeSCacheTestSetProfile,
testAttributeSCacheTestProcessEventNotFound,
testAttributeSCacheTestReload,
testAttributeSCacheTestProcessEventFound,
testAttributeSKillEngine,
}
)
// Test start here
func TestAttributeSIT(t *testing.T) {
attrsTests := sTestsAlsPrf
switch *utils.DBType {
case utils.MetaInternal:
attrsTests = sTestsAlsPrf[:29]
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
alsPrfCfgPath = path.Join(*utils.DataDir, "conf", "samples", alsPrfConfigDIR)
alsPrfCfg, err = config.NewCGRConfigFromPath(alsPrfCfgPath)
if err != nil {
t.Error(err)
}
}
func testAttributeSInitDataDb(t *testing.T) {
if err := engine.InitDataDB(alsPrfCfg); err != nil {
t.Fatal(err)
}
}
// Wipe out the cdr database
func testAttributeSResetStorDb(t *testing.T) {
if err := engine.InitStorDb(alsPrfCfg); err != nil {
t.Fatal(err)
}
}
// Start CGR Engine
func testAttributeSStartEngine(t *testing.T) {
if _, err := engine.StopStartEngine(alsPrfCfgPath, *utils.WaitRater); err != nil {
t.Fatal(err)
}
}
// Connect rpc client to rater
func testAttributeSRPCConn(t *testing.T) {
var err error
attrSRPC, err = newRPCClient(alsPrfCfg.ListenCfg()) // We connect over JSON so we can also troubleshoot if needed
if err != nil {
t.Fatal(err)
}
}
func testAttributeSGetAlsPrfBeforeSet(t *testing.T) {
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ApierTest"}},
&reply); err == nil || err.Error() != utils.ErrNotFound.Error() {
t.Error(err)
}
}
func testAttributeSLoadFromFolder(t *testing.T) {
var reply string
attrs := &utils.AttrLoadTpFromFolder{FolderPath: path.Join(*utils.DataDir, "tariffplans", "oldtutorial")}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1LoadTariffPlanFromFolder, attrs, &reply); err != nil {
t.Error(err)
}
time.Sleep(100 * time.Millisecond)
}
func testAttributeSGetAttributeForEvent(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSGetAttributeForEvent",
Event: map[string]any{
utils.AccountField: "1007",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
}
eAttrPrf := &engine.AttributeProfile{
Tenant: ev.Tenant,
ID: "ATTR_1",
FilterIDs: []string{"*string:~*req.Account:1007"},
Contexts: []string{utils.MetaCDRs, utils.MetaSessionS},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 1, 14, 0, 0, 0, 0, time.UTC)},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + utils.AccountField,
Value: config.NewRSRParsersMustCompile("1001", utils.InfieldSep),
Type: utils.MetaConstant,
FilterIDs: []string{},
},
{
Path: utils.MetaReq + utils.NestingSep + utils.Subject,
Value: config.NewRSRParsersMustCompile("1001", utils.InfieldSep),
Type: utils.MetaConstant,
FilterIDs: []string{},
},
},
Weight: 10.0,
}
if *utils.Encoding == utils.MetaGOB {
eAttrPrf.Attributes[0].FilterIDs = nil
eAttrPrf.Attributes[1].FilterIDs = nil
}
eAttrPrf.Compile()
var attrReply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1GetAttributeForEvent,
ev, &attrReply); err != nil {
t.Fatal(err)
}
attrReply.Compile() // Populate private variables in RSRParsers
sort.Strings(attrReply.Contexts)
if !reflect.DeepEqual(eAttrPrf, attrReply) {
t.Errorf("Expecting: %+v, received: %+v", utils.ToJSON(eAttrPrf), utils.ToJSON(attrReply))
}
ev.Tenant = utils.EmptyString
ev.ID = "randomID"
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1GetAttributeForEvent,
ev, &attrReply); err != nil {
t.Fatal(err)
}
attrReply.Compile() // Populate private variables in RSRParsers
sort.Strings(attrReply.Contexts)
if !reflect.DeepEqual(eAttrPrf, attrReply) {
t.Errorf("Expecting: %+v, received: %+v", utils.ToJSON(eAttrPrf), utils.ToJSON(attrReply))
}
}
func testAttributeSGetAttributeForEventNotFound(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSGetAttributeForEventWihMetaAnyContext",
Event: map[string]any{
utils.AccountField: "dan",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaCDRs,
},
}
eAttrPrf2 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: ev.Tenant,
ID: "ATTR_3",
FilterIDs: []string{"*string:~*req.Account:dan"},
Contexts: []string{utils.MetaSessionS},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 1, 14, 0, 0, 0, 0, time.UTC)},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + utils.AccountField,
Value: config.NewRSRParsersMustCompile("1001", utils.InfieldSep),
},
},
Weight: 10.0,
},
}
eAttrPrf2.Compile()
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, eAttrPrf2, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_3"}}, &reply); err != nil {
t.Fatal(err)
}
reply.Compile()
if !reflect.DeepEqual(eAttrPrf2.AttributeProfile, reply) {
t.Errorf("Expecting : %+v, received: %+v", eAttrPrf2, reply)
}
var attrReply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1GetAttributeForEvent,
ev, &attrReply); err == nil || err.Error() != utils.ErrNotFound.Error() {
t.Error(err)
}
}
func testAttributeSGetAttributeForEventWithMetaAnyContext(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSGetAttributeForEventWihMetaAnyContext",
Event: map[string]any{
utils.AccountField: "dan",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaCDRs,
},
}
eAttrPrf2 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: ev.Tenant,
ID: "ATTR_2",
FilterIDs: []string{"*string:~*req.Account:dan"},
Contexts: []string{utils.MetaAny},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 1, 14, 0, 0, 0, 0, time.UTC)},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + utils.AccountField,
Value: config.NewRSRParsersMustCompile("1001", utils.InfieldSep),
},
},
Weight: 10.0,
},
}
eAttrPrf2.Compile()
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, eAttrPrf2, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_2"}}, &reply); err != nil {
t.Fatal(err)
}
reply.Compile()
if !reflect.DeepEqual(eAttrPrf2.AttributeProfile, reply) {
t.Errorf("Expecting : %+v, received: %+v", eAttrPrf2.AttributeProfile, reply)
}
var attrReply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1GetAttributeForEvent,
ev, &attrReply); err != nil {
t.Fatal(err)
}
attrReply.Compile()
if !reflect.DeepEqual(eAttrPrf2.AttributeProfile, attrReply) {
t.Errorf("Expecting: %s, received: %s", utils.ToJSON(eAttrPrf2.AttributeProfile), utils.ToJSON(attrReply))
}
}
func testAttributeSProcessEvent(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSProcessEvent",
Event: map[string]any{
utils.AccountField: "1007",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
}
eRply := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:ATTR_1"},
AlteredFields: []string{utils.MetaReq + utils.NestingSep + utils.AccountField,
utils.MetaReq + utils.NestingSep + utils.Subject},
CGREvent: &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSProcessEvent",
Event: map[string]any{
utils.AccountField: "1001",
utils.Subject: "1001",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err != nil {
t.Error(err)
} else {
sort.Strings(eRply.AlteredFields)
sort.Strings(rplyEv.AlteredFields)
if !reflect.DeepEqual(eRply, &rplyEv) { // second for reversed order of attributes
t.Errorf("Expecting: %s, received: %s",
utils.ToJSON(eRply), utils.ToJSON(rplyEv))
}
}
ev.Tenant = ""
ev.ID = "randomID"
eRply.ID = "randomID"
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err != nil {
t.Error(err)
} else {
sort.Strings(eRply.AlteredFields)
sort.Strings(rplyEv.AlteredFields)
if !reflect.DeepEqual(eRply, &rplyEv) { // second for reversed order of attributes
t.Errorf("Expecting: %s, received: %s",
utils.ToJSON(eRply), utils.ToJSON(rplyEv))
}
}
}
func testAttributeSProcessEventNotFound(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSProcessEventNotFound",
Event: map[string]any{
utils.AccountField: "Inexistent",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err == nil ||
err.Error() != utils.ErrNotFound.Error() {
t.Error(err)
}
}
func testAttributeSProcessEventMissing(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSProcessEvent",
Event: map[string]any{
utils.AccountField: "NonExist",
utils.Category: "*attributes",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err == nil ||
err.Error() != "MANDATORY_IE_MISSING: [Category]" {
t.Error(err)
}
}
func testAttributeSProcessEventWithNoneSubstitute(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSWithNoneSubstitute",
Event: map[string]any{
utils.AccountField: "1008",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
}
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "cgrates.org",
ID: "AttributeWithNonSubstitute",
Contexts: []string{utils.MetaSessionS, utils.MetaCDRs},
FilterIDs: []string{"*string:~*req.Account:1008"},
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{
{
FilterIDs: []string{"*string:~*req.Account:1008"},
Path: utils.MetaReq + utils.NestingSep + utils.AccountField,
Value: config.NewRSRParsersMustCompile("1001", utils.InfieldSep),
},
{
Path: utils.MetaReq + utils.NestingSep + utils.Subject,
Value: config.NewRSRParsersMustCompile(utils.MetaRemove, utils.InfieldSep),
},
},
Weight: 20,
},
}
alsPrf.Compile()
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
eRply := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:AttributeWithNonSubstitute"},
AlteredFields: []string{utils.MetaReq + utils.NestingSep + utils.AccountField,
utils.MetaReq + utils.NestingSep + utils.Subject},
CGREvent: &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSWithNoneSubstitute",
Event: map[string]any{
utils.AccountField: "1001",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err != nil {
t.Error(err)
}
sort.Strings(rplyEv.AlteredFields)
if !reflect.DeepEqual(eRply, &rplyEv) {
t.Errorf("Expecting: %s, received: %s",
utils.ToJSON(eRply), utils.ToJSON(rplyEv))
}
}
func testAttributeSProcessEventWithNoneSubstitute2(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSWithNoneSubstitute",
Event: map[string]any{
utils.AccountField: "1008",
utils.Subject: "1008",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
}
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "cgrates.org",
ID: "AttributeWithNonSubstitute",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Account:1008"},
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{
{
FilterIDs: []string{"*string:~*req.Account:1008"},
Path: utils.MetaReq + utils.NestingSep + utils.AccountField,
Value: config.NewRSRParsersMustCompile("1001", utils.InfieldSep),
},
{
Path: utils.MetaReq + utils.NestingSep + utils.Subject,
Value: config.NewRSRParsersMustCompile(utils.MetaRemove, utils.InfieldSep),
},
},
Weight: 20,
},
}
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
eRply := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:AttributeWithNonSubstitute"},
AlteredFields: []string{"*req.Account", "*req.Subject"},
CGREvent: &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSWithNoneSubstitute",
Event: map[string]any{
utils.AccountField: "1001",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
},
}
eRply2 := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:AttributeWithNonSubstitute"},
AlteredFields: []string{utils.MetaReq + utils.NestingSep + utils.Subject,
utils.MetaReq + utils.NestingSep + utils.AccountField},
CGREvent: &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSWithNoneSubstitute",
Event: map[string]any{
utils.AccountField: "1001",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(eRply, &rplyEv) &&
!reflect.DeepEqual(eRply2, &rplyEv) {
t.Errorf("Expecting: %s, received: %s",
utils.ToJSON(eRply), utils.ToJSON(rplyEv))
}
}
func testAttributeSProcessEventWithNoneSubstitute3(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSWithNoneSubstitute",
Event: map[string]any{
utils.AccountField: "1008",
utils.Subject: "1001",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
}
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "cgrates.org",
ID: "AttributeWithNonSubstitute",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Account:1008"},
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{
{
FilterIDs: []string{"*string:~*req.Account:1008"},
Path: utils.MetaReq + utils.NestingSep + utils.AccountField,
Value: config.NewRSRParsersMustCompile("1001", utils.InfieldSep),
},
{
FilterIDs: []string{"*string:~*req.Subject:1008"},
Path: utils.MetaReq + utils.NestingSep + utils.Subject,
Value: config.NewRSRParsersMustCompile(utils.MetaRemove, utils.InfieldSep),
},
},
Weight: 20,
},
}
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
eRply := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:AttributeWithNonSubstitute"},
AlteredFields: []string{"*req.Account"},
CGREvent: &utils.CGREvent{
Tenant: "cgrates.org",
ID: "testAttributeSWithNoneSubstitute",
Event: map[string]any{
utils.AccountField: "1001",
utils.Subject: "1001",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(eRply, &rplyEv) {
t.Errorf("Expecting: %s, received: %s",
utils.ToJSON(eRply), utils.ToJSON(rplyEv))
}
}
func testAttributeSProcessEventWithHeader(t *testing.T) {
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_Header",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Field1:Value1"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field2",
Value: config.NewRSRParsersMustCompile("~*req.Field1", utils.InfieldSep),
},
},
Blocker: true,
Weight: 5,
},
}
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
attrArgs := &utils.CGREvent{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "HeaderEventForAttribute",
Event: map[string]any{
"Field1": "Value1",
},
APIOpts: map[string]any{
utils.OptsAttributesProcessRuns: 1.,
utils.OptsContext: utils.MetaSessionS,
},
}
eRply := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:ATTR_Header"},
AlteredFields: []string{"*req.Field2"},
CGREvent: &utils.CGREvent{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "HeaderEventForAttribute",
Event: map[string]any{
"Field1": "Value1",
"Field2": "Value1",
},
APIOpts: map[string]any{
utils.OptsAttributesProcessRuns: 1.,
utils.OptsContext: utils.MetaSessionS,
},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
attrArgs, &rplyEv); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(eRply, &rplyEv) {
t.Errorf("Expecting: %s, received: %s",
utils.ToJSON(eRply), utils.ToJSON(rplyEv))
}
}
// func testAttributeSProcessEventSentryPeer(t *testing.T) {
// attrPrf1 := &engine.AttributeProfileWithAPIOpts{
// AttributeProfile: &engine.AttributeProfile{
// Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
// ID: "ATTR_SENTRY_IP",
// Contexts: []string{utils.MetaAny},
// FilterIDs: []string{"*sentrypeer:~*req.IP:*ip"},
// ActivationInterval: &utils.ActivationInterval{
// ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
// },
// Attributes: []*engine.Attribute{
// {
// Path: utils.MetaReq + utils.NestingSep + "IP",
// Value: config.NewRSRParsersMustCompile("BLACKLIST_IP", utils.InfieldSep),
// Type: utils.MetaConstant,
// },
// },
// Blocker: true,
// Weight: 5,
// },
// }
// var result string
// if err := attrSRPC.Call(context.Background(),utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
// t.Error(err)
// } else if result != utils.OK {
// t.Error("Unexpected reply returned", result)
// }
// attrPrf2 := &engine.AttributeProfileWithAPIOpts{
// AttributeProfile: &engine.AttributeProfile{
// Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
// ID: "ATTR_SENTRY_NUMBER",
// Contexts: []string{utils.MetaAny},
// FilterIDs: []string{"*sentrypeer:~*req.Destination:*number"},
// ActivationInterval: &utils.ActivationInterval{
// ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
// },
// Attributes: []*engine.Attribute{
// {
// Path: utils.MetaReq + utils.NestingSep + utils.Destination,
// Value: config.NewRSRParsersMustCompile("BLACKLIST_DEST", utils.InfieldSep),
// Type: utils.MetaConstant,
// },
// },
// Blocker: true,
// Weight: 5,
// },
// }
// if err := attrSRPC.Call(context.Background(),utils.APIerSv1SetAttributeProfile, attrPrf2, &result); err != nil {
// t.Error(err)
// } else if result != utils.OK {
// t.Error("Unexpected reply returned", result)
// }
// var reply *engine.AttributeProfile
// if err := attrSRPC.Call(context.Background(),utils.APIerSv1GetAttributeProfile,
// utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_SENTRY_IP"}}, &reply); err != nil {
// t.Fatal(err)
// }
// reply.Compile()
// if !reflect.DeepEqual(attrPrf1.AttributeProfile, reply) {
// t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, reply)
// }
// if err := attrSRPC.Call(context.Background(),utils.APIerSv1GetAttributeProfile,
// utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_SENTRY_NUMBER"}}, &reply); err != nil {
// t.Fatal(err)
// }
// reply.Compile()
// if !reflect.DeepEqual(attrPrf2.AttributeProfile, reply) {
// t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, reply)
// }
// attrArgs := &utils.CGREvent{
// Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
// ID: "HAttribute",
// Event: map[string]any{
// "IP": "45.155.91.135",
// utils.Destination: "46423112877",
// },
// APIOpts: map[string]any{
// utils.OptsAttributesProcessRuns: 1.,
// utils.OptsContext: utils.MetaAny,
// },
// }
// eRply := &engine.AttrSProcessEventReply{
// MatchedProfiles: []string{"cgrates.org:ATTR_SENTRY_IP"},
// AlteredFields: []string{"*req.IP"},
// CGREvent: &utils.CGREvent{
// Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
// ID: "HAttribute",
// Event: map[string]any{
// "IP": "BLACKLIST_IP",
// utils.Destination: "46423112877",
// },
// APIOpts: map[string]any{
// utils.OptsAttributesProcessRuns: 1.,
// utils.OptsContext: utils.MetaAny,
// },
// },
// }
// var rplyEv engine.AttrSProcessEventReply
// if err := attrSRPC.Call(context.Background(),utils.AttributeSv1ProcessEvent,
// attrArgs, &rplyEv); err != nil {
// t.Error(err)
// } else if !reflect.DeepEqual(eRply, &rplyEv) {
// t.Errorf("Expecting: %s, received: %s",
// utils.ToJSON(eRply), utils.ToJSON(rplyEv))
// }
// }
func testAttributeSGetAttPrfIDs(t *testing.T) {
expected := []string{"ATTR_2", "ATTR_RAD", "ATTR_PASS", "ATTR_1", "ATTR_3", "ATTR_Header", "AttributeWithNonSubstitute"}
var result []string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{}, &result); err != nil {
t.Error(err)
} else if len(expected) != len(result) {
t.Errorf("Expecting : %+v, received: %+v", expected, result)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{Tenant: "cgrates.org"}, &result); err != nil {
t.Error(err)
} else if len(expected) != len(result) {
t.Errorf("Expecting : %+v, received: %+v", expected, result)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{
Tenant: "cgrates.org",
Paginator: utils.Paginator{Limit: utils.IntPointer(10)},
}, &result); err != nil {
t.Error(err)
} else if 10 < len(result) {
t.Errorf("Expecting : %+v, received: %+v", expected, result)
}
expected = []string{"ATTR_1", "ATTR_2", "ATTR_3", "ATTR_Header", "ATTR_PASS", "ATTR_RAD"}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{
Tenant: "cgrates.org",
Search: "ATTR", // test for prefix
Paginator: utils.Paginator{Limit: utils.IntPointer(10)},
}, &result); err != nil {
t.Error(err)
}
slices.Sort(result)
if !reflect.DeepEqual(expected, result) {
t.Errorf("Expecting : %+v, received: %+v", expected, result)
}
expected = []string{"ATTR_2"}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{
Tenant: "cgrates.org",
Search: "_2", // test for suffix
Paginator: utils.Paginator{Limit: utils.IntPointer(10)},
}, &result); err != nil {
t.Error(err)
}
if !reflect.DeepEqual(expected, result) {
t.Errorf("Expecting : %+v, received: %+v", expected, result)
}
expected = []string{"ATTR_1", "ATTR_2", "ATTR_3", "ATTR_Header", "ATTR_PASS", "ATTR_RAD"}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{
Tenant: "cgrates.org",
Search: "TTR_", // test for middle of body
Paginator: utils.Paginator{Limit: utils.IntPointer(10)},
}, &result); err != nil {
t.Error(err)
}
slices.Sort(result)
if !reflect.DeepEqual(expected, result) {
t.Errorf("Expecting : %+v, received: %+v", expected, result)
}
expected = []string{}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{
Tenant: "cgrates.org",
Search: "non matching string", // test for non matching
Paginator: utils.Paginator{Limit: utils.IntPointer(10)},
}, &result); err == nil || err.Error() != utils.ErrNotFound.Error() {
t.Errorf("expected error <%v>, received <%v>", utils.ErrNotFound, err)
}
}
func testAttributeSSetAlsPrfBrokenReference(t *testing.T) {
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "cgrates.org",
ID: "ApierTest",
Contexts: []string{utils.MetaSessionS, utils.MetaCDRs},
FilterIDs: []string{"FLTR_ACNT_danBroken", "FLTR_DST_DEBroken"},
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 + "FL1",
Value: config.NewRSRParsersMustCompile("Al1", utils.InfieldSep),
},
},
Weight: 20,
},
}
var result string
expErr := "SERVER_ERROR: broken reference to filter: <FLTR_ACNT_danBroken> for item with ID: cgrates.org:ApierTest"
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err == nil || err.Error() != expErr {
t.Fatalf("Expected error: %q, received: %q", expErr, err)
}
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ApierTest"}}, &reply); err == nil ||
err.Error() != utils.ErrNotFound.Error() {
t.Fatal(err)
}
}
func testAttributeSSetAlsPrf(t *testing.T) {
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "cgrates.org",
ID: "ApierTest",
Contexts: []string{utils.MetaSessionS, utils.MetaCDRs},
FilterIDs: []string{"FLTR_ACNT_dan", "FLTR_DST_DE"},
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 + "FL1",
Value: config.NewRSRParsersMustCompile("Al1", utils.InfieldSep),
},
},
Weight: 20,
},
}
alsPrf.Compile()
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ApierTest"}}, &reply); err != nil {
t.Fatal(err)
}
reply.Compile()
if !reflect.DeepEqual(alsPrf.AttributeProfile, reply) {
t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, reply)
}
}
func testAttributeSUpdateAlsPrf(t *testing.T) {
alsPrf.Attributes = []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "FL1",
Value: config.NewRSRParsersMustCompile("Al1", utils.InfieldSep),
},
{
Path: utils.MetaReq + utils.NestingSep + "FL2",
Value: config.NewRSRParsersMustCompile("Al2", utils.InfieldSep),
},
}
alsPrf.Compile()
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ApierTest"}}, &reply); err != nil {
t.Fatal(err)
}
reply.Compile()
sort.Strings(reply.FilterIDs)
sort.Strings(alsPrf.AttributeProfile.FilterIDs)
sort.Strings(reply.Contexts)
sort.Strings(alsPrf.AttributeProfile.Contexts)
if !reflect.DeepEqual(alsPrf.AttributeProfile, reply) {
t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, reply)
}
}
func testAttributeSRemAlsPrf(t *testing.T) {
var resp string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: alsPrf.Tenant, ID: alsPrf.ID}}, &resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if alsPrfConfigDIR == "tutinternal" { // do not double remove the profile
return
}
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ApierTest"}},
&reply); err == nil || err.Error() != utils.ErrNotFound.Error() {
t.Error(err)
}
// remove twice shoud return not found
resp = ""
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: alsPrf.Tenant, ID: alsPrf.ID}}, &resp); err.Error() != utils.ErrNotFound.Error() {
t.Errorf("Expected error: %v received: %v", utils.ErrNotFound, err)
}
}
func testAttributeSSetAlsPrf2(t *testing.T) {
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "golant",
ID: "ATTR_972587832508_SESSIONAUTH",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Account:972587832508"},
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 + utils.Subject,
Value: config.RSRParsers{
&config.RSRParser{
Rules: "roam",
},
},
},
},
Blocker: false,
Weight: 10,
},
}
alsPrf.Compile()
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "golant", ID: "ATTR_972587832508_SESSIONAUTH"}}, &reply); err != nil {
t.Fatal(err)
}
reply.Compile()
if !reflect.DeepEqual(alsPrf.AttributeProfile, reply) {
t.Errorf("Expecting : %+v, received: %+v", alsPrf.AttributeProfile, reply)
}
}
func testAttributeSSetAlsPrf3(t *testing.T) {
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "golant",
ID: "ATTR_972587832508_SESSIONAUTH",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:Account:972587832508"},
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 + utils.Subject,
Value: config.RSRParsers{
&config.RSRParser{
Rules: "",
},
},
},
},
Blocker: false,
Weight: 10,
},
}
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err == nil {
t.Error(err)
}
}
func testAttributeSSetAlsPrf4(t *testing.T) {
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "golant",
ID: "ATTR_972587832508_SESSIONAUTH",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Account:972587832508"},
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 + utils.Subject,
Value: config.RSRParsers{
&config.RSRParser{},
},
},
},
Blocker: false,
Weight: 10,
},
}
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &result); err == nil {
t.Error(err)
}
}
func testAttributeSPing(t *testing.T) {
var resp string
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1Ping, new(utils.CGREvent), &resp); err != nil {
t.Error(err)
} else if resp != utils.Pong {
t.Error("Unexpected reply returned", resp)
}
}
func testAttributeSProcessEventWithSearchAndReplace(t *testing.T) {
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_Search_and_replace",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Category:call"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Category",
Value: config.NewRSRParsersMustCompile("~*req.Category:s/(.*)/${1}_suffix/", utils.InfieldSep),
},
},
Blocker: true,
Weight: 10,
},
}
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
attrArgs := &utils.CGREvent{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "HeaderEventForAttribute",
Event: map[string]any{
"Category": "call",
},
APIOpts: map[string]any{
utils.OptsAttributesProcessRuns: 1.,
utils.OptsContext: utils.MetaSessionS,
},
}
eRply := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:ATTR_Search_and_replace"},
AlteredFields: []string{"*req.Category"},
CGREvent: &utils.CGREvent{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "HeaderEventForAttribute",
Event: map[string]any{
"Category": "call_suffix",
},
APIOpts: map[string]any{
utils.OptsAttributesProcessRuns: 1.,
utils.OptsContext: utils.MetaSessionS,
},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
attrArgs, &rplyEv); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(eRply, &rplyEv) {
t.Errorf("Expecting: %s, received: %s",
utils.ToJSON(eRply), utils.ToJSON(rplyEv))
}
}
func testAttributeSProcessWithMultipleRuns(t *testing.T) {
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.InitialField:InitialValue"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
}
attrPrf2 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_2",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Field1:Value1"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field2",
Value: config.NewRSRParsersMustCompile("Value2", utils.InfieldSep),
},
},
Weight: 20,
},
}
attrPrf3 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_3",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.NotFound:NotFound"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field3",
Value: config.NewRSRParsersMustCompile("Value3", utils.InfieldSep),
},
},
Weight: 30,
},
}
// Add attribute in DM
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf2, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf3, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
attrArgs := &utils.CGREvent{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: utils.GenUUID(),
Event: map[string]any{
"InitialField": "InitialValue",
},
APIOpts: map[string]any{
utils.OptsAttributesProcessRuns: 4,
utils.OptsContext: utils.MetaSessionS,
},
}
eRply := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:ATTR_1", "cgrates.org:ATTR_2", "cgrates.org:ATTR_1", "cgrates.org:ATTR_2"},
AlteredFields: []string{"*req.Field1", "*req.Field2"},
CGREvent: &utils.CGREvent{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: utils.GenUUID(),
Event: map[string]any{
"InitialField": "InitialValue",
"Field1": "Value1",
"Field2": "Value2",
},
APIOpts: map[string]any{
utils.OptsAttributesProcessRuns: 4,
},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
attrArgs, &rplyEv); err != nil {
t.Fatal(err)
} else if !reflect.DeepEqual(eRply.MatchedProfiles, rplyEv.MatchedProfiles) {
t.Errorf("Expecting %+v, received: %+v", eRply.MatchedProfiles, rplyEv.MatchedProfiles)
}
sort.Strings(rplyEv.AlteredFields)
if !reflect.DeepEqual(eRply.AlteredFields, rplyEv.AlteredFields) {
t.Errorf("Expecting %+v, received: %+v", eRply.AlteredFields, rplyEv.AlteredFields)
} else if !reflect.DeepEqual(eRply.CGREvent.Event, rplyEv.CGREvent.Event) {
t.Errorf("Expecting %+v, received: %+v", eRply.CGREvent.Event, rplyEv.CGREvent.Event)
}
}
func testAttributeSProcessWithMultipleRuns2(t *testing.T) {
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.InitialField:InitialValue"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
}
attrPrf2 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_2",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Field1:Value1"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field2",
Value: config.NewRSRParsersMustCompile("Value2", utils.InfieldSep),
},
},
Weight: 20,
},
}
attrPrf3 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_3",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Field2:Value2"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field3",
Value: config.NewRSRParsersMustCompile("Value3", utils.InfieldSep),
},
},
Weight: 30,
},
}
// Add attributeProfiles
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf2, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf3, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
attrArgs := &utils.CGREvent{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: utils.GenUUID(),
Event: map[string]any{
"InitialField": "InitialValue",
},
APIOpts: map[string]any{
utils.OptsAttributesProcessRuns: 4,
utils.OptsContext: utils.MetaSessionS,
},
}
eRply := &engine.AttrSProcessEventReply{
MatchedProfiles: []string{"cgrates.org:ATTR_1", "cgrates.org:ATTR_2", "cgrates.org:ATTR_3", "cgrates.org:ATTR_2"},
AlteredFields: []string{"*req.Field1", "*req.Field2", "*req.Field3"},
CGREvent: &utils.CGREvent{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: utils.GenUUID(),
Event: map[string]any{
"InitialField": "InitialValue",
"Field1": "Value1",
"Field2": "Value2",
"Field3": "Value3",
},
APIOpts: map[string]any{
utils.OptsAttributesProcessRuns: 4,
},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
attrArgs, &rplyEv); err != nil {
t.Fatal(err)
} else if !reflect.DeepEqual(eRply.MatchedProfiles, rplyEv.MatchedProfiles) {
t.Errorf("Expecting %+v, received: %+v", eRply.MatchedProfiles, rplyEv.MatchedProfiles)
}
sort.Strings(rplyEv.AlteredFields)
if !reflect.DeepEqual(eRply.AlteredFields, rplyEv.AlteredFields) {
t.Errorf("Expecting %+v, received: %+v", eRply.AlteredFields, rplyEv.AlteredFields)
} else if !reflect.DeepEqual(eRply.CGREvent.Event, rplyEv.CGREvent.Event) {
t.Errorf("Expecting %+v, received: %+v", eRply.CGREvent.Event, rplyEv.CGREvent.Event)
}
}
func testAttributeSGetAttributeProfileIDsCount(t *testing.T) {
var reply int
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{}, &reply); err != nil {
t.Error(err)
} else if reply != 8 {
t.Errorf("Expecting: 8, received: %+v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != 8 {
t.Errorf("Expecting: 8, received: %+v", reply)
}
var resp string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{
Tenant: "cgrates.org",
ID: "ATTR_1",
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}}, &resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != 7 {
t.Errorf("Expecting: 7, received: %+v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{
Tenant: "cgrates.org",
ID: "ATTR_2",
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}}, &resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != 6 {
t.Errorf("Expecting: 6, received: %+v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{
Tenant: "cgrates.org",
ID: "ATTR_3",
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}}, &resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != 5 {
t.Errorf("Expecting: 5, received: %+v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{
Tenant: "cgrates.org",
ID: "ATTR_Header",
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}}, &resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != 4 {
t.Errorf("Expecting: 4, received: %+v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{
Tenant: "cgrates.org",
ID: "ATTR_PASS",
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}}, &resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != 3 {
t.Errorf("Expecting: 3, received: %+v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{
Tenant: "cgrates.org",
ID: "ATTR_Search_and_replace",
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}},
&resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != 2 {
t.Errorf("Expecting: 2, received: %+v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{
Tenant: "cgrates.org",
ID: "ATTR_RAD",
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}},
&resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err != nil {
t.Error(err)
} else if reply != 1 {
t.Errorf("Expecting: 1, received: %+v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{
Tenant: "cgrates.org",
ID: "AttributeWithNonSubstitute",
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}},
&resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileCount,
&utils.TenantWithAPIOpts{Tenant: "cgrates.org"}, &reply); err == nil || err.Error() != utils.ErrNotFound.Error() {
t.Error(err)
}
}
func testAttributeSKillEngine(t *testing.T) {
if err := engine.KillEngine(100); err != nil {
t.Error(err)
}
}
// Start tests for caching
func testAttributeSCachingMetaNone(t *testing.T) {
//*none option should not add attribute in cache only in Datamanager
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.InitialField:InitialValue"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaNone,
},
}
// set the profile
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply bool
argsCache := utils.ArgsGetCacheItem{
CacheID: utils.CacheAttributeProfiles,
ItemID: "cgrates.org:ATTR_1",
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1HasItem, argsCache, &reply); err != nil {
t.Error(err)
} else if reply {
t.Errorf("Expected: false, received:%v", reply)
}
var rcvKeys []string
argsCache2 := utils.ArgsGetCacheItemIDs{
CacheID: utils.CacheAttributeProfiles,
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1GetItemIDs, argsCache2, &rcvKeys); err == nil ||
err.Error() != utils.ErrNotFound.Error() {
t.Fatalf("Expected error: %s received error: %s and reply: %v ",
utils.ErrNotFound, err.Error(), rcvKeys)
}
//check in dataManager
expected := []string{"ATTR_1"}
var rcvIDs []string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{Tenant: "cgrates.org"}, &rcvIDs); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(expected, rcvIDs) {
t.Errorf("Expecting : %+v, received: %+v", expected, rcvIDs)
}
}
func testAttributeSCachingMetaLoad(t *testing.T) {
//*load option should add attribute in cache and in Datamanager
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.InitialField:InitialValue"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaLoad,
},
}
// set the profile
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply bool
argsCache := utils.ArgsGetCacheItem{
CacheID: utils.CacheAttributeProfiles,
ItemID: "cgrates.org:ATTR_1",
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1HasItem, argsCache, &reply); err != nil {
t.Error(err)
} else if !reply {
t.Errorf("Expected: true, received:%v", reply)
}
var rcvKeys []string
expectedIDs := []string{"cgrates.org:ATTR_1"}
argsCache2 := utils.ArgsGetCacheItemIDs{
CacheID: utils.CacheAttributeProfiles,
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1GetItemIDs, argsCache2, &rcvKeys); err != nil {
t.Fatal(err)
} else if !reflect.DeepEqual(rcvKeys, expectedIDs) {
t.Errorf("Expecting : %+v, received: %+v", expectedIDs, rcvKeys)
}
rcvKeys = nil
argsCache2 = utils.ArgsGetCacheItemIDs{
CacheID: utils.CacheAttributeFilterIndexes,
}
expectedIDs = []string{"cgrates.org:*sessions:*string:*req.InitialField:InitialValue"}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1GetItemIDs, argsCache2, &rcvKeys); err != nil {
t.Fatal(err)
} else if !reflect.DeepEqual(rcvKeys, expectedIDs) {
t.Errorf("Expecting : %+v, received: %+v", expectedIDs, rcvKeys)
}
//check in dataManager
expected := []string{"ATTR_1"}
var rcvIDs []string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{Tenant: "cgrates.org"}, &rcvIDs); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(expected, rcvIDs) {
t.Errorf("Expecting : %+v, received: %+v", expected, rcvIDs)
}
//remove from cache and DataManager the profile
var resp string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: attrPrf1.Tenant, ID: attrPrf1.ID},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
}}, &resp); err != nil {
t.Error(err)
} else if resp != utils.OK {
t.Error("Unexpected reply returned", resp)
}
argsCache = utils.ArgsGetCacheItem{
CacheID: utils.CacheAttributeProfiles,
ItemID: "cgrates.org:ATTR_1",
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1HasItem, argsCache, &reply); err != nil {
t.Error(err)
} else if reply {
t.Errorf("Expected: false, received:%v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1GetItemIDs, argsCache2, &rcvKeys); err == nil ||
err.Error() != utils.ErrNotFound.Error() {
t.Fatalf("Expected error: %s received error: %s and reply: %v ",
utils.ErrNotFound, err, rcvKeys)
}
//check in dataManager
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{Tenant: "cgrates.org"}, &rcvIDs); err == nil ||
err.Error() != utils.ErrNotFound.Error() {
t.Fatalf("Expected error: %s received error: %s and reply: %v ",
utils.ErrNotFound, err, rcvIDs)
}
}
func testAttributeSCachingMetaReload1(t *testing.T) {
//*reload add the attributes in cache if was there before
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.InitialField:InitialValue"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaReload,
},
}
// set the profile
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply bool
argsCache := utils.ArgsGetCacheItem{
CacheID: utils.CacheAttributeProfiles,
ItemID: "cgrates.org:ATTR_1",
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1HasItem, argsCache, &reply); err != nil {
t.Error(err)
} else if reply {
t.Errorf("Expected: false, received:%v", reply)
}
var rcvKeys []string
argsCache2 := utils.ArgsGetCacheItemIDs{
CacheID: utils.CacheAttributeProfiles,
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1GetItemIDs, argsCache2, &rcvKeys); err == nil ||
err.Error() != utils.ErrNotFound.Error() {
t.Fatalf("Expected error: %s received error: %s and reply: %v ",
utils.ErrNotFound, err, rcvKeys)
}
//check in dataManager
expected := []string{"ATTR_1"}
var rcvIDs []string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{Tenant: "cgrates.org"}, &rcvIDs); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(expected, rcvIDs) {
t.Errorf("Expecting : %+v, received: %+v", expected, rcvIDs)
}
}
func testAttributeSCachingMetaReload2(t *testing.T) {
//add cache with *load option
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.InitialField:InitialValue"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaLoad,
},
}
// set the profile
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_1"}}, &reply); err != nil {
t.Fatal(err)
}
attrPrf1.Compile()
reply.Compile()
if !reflect.DeepEqual(attrPrf1.AttributeProfile, reply) {
t.Errorf("Expecting : %+v, received: %+v", attrPrf1.AttributeProfile, reply)
}
//add cache with *reload option
// should overwrite the first
attrPrf2 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Test:Test"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaReload,
},
}
// set the profile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf2, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{Tenant: "cgrates.org", ID: "ATTR_1"}}, &reply); err != nil {
t.Fatal(err)
}
attrPrf2.Compile()
reply.Compile()
if !reflect.DeepEqual(attrPrf2.AttributeProfile, reply) {
t.Errorf("Expecting : %+v, received: %+v", attrPrf2.AttributeProfile, reply)
}
}
func testAttributeSCachingMetaRemove(t *testing.T) {
//add cache with *load option
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.InitialField:InitialValue"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaLoad,
},
}
// set the profile
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
var reply bool
argsCache := utils.ArgsGetCacheItem{
CacheID: utils.CacheAttributeProfiles,
ItemID: "cgrates.org:ATTR_1",
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1HasItem, argsCache, &reply); err != nil {
t.Error(err)
} else if !reply {
t.Errorf("Expected: true, received:%v", reply)
}
var rcvKeys []string
expectedIDs := []string{"cgrates.org:ATTR_1"}
argsCache2 := utils.ArgsGetCacheItemIDs{
CacheID: utils.CacheAttributeProfiles,
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1GetItemIDs, argsCache2, &rcvKeys); err != nil {
t.Fatal(err)
} else if !reflect.DeepEqual(rcvKeys, expectedIDs) {
t.Errorf("Expecting : %+v, received: %+v", expectedIDs, rcvKeys)
}
// add with *remove cache option
// should delete it from cache
attrPrf2 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_1",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.Test:Test"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaRemove,
},
}
// set the profile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf2, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1HasItem, argsCache, &reply); err != nil {
t.Error(err)
} else if reply {
t.Errorf("Expected: false, received:%v", reply)
}
if err := attrSRPC.Call(context.Background(), utils.CacheSv1GetItemIDs, argsCache2, &rcvKeys); err == nil ||
err.Error() != utils.ErrNotFound.Error() {
t.Fatalf("Expected error: %s received error: %s and reply: %v ",
utils.ErrNotFound, err.Error(), rcvKeys)
}
//check in dataManager
expected := []string{"ATTR_1"}
var rcvIDs []string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfileIDs, &utils.PaginatorWithTenant{Tenant: "cgrates.org"}, &rcvIDs); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(expected, rcvIDs) {
t.Errorf("Expecting : %+v, received: %+v", expected, rcvIDs)
}
}
func testAttributeSSetAttributeWithEmptyPath(t *testing.T) {
eAttrPrf2 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: "cgrates.org",
ID: "ATTR_3",
FilterIDs: []string{"*string:~*req.Account:dan"},
Contexts: []string{utils.MetaSessionS},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 1, 14, 0, 0, 0, 0, time.UTC)},
Attributes: []*engine.Attribute{
{
Value: config.NewRSRParsersMustCompile("1001", utils.InfieldSep),
},
},
Weight: 10.0,
},
}
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, eAttrPrf2, &result); err == nil {
t.Errorf("Expected error received nil")
}
}
func testAttributeSCacheOpts(t *testing.T) {
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_WITH_OPTS",
Contexts: []string{utils.MetaSessionS},
FilterIDs: []string{"*string:~*req.InitialField:InitialValue"},
ActivationInterval: &utils.ActivationInterval{
ActivationTime: time.Date(2014, 7, 14, 14, 25, 0, 0, time.UTC),
},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
"Method": "SetAttributeProfile",
"CustomField": "somethingCustom",
},
}
// set the profile
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
}
func testAttributeSSetAlsPrfWithoutTenant(t *testing.T) {
var reply string
alsPrf = &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
ID: "ApierTest1",
Contexts: []string{utils.MetaSessionS, utils.MetaCDRs},
FilterIDs: []string{"FLTR_ACNT_dan", "FLTR_DST_DE"},
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 + "FL1",
Value: config.NewRSRParsersMustCompile("Al1", utils.InfieldSep),
},
},
Weight: 20,
},
}
alsPrf.Compile()
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, alsPrf, &reply); err != nil {
t.Error(err)
} else if reply != utils.OK {
t.Error("Unexpected reply returned", reply)
}
alsPrf.AttributeProfile.Tenant = "cgrates.org"
var result *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{ID: "ApierTest1"}},
&result); err != nil {
t.Error(err)
} else if result.Compile(); !reflect.DeepEqual(alsPrf.AttributeProfile, result) {
t.Errorf("Expected %+v, received %+v", utils.ToJSON(alsPrf.AttributeProfile), utils.ToJSON(result))
}
}
func testAttributeSRmvAlsPrfWithoutTenant(t *testing.T) {
var reply string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1RemoveAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{ID: "ApierTest1"}},
&reply); err != nil {
t.Error(err)
} else if reply != utils.OK {
t.Error("Unexpected reply returned", reply)
}
var result *engine.AttributeProfile
if err := attrSRPC.Call(context.Background(), utils.APIerSv1GetAttributeProfile,
&utils.TenantIDWithAPIOpts{TenantID: &utils.TenantID{ID: "ApierTest1"}},
&result); err == nil || err.Error() != utils.ErrNotFound.Error() {
t.Error(err)
}
}
func testAttributeSCacheTestProcessEventNotFound(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
Event: map[string]any{
utils.AccountField: "1007",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
utils.OptsAttributesProfileIDs: []string{"ATTR_CACHE"},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err == nil || err.Error() != utils.ErrNotFound.Error() {
t.Error(err)
}
}
func testAttributeSCacheTestProcessEventFound(t *testing.T) {
ev := &utils.CGREvent{
Tenant: "cgrates.org",
Event: map[string]any{
utils.AccountField: "1007",
utils.Destination: "+491511231234",
},
APIOpts: map[string]any{
utils.OptsContext: utils.MetaSessionS,
utils.OptsAttributesProfileIDs: []string{"ATTR_CACHE"},
},
}
var rplyEv engine.AttrSProcessEventReply
if err := attrSRPC.Call(context.Background(), utils.AttributeSv1ProcessEvent,
ev, &rplyEv); err != nil {
t.Error(err)
}
}
func testAttributeSCacheTestSetProfile(t *testing.T) {
//*none option should not add attribute in cache only in Datamanager
attrPrf1 := &engine.AttributeProfileWithAPIOpts{
AttributeProfile: &engine.AttributeProfile{
Tenant: config.CgrConfig().GeneralCfg().DefaultTenant,
ID: "ATTR_CACHE",
Contexts: []string{utils.MetaSessionS},
Attributes: []*engine.Attribute{
{
Path: utils.MetaReq + utils.NestingSep + "Field1",
Value: config.NewRSRParsersMustCompile("Value1", utils.InfieldSep),
},
},
Weight: 10,
},
APIOpts: map[string]any{
utils.CacheOpt: utils.MetaNone,
},
}
// set the profile
var result string
if err := attrSRPC.Call(context.Background(), utils.APIerSv1SetAttributeProfile, attrPrf1, &result); err != nil {
t.Error(err)
} else if result != utils.OK {
t.Error("Unexpected reply returned", result)
}
}
func testAttributeSCacheTestReload(t *testing.T) {
cache := &utils.AttrReloadCacheWithAPIOpts{
AttributeProfileIDs: []string{"cgrates.org:ATTR_CACHE"},
}
var reply string
if err := attrSRPC.Call(context.Background(), utils.CacheSv1ReloadCache, cache, &reply); err != nil {
t.Error("Got error on CacheSv1.ReloadCache: ", err.Error())
} else if reply != utils.OK {
t.Error("Calling CacheSv1.ReloadCache got reply: ", reply)
}
}