Files
cgrates/engine/storage_test.go
ionutboangiu 1a1ba55b25 Switch to ugorji/go/codec
Should be equivalent to ugocodec if enabling TimeNotBuiltin.
2024-04-16 21:10:13 +02:00

1917 lines
52 KiB
Go

/*
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 <http://www.gnu.org/licenses/>
*/
package engine
import (
"bytes"
"reflect"
"slices"
"sort"
"strings"
"testing"
"time"
"github.com/cgrates/cgrates/config"
"github.com/cgrates/cgrates/utils"
)
func TestMsgpackStructsAdded(t *testing.T) {
var a = struct{ First string }{"test"}
var b = struct {
First string
Second string
}{}
m := NewCodecMsgpackMarshaler()
buf, err := m.Marshal(&a)
if err != nil {
t.Error("error marshaling structure: ", err)
}
err = m.Unmarshal(buf, &b)
if err != nil || b.First != "test" || b.Second != "" {
t.Error("error unmarshalling structure: ", b, err)
}
}
func TestMsgpackStructsMissing(t *testing.T) {
var a = struct {
First string
Second string
}{"test1", "test2"}
var b = struct{ First string }{}
m := NewCodecMsgpackMarshaler()
buf, err := m.Marshal(&a)
if err != nil {
t.Error("error marshaling structure: ", err)
}
err = m.Unmarshal(buf, &b)
if err != nil || b.First != "test1" {
t.Error("error unmarshalling structure: ", b, err)
}
}
func TestMsgpackTime(t *testing.T) {
t1 := time.Date(2013, 8, 28, 22, 27, 30, 11, time.UTC)
m := NewCodecMsgpackMarshaler()
buf, err := m.Marshal(&t1)
if err != nil {
t.Error("error marshaling structure: ", err)
}
var t2 time.Time
err = m.Unmarshal(buf, &t2)
if err != nil || t1 != t2 || !t1.Equal(t2) {
t.Errorf("error unmarshalling structure: %#v %#v %v", t1, t2, err)
}
}
func TestStorageDestinationContainsPrefixShort(t *testing.T) {
dest, err := dm.GetDestination("NAT", false, true, utils.NonTransactional)
precision := dest.containsPrefix("0723")
if err != nil || precision != 4 {
t.Error("Error finding prefix: ", err, precision)
}
}
func TestStorageDestinationContainsPrefixLong(t *testing.T) {
dest, err := dm.GetDestination("NAT", false, true, utils.NonTransactional)
precision := dest.containsPrefix("0723045326")
if err != nil || precision != 4 {
t.Error("Error finding prefix: ", err, precision)
}
}
func TestStorageDestinationContainsPrefixNotExisting(t *testing.T) {
dest, err := dm.GetDestination("NAT", false, true, utils.NonTransactional)
precision := dest.containsPrefix("072")
if err != nil || precision != 0 {
t.Error("Error finding prefix: ", err, precision)
}
}
/*
func TestStorageCacheRefresh(t *testing.T) {
dm.SetDestination(&Destination{"T11", []string{"0"}}, utils.NonTransactional)
dm.GetDestination("T11", false, utils.NonTransactional)
dm.SetDestination(&Destination{"T11", []string{"1"}}, utils.NonTransactional)
t.Log("Test cache refresh")
err := LoadAllDataDBCache(dm)
if err != nil {
t.Error("Error cache rating: ", err)
}
d, err := dm.GetDestination("T11", false, utils.NonTransactional)
p := d.containsPrefix("1")
if err != nil || p == 0 {
t.Error("Error refreshing cache:", d)
}
}
*/
func TestStorageDisabledAccount(t *testing.T) {
acc, err := dm.GetAccount("cgrates.org:alodis")
if err != nil || acc == nil {
t.Error("Error loading disabled user account: ", err, acc)
}
if acc.Disabled != true || acc.AllowNegative != true {
t.Errorf("Error loading user account properties: %+v", acc)
}
}
// Install fails to detect them and starting server will panic, these tests will fix this
func TestStoreInterfaces(t *testing.T) {
rds := new(RedisStorage)
var _ DataDB = rds
sql := new(SQLStorage)
var _ CdrStorage = sql
}
func TestDifferentUuid(t *testing.T) {
a1, err := dm.GetAccount("cgrates.org:12345")
if err != nil {
t.Error("Error getting account: ", err)
}
a2, err := dm.GetAccount("cgrates.org:123456")
if err != nil {
t.Error("Error getting account: ", err)
}
if a1.BalanceMap[utils.MetaVoice][0].Uuid == a2.BalanceMap[utils.MetaVoice][0].Uuid ||
a1.BalanceMap[utils.MetaMonetary][0].Uuid == a2.BalanceMap[utils.MetaMonetary][0].Uuid {
t.Errorf("Identical uuids in different accounts: %+v <-> %+v", a1.BalanceMap[utils.MetaVoice][0], a1.BalanceMap[utils.MetaMonetary][0])
}
}
func TestStorageTask(t *testing.T) {
// clean previous unused tasks
for i := 0; i < 21; i++ {
dm.DataDB().PopTask()
}
if err := dm.DataDB().PushTask(&Task{Uuid: "1"}); err != nil {
t.Error("Error pushing task: ", err)
}
if err := dm.DataDB().PushTask(&Task{Uuid: "2"}); err != nil {
t.Error("Error pushing task: ", err)
}
if err := dm.DataDB().PushTask(&Task{Uuid: "3"}); err != nil {
t.Error("Error pushing task: ", err)
}
if err := dm.DataDB().PushTask(&Task{Uuid: "4"}); err != nil {
t.Error("Error pushing task: ", err)
}
if task, err := dm.DataDB().PopTask(); err != nil && task.Uuid != "1" {
t.Error("Error poping task: ", task, err)
}
if task, err := dm.DataDB().PopTask(); err != nil && task.Uuid != "2" {
t.Error("Error poping task: ", task, err)
}
if task, err := dm.DataDB().PopTask(); err != nil && task.Uuid != "3" {
t.Error("Error poping task: ", task, err)
}
if task, err := dm.DataDB().PopTask(); err != nil && task.Uuid != "4" {
t.Error("Error poping task: ", task, err)
}
if task, err := dm.DataDB().PopTask(); err == nil && task != nil {
t.Errorf("Error poping task %+v, %v ", task, err)
}
}
func TestStorageDecodeCodecMsgpackMarshaler(t *testing.T) {
type stc struct {
Name string
}
var s stc
mp := make(map[string]any)
var slc []string
var slcB []byte
var arr *[1]int
var nm int
var fl float64
var str string
var bl bool
var td time.Duration
tests := []struct {
name string
expBytes []byte
val any
decode any
rng bool
}{
{
name: "map",
expBytes: []byte{129, 164, 107, 101, 121, 49, 166, 118, 97, 108, 117, 101, 49},
val: map[string]any{"key1": "value1"},
decode: mp,
rng: true,
},
{
name: "int",
expBytes: []byte{1},
val: 1,
decode: nm,
rng: false,
},
{
name: "string",
expBytes: []byte{164, 116, 101, 115, 116},
val: "test",
decode: str,
rng: false,
},
{
name: "float64",
expBytes: []byte{203, 63, 248, 0, 0, 0, 0, 0, 0},
val: 1.5,
decode: fl,
rng: false,
},
{
name: "boolean",
expBytes: []byte{195},
val: true,
decode: bl,
rng: false,
},
{
name: "slice",
expBytes: []byte{145, 164, 118, 97, 108, 49},
val: []string{"val1"},
decode: slc,
rng: true,
},
{
name: "array",
expBytes: []byte{145, 1},
val: &[1]int{1},
decode: arr,
rng: true,
},
{
name: "struct",
expBytes: []byte{129, 164, 78, 97, 109, 101, 164, 116, 101, 115, 116},
val: stc{"test"},
decode: s,
rng: true,
},
{
name: "time duration",
expBytes: []byte{210, 59, 154, 202, 0},
val: 1 * time.Second,
decode: td,
rng: false,
},
{
name: "slice of bytes",
expBytes: []byte{162, 5, 8},
val: []byte{5, 8},
decode: slcB,
rng: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
ms := NewCodecMsgpackMarshaler()
b, err := ms.Marshal(tt.val)
if err != nil {
t.Fatal(err)
}
if !bytes.Equal(b, tt.expBytes) {
t.Fatalf("expected: %+v,\nreceived: %+v", tt.expBytes, b)
}
err = ms.Unmarshal(b, &tt.decode)
if err != nil {
t.Fatal(err)
}
if tt.rng {
if !reflect.DeepEqual(tt.decode, tt.val) {
t.Errorf("expected %v, received %v", tt.val, tt.decode)
}
} else {
if tt.decode != tt.val {
t.Errorf("expected %v, received %v", tt.val, tt.decode)
}
}
})
}
}
/************************** Benchmarks *****************************/
func GetUB() *Account {
uc := &UnitCounter{
Counters: CounterFilters{&CounterFilter{Value: 1}, &CounterFilter{Filter: &BalanceFilter{Weight: utils.Float64Pointer(20), DestinationIDs: utils.StringMapPointer(utils.NewStringMap("NAT"))}}, &CounterFilter{Filter: &BalanceFilter{Weight: utils.Float64Pointer(10), DestinationIDs: utils.StringMapPointer(utils.NewStringMap("RET"))}}},
}
at := &ActionTrigger{
ID: "some_uuid",
ThresholdValue: 100.0,
Balance: &BalanceFilter{
Type: utils.StringPointer(utils.MetaMonetary),
DestinationIDs: utils.StringMapPointer(utils.NewStringMap("NAT")),
},
Weight: 10.0,
ActionsID: "Commando",
}
var zeroTime time.Time
zeroTime = zeroTime.UTC() // for deep equal to find location
ub := &Account{
ID: "rif",
AllowNegative: true,
BalanceMap: map[string]Balances{
utils.MetaSMS: {&Balance{Value: 14, ExpirationDate: zeroTime}},
utils.MetaData: {&Balance{Value: 1024, ExpirationDate: zeroTime}},
utils.MetaVoice: {&Balance{Weight: 20, DestinationIDs: utils.NewStringMap("NAT")},
&Balance{Weight: 10, DestinationIDs: utils.NewStringMap("RET")}}},
UnitCounters: UnitCounters{utils.MetaSMS: []*UnitCounter{uc, uc}},
ActionTriggers: ActionTriggers{at, at, at},
}
return ub
}
func BenchmarkMarshallerJSONStoreRestore(b *testing.B) {
b.StopTimer()
i := &RateInterval{
Timing: &RITiming{
Months: []time.Month{time.February},
MonthDays: []int{1},
WeekDays: []time.Weekday{time.Wednesday, time.Thursday},
StartTime: "14:30:00",
EndTime: "15:00:00"}}
ap := &RatingPlan{Id: "test"}
ap.AddRateInterval("NAT", i)
ub := GetUB()
ap1 := RatingPlan{}
ub1 := &Account{}
b.StartTimer()
ms := new(JSONMarshaler)
for i := 0; i < b.N; i++ {
result, _ := ms.Marshal(ap)
ms.Unmarshal(result, ap1)
result, _ = ms.Marshal(ub)
ms.Unmarshal(result, ub1)
}
}
func BenchmarkMarshallerBSONStoreRestore(b *testing.B) {
b.StopTimer()
i := &RateInterval{
Timing: &RITiming{
Months: []time.Month{time.February},
MonthDays: []int{1},
WeekDays: []time.Weekday{time.Wednesday, time.Thursday},
StartTime: "14:30:00",
EndTime: "15:00:00"}}
ap := &RatingPlan{Id: "test"}
ap.AddRateInterval("NAT", i)
ub := GetUB()
ap1 := RatingPlan{}
ub1 := &Account{}
b.StartTimer()
ms := new(BSONMarshaler)
for i := 0; i < b.N; i++ {
result, _ := ms.Marshal(ap)
ms.Unmarshal(result, ap1)
result, _ = ms.Marshal(ub)
ms.Unmarshal(result, ub1)
}
}
func BenchmarkMarshallerJSONBufStoreRestore(b *testing.B) {
b.StopTimer()
i := &RateInterval{
Timing: &RITiming{Months: []time.Month{time.February},
MonthDays: []int{1},
WeekDays: []time.Weekday{time.Wednesday, time.Thursday},
StartTime: "14:30:00",
EndTime: "15:00:00"}}
ap := &RatingPlan{Id: "test"}
ap.AddRateInterval("NAT", i)
ub := GetUB()
ap1 := RatingPlan{}
ub1 := &Account{}
b.StartTimer()
ms := new(JSONBufMarshaler)
for i := 0; i < b.N; i++ {
result, _ := ms.Marshal(ap)
ms.Unmarshal(result, ap1)
result, _ = ms.Marshal(ub)
ms.Unmarshal(result, ub1)
}
}
func BenchmarkMarshallerGOBStoreRestore(b *testing.B) {
b.StopTimer()
i := &RateInterval{
Timing: &RITiming{Months: []time.Month{time.February},
MonthDays: []int{1},
WeekDays: []time.Weekday{time.Wednesday, time.Thursday},
StartTime: "14:30:00",
EndTime: "15:00:00"}}
ap := &RatingPlan{Id: "test"}
ap.AddRateInterval("NAT", i)
ub := GetUB()
ap1 := RatingPlan{}
ub1 := &Account{}
b.StartTimer()
ms := new(GOBMarshaler)
for i := 0; i < b.N; i++ {
result, _ := ms.Marshal(ap)
ms.Unmarshal(result, ap1)
result, _ = ms.Marshal(ub)
ms.Unmarshal(result, ub1)
}
}
func BenchmarkMarshallerCodecMsgpackStoreRestore(b *testing.B) {
b.StopTimer()
i := &RateInterval{
Timing: &RITiming{
Months: []time.Month{time.February},
MonthDays: []int{1},
WeekDays: []time.Weekday{time.Wednesday, time.Thursday},
StartTime: "14:30:00",
EndTime: "15:00:00"}}
ap := &RatingPlan{Id: "test"}
ap.AddRateInterval("NAT", i)
ub := GetUB()
ap1 := RatingPlan{}
ub1 := &Account{}
b.StartTimer()
ms := NewCodecMsgpackMarshaler()
for i := 0; i < b.N; i++ {
result, _ := ms.Marshal(ap)
ms.Unmarshal(result, ap1)
result, _ = ms.Marshal(ub)
ms.Unmarshal(result, ub1)
}
}
func BenchmarkMarshallerBincStoreRestore(b *testing.B) {
b.StopTimer()
i := &RateInterval{
Timing: &RITiming{
Months: []time.Month{time.February},
MonthDays: []int{1},
WeekDays: []time.Weekday{time.Wednesday, time.Thursday},
StartTime: "14:30:00",
EndTime: "15:00:00"}}
ap := &RatingPlan{Id: "test"}
ap.AddRateInterval("NAT", i)
ub := GetUB()
ap1 := RatingPlan{}
ub1 := &Account{}
b.StartTimer()
ms := NewBincMarshaler()
for i := 0; i < b.N; i++ {
result, _ := ms.Marshal(ap)
ms.Unmarshal(result, ap1)
result, _ = ms.Marshal(ub)
ms.Unmarshal(result, ub1)
}
}
func TestIDBRemoveIndexesDrv(t *testing.T) {
idb := NewInternalDB(nil, nil, true, map[string]*config.ItemOpt{
"chID": {
Limit: 3,
TTL: 4 * time.Minute,
StaticTTL: false,
Remote: true,
Replicate: true,
RouteID: "route",
APIKey: "api",
},
"chID2": {
Limit: 3,
TTL: 4 * time.Minute,
StaticTTL: false,
Remote: true,
Replicate: true,
RouteID: "route",
APIKey: "api",
},
},
)
idb.db.Set("chID", "itmID", true, []string{utils.EmptyString}, true, "trID")
idb.db.Set("chID2", "itmIDv", true, []string{"grpID"}, true, "trID")
if err := idb.RemoveIndexesDrv("chID", utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
}
if err := idb.RemoveIndexesDrv("chID2", "itmID", "v"); err != nil {
t.Error(err)
}
if has := idb.db.HasGroup("chID", utils.EmptyString); has {
t.Error("group should be removed")
}
}
func TestIDBGetDispatcherHostDrv(t *testing.T) {
idb := NewInternalDB(nil, nil, true, map[string]*config.ItemOpt{
utils.CacheDispatcherHosts: {
Limit: 2,
Remote: true,
},
})
dsp := &DispatcherHost{
Tenant: "cgrates.org",
}
tenant, acc := "cgrates", "acc1"
idb.db.Set(utils.CacheDispatcherHosts, utils.ConcatenatedKey(tenant, acc), dsp, []string{"id", "id3"}, true, utils.NonTransactional)
if val, err := idb.GetDispatcherHostDrv(tenant, acc); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(dsp, val) {
t.Errorf("expected %+v,received %+v", utils.ToJSON(dsp), utils.ToJSON(val))
}
}
func TestIDBRemoveDispatcherHostDrv(t *testing.T) {
idb := NewInternalDB(nil, nil, true, map[string]*config.ItemOpt{
utils.CacheDispatcherHosts: {
Limit: 2,
Remote: true,
},
})
dsp := &DispatcherHost{
Tenant: "cgrates.org",
}
tenant, acc := "cgrates", "acc1"
idb.db.Set(utils.CacheDispatcherHosts, utils.ConcatenatedKey(tenant, acc), dsp, []string{"id", "id3"}, true, utils.NonTransactional)
if err := idb.RemoveDispatcherHostDrv(tenant, acc); err != nil {
t.Error(err)
}
if _, has := idb.db.Get(utils.CacheDispatcherHosts, utils.ConcatenatedKey(tenant, acc)); has {
t.Error("should been removed")
}
}
func TestIDBSetStatQueueDrvNil(t *testing.T) {
idb := NewInternalDB(nil, nil, true, map[string]*config.ItemOpt{
utils.CacheStatQueues: {
Limit: 4,
StaticTTL: true,
},
})
ssq := &StoredStatQueue{
Tenant: "cgrates",
ID: "id",
SQItems: []SQItem{
{
EventID: "event1",
},
{
EventID: "event2",
},
},
Compressed: false,
SQMetrics: map[string][]byte{
strings.Join([]string{utils.MetaASR, "test"}, "#"): []byte("val"),
},
}
if err := idb.SetStatQueueDrv(ssq, nil); err == nil {
t.Error(err)
}
}
func TestGetTpTableIds(t *testing.T) {
cfg := config.NewDefaultCGRConfig()
cfg.DataDbCfg().Items = map[string]*config.ItemOpt{
utils.CacheTBLTPRates: {
Limit: 3,
StaticTTL: true,
},
}
db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
tpid := "*prf"
paginator := &utils.PaginatorWithSearch{
Paginator: &utils.Paginator{},
Search: "",
}
expIds := []string{"Item1", "Item2"}
filters := map[string]string{}
db.db.Set(utils.CacheTBLTPRates, "*prf:Item1", "val", []string{"grpId"}, true, utils.NonTransactional)
db.db.Set(utils.CacheTBLTPRates, "*prf:Item2", "val", []string{"grpId"}, true, utils.NonTransactional)
if val, err := db.GetTpTableIds(tpid, utils.TBLTPRates, []string{}, filters, paginator); err != nil {
t.Error(err)
} else {
sort.Slice(val, func(i, j int) bool {
return val[i] < val[j]
})
if !reflect.DeepEqual(val, expIds) {
t.Errorf("expected %v,received %v", utils.ToJSON(val), utils.ToJSON(expIds))
}
}
}
func TestIDBGetTpIds(t *testing.T) {
cfg := config.NewDefaultCGRConfig()
cfg.DataDbCfg().Items = map[string]*config.ItemOpt{
utils.CacheTBLTPRates: {
Limit: 3,
StaticTTL: true,
},
}
db := NewInternalDB(nil, nil, true, cfg.DataDbCfg().Items)
db.db.Set(utils.CacheTBLTPRates, "item_ID1", "value", []string{"grpID"}, true, utils.NonTransactional)
db.db.Set(utils.CacheTBLTPRates, "item_ID2", "value", []string{"grpID"}, true, utils.NonTransactional)
exp := []string{"item_ID1", "item_ID2"}
val, err := db.GetTpIds(utils.TBLTPRates)
if err != nil {
t.Error(err)
}
sort.Slice(val, func(i, j int) bool {
return val[i] < val[j]
})
if !reflect.DeepEqual(val, exp) {
t.Errorf("expected %v,received %v", utils.ToJSON(val), utils.ToJSON(exp))
}
}
func TestIDBTpResources(t *testing.T) {
cfg := config.NewDefaultCGRConfig()
storDB := NewInternalDB(nil, nil, false, cfg.StorDbCfg().Items)
// READ
if _, err := storDB.GetTPResources("TP1", utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
//WRITE
var snd = []*utils.TPResourceProfile{
{
TPid: "TP1",
ID: "RP1",
Weight: 10.8,
FilterIDs: []string{"FILTR_RES_1"},
ThresholdIDs: []string{"TH1"},
Stored: true,
},
{
TPid: "TP1",
ID: "RP2",
Weight: 20.6,
ThresholdIDs: []string{"TH2"},
FilterIDs: []string{"FLTR_RES_2"},
Blocker: true,
Stored: false,
},
}
if err := storDB.SetTPResources(snd); err != nil {
t.Error(err)
}
// READ
if rcv, err := storDB.GetTPResources("TP1", utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(a, b int) bool {
return rcv[a].ID < rcv[b].ID
}); !slices.Equal(snd, rcv) {
t.Errorf("Expecting: %+v, received: %+v ", utils.ToJSON(snd), utils.ToJSON(rcv))
}
// UPDATE
snd[0].Weight = 2.1
snd[1].Weight = 8.1
if err := storDB.SetTPResources(snd); err != nil {
t.Error(err)
}
// READ
if rcv, err := storDB.GetTPResources("TP1", utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(a, b int) bool {
return rcv[a].ID < rcv[b].ID
}); rcv[0].Weight != snd[0].Weight {
t.Errorf("Expecting: %+v, received: %+v ", utils.ToJSON(snd[0]), utils.ToJSON(rcv[0]))
}
//tpIDs
expIds := []string{":RP1", ":RP2"}
if tpIds, err := storDB.GetTpTableIds("TP1", utils.TBLTPResources, utils.TPDistinctIds{utils.TenantCfg, utils.IDCfg}, nil, &utils.PaginatorWithSearch{}); err != nil {
t.Error(err)
} else if slices.Sort(tpIds); !slices.Equal(tpIds, expIds) {
t.Errorf("Expected %v,Received %v", expIds, tpIds)
}
// REMOVE
if err := storDB.RemTpData("", "TP1", nil); err != nil {
t.Error(err)
}
// READ
if _, err := storDB.GetTPResources("TP1", utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
}
func TestIDBTpStats(t *testing.T) {
cfg := config.NewDefaultCGRConfig()
storDB := NewInternalDB(nil, nil, false, cfg.StorDbCfg().Items)
// READ
if _, err := storDB.GetTPStats("TP1", utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
//WRITE
eTPs := []*utils.TPStatProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "Stats1",
FilterIDs: []string{"FLTR_1"},
QueueLength: 100,
TTL: "1s",
Metrics: []*utils.MetricWithFilters{
{
MetricID: utils.MetaASR,
},
},
ThresholdIDs: []string{"*none"},
Weight: 20.0,
Stored: true,
MinItems: 1,
},
}
if err := storDB.SetTPStats(eTPs); err != nil {
t.Error(err)
}
// READ
if rcv, err := storDB.GetTPStats("TP1", utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(eTPs[0], rcv[0]) {
t.Errorf("Expecting: %+v,\n received: %+v", utils.ToJSON(eTPs[0]), utils.ToJSON(rcv[0]))
}
// UPDATE
eTPs[0].Metrics = []*utils.MetricWithFilters{
{
MetricID: utils.MetaACD,
},
}
if err := storDB.SetTPStats(eTPs); err != nil {
t.Error(err)
}
// READ
if rcv, err := storDB.GetTPStats("TP1", utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if eTPs[0].Metrics[0].MetricID != rcv[0].Metrics[0].MetricID {
t.Errorf("Expecting: %+v,\n received: %+v", utils.ToJSON(eTPs[0]), utils.ToJSON(rcv[0]))
}
// REMOVE
if err := storDB.RemTpData(utils.TBLTPStats, "TP1", nil); err != nil {
t.Error(err)
}
// READ
if ids, err := storDB.GetTPStats("TP1", utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
t.Error(utils.ToJSON(ids))
}
}
func TestIDBTPThresholds(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
//READ
if _, err := storDB.GetTPThresholds("TH1", utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
//WRITE
tpThresholds := []*utils.TPThresholdProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "Th1",
FilterIDs: []string{"*string:~*req.Account:1002", "*string:~*req.DryRun:*default"},
MaxHits: -1,
MinSleep: "1s",
Blocker: true,
Weight: 10,
ActionIDs: []string{"ACT_TOPUP_RST"},
Async: true,
},
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "Th2",
FilterIDs: []string{"*string:~*req.Destination:10"},
MaxHits: -1,
MinSleep: "1s",
Blocker: true,
Weight: 20,
ActionIDs: []string{"ACT_LOG_WARNING"},
Async: true,
},
}
if err := storDB.SetTPThresholds(tpThresholds); err != nil {
t.Errorf("Unable to set TPThresholds")
}
//READ
if rcv, err := storDB.GetTPThresholds(tpThresholds[0].TPid, utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); !slices.Equal(rcv, tpThresholds) {
t.Errorf("Expecting: %+v , Received: %+v", utils.ToJSON(tpThresholds), utils.ToJSON(rcv))
}
//UPDATE
tpThresholds[0].FilterIDs = []string{"*string:~*req.Destination:101"}
tpThresholds[1].FilterIDs = []string{"*string:~*req.Destination:101"}
if err := storDB.SetTPThresholds(tpThresholds); err != nil {
t.Errorf("Unable to set TPThresholds")
}
//READ
if rcv, err := storDB.GetTPThresholds(tpThresholds[0].TPid, utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(tpThresholds[0], rcv[0]) &&
!reflect.DeepEqual(tpThresholds[0], rcv[1]) {
t.Errorf("Expecting:\n%+v\nReceived:\n%+v\n||\n%+v",
utils.ToJSON(tpThresholds[0]), utils.ToJSON(rcv[0]), utils.ToJSON(rcv[1]))
}
//REMOVE and READ
if err := storDB.RemTpData(utils.EmptyString, tpThresholds[0].TPid, nil); err != nil {
t.Error(err)
} else if _, err := storDB.GetTPRoutes(tpThresholds[0].TPid, utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
}
func TestIDBTPFilters(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
//READ
if _, err := storDB.GetTPFilters("TP1", utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
//WRITE
tpFilters := []*utils.TPFilterProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "Filter1",
Filters: []*utils.TPFilter{
{
Type: utils.MetaString,
Element: "Account",
Values: []string{"1001", "1002"},
},
},
},
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "Filter2",
Filters: []*utils.TPFilter{
{
Type: utils.MetaPrefix,
Element: "Destination",
Values: []string{"10"},
},
},
},
}
if err := storDB.SetTPFilters(tpFilters); err != nil {
t.Errorf("Unable to set TPFilters")
}
//READ
if rcv, err := storDB.GetTPFilters(tpFilters[0].TPid, utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); !slices.Equal(rcv, tpFilters) {
t.Errorf("Expecting: %+v , Received: %+v", utils.ToJSON(tpFilters), utils.ToJSON(rcv))
}
//UPDATE and WRITE
tpFilters[1].Filters[0].Element = "Account"
if err := storDB.SetTPFilters(tpFilters); err != nil {
t.Errorf("Unable to set TPFilters")
}
//READ
if rcv, err := storDB.GetTPFilters(tpFilters[0].TPid, utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); rcv[1].Filters[0].Element != tpFilters[1].Filters[0].Element {
t.Errorf("Expecting: %+v , Received: %+v", utils.ToJSON(tpFilters[1]), utils.ToJSON(rcv[1]))
}
//REMOVE and READ
if err := storDB.RemTpData(utils.EmptyString, tpFilters[0].TPid, nil); err != nil {
t.Error(err)
} else if _, err := storDB.GetTPFilters(tpFilters[0].TPid, utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
}
func TestIDTPRoutes(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
//READ
if _, err := storDB.GetTPRoutes("TP1", utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
//WRITE
tpRoutes := []*utils.TPRouteProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "SUPL_1",
FilterIDs: []string{"*string:~*req.Accout:1007"},
Sorting: "*lowest_cost",
Routes: []*utils.TPRoute{
{
ID: "supplier1",
FilterIDs: []string{"FLTR_1"},
AccountIDs: []string{"Acc1", "Acc2"},
RatingPlanIDs: []string{"RPL_1"},
ResourceIDs: []string{"ResGroup1"},
StatIDs: []string{"Stat1"},
Weight: 10,
Blocker: false,
RouteParameters: "SortingParam1",
},
},
Weight: 20,
},
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "SUPL_2",
FilterIDs: []string{"*string:~*req.Destination:100"},
Sorting: "*lowest_cost",
Routes: []*utils.TPRoute{
{
ID: "supplier1",
FilterIDs: []string{"FLTR_1"},
AccountIDs: []string{"Acc1", "Acc2"},
RatingPlanIDs: []string{"RPL_1"},
ResourceIDs: []string{"ResGroup1"},
StatIDs: []string{"Stat1"},
Weight: 10,
Blocker: false,
RouteParameters: "SortingParam2",
},
},
Weight: 10,
},
}
if err := storDB.SetTPRoutes(tpRoutes); err != nil {
t.Errorf("Unable to set TPRoutes")
}
//READ
if rcv, err := storDB.GetTPRoutes(tpRoutes[0].TPid, utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); !slices.Equal(rcv, tpRoutes) {
t.Errorf("Expecting: %v Received: %+v", utils.ToJSON(tpRoutes), utils.ToJSON(rcv))
}
//UPDATE
tpRoutes[0].Sorting = "*higher_cost"
if err := storDB.SetTPRoutes(tpRoutes); err != nil {
t.Errorf("Unable to set TPRoutes")
}
//READ
if rcv, err := storDB.GetTPRoutes(tpRoutes[0].TPid, utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); tpRoutes[0].Sorting != rcv[0].Sorting {
t.Errorf("Expecting: %v Received: %+v", utils.ToJSON(tpRoutes[0]), utils.ToJSON(rcv[0]))
}
//REMOVE and READ
if err := storDB.RemTpData(utils.EmptyString, tpRoutes[0].TPid, nil); err != nil {
t.Error(err)
} else if _, err := storDB.GetTPRoutes(tpRoutes[0].TPid, utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
}
func TestIDBTPAttributes(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
//READ
if _, err := storDB.GetTPAttributes("TP_ID", utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
//WRITE
tpAProfile := []*utils.TPAttributeProfile{
{
TPid: "TP_ID",
Tenant: "cgrates.org",
ID: "APROFILE_ID1",
Attributes: []*utils.TPAttribute{
{
Type: utils.MetaString,
Path: utils.MetaReq + utils.NestingSep + utils.AccountField + utils.InInFieldSep,
Value: "101",
FilterIDs: []string{"*string:~*req.Account:101"},
},
{
Type: utils.MetaString,
Path: utils.MetaReq + utils.NestingSep + utils.AccountField + utils.InInFieldSep,
Value: "108",
FilterIDs: []string{"*string:~*req.Account:102"},
},
},
},
{
TPid: "TP_ID",
Tenant: "cgrates.org",
ID: "APROFILE_ID2",
Attributes: []*utils.TPAttribute{
{
Type: utils.MetaString,
Path: utils.MetaReq + utils.NestingSep + utils.Destination + utils.InInFieldSep,
Value: "12",
FilterIDs: []string{"*string:~*req.Destination:11"},
},
{
Type: utils.MetaString,
Path: utils.MetaReq + utils.NestingSep + utils.Destination + utils.InInFieldSep,
Value: "13",
FilterIDs: []string{"*string:~*req.Destination:10"},
},
},
},
}
if err := storDB.SetTPAttributes(tpAProfile); err != nil {
t.Errorf("Unable to set TPAttributeProfile:%s", err)
}
//READ
if rcv, err := storDB.GetTPAttributes(tpAProfile[0].TPid, utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); !slices.Equal(rcv, tpAProfile) {
t.Errorf("Expected %v, Received %v", utils.ToJSON(rcv), utils.ToJSON(tpAProfile))
}
//UPDATE
tpAProfile[0].Attributes[0].Value = "107"
if err := storDB.SetTPAttributes(tpAProfile); err != nil {
t.Error(err)
}
//READ
if rcv, err := storDB.GetTPAttributes(tpAProfile[0].TPid, utils.EmptyString, utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); tpAProfile[0].Attributes[0].Value != rcv[0].Attributes[0].Value {
t.Errorf("Expected %v, Received %v", utils.ToJSON(rcv[0]), utils.ToJSON(tpAProfile[0]))
}
//REMOVE and READ
if err := storDB.RemTpData(utils.EmptyString, tpAProfile[0].TPid, nil); err != nil {
t.Error(err)
} else if _, err := storDB.GetTPAttributes(tpAProfile[0].TPid, utils.EmptyString, utils.EmptyString); err != utils.ErrNotFound {
t.Error(err)
}
}
func TestIDBRemTpData(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
tpAccActions := []*utils.TPAccountActions{
{
TPid: "TP1",
LoadId: "ID",
Tenant: "cgrates.org",
Account: "1001",
ActionPlanId: "PREPAID_10",
AllowNegative: true,
Disabled: false,
},
}
if err := storDB.SetTPAccountActions(tpAccActions); err != nil {
t.Error(err)
}
tpRatingProfiles := []*utils.TPRatingProfile{
{
TPid: "TP1",
LoadId: "TEST_LOADID",
Tenant: "cgrates.org",
Category: "call",
Subject: "*any",
},
}
if err := storDB.SetTPRatingProfiles(tpRatingProfiles); err != nil {
t.Error(err)
}
if err := storDB.RemTpData(utils.TBLTPAccountActions, tpAccActions[0].TPid, map[string]string{"tenant": "cgrates.org"}); err != nil {
t.Error(err)
}
if err := storDB.RemTpData(utils.TBLTPRatingProfiles, tpRatingProfiles[0].TPid, map[string]string{"category": "call"}); err != nil {
t.Error(err)
}
}
func TestIDBTpSharedGroups(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
// READ
if _, err := storDB.GetTPSharedGroups("TP1", ""); err != utils.ErrNotFound {
t.Error(err)
}
// WRITE
var snd = []*utils.TPSharedGroups{
{
TPid: "TP1",
ID: "1",
SharedGroups: []*utils.TPSharedGroup{
{
Account: "test",
Strategy: "*lowest_cost",
RatingSubject: "test",
},
},
},
{
TPid: "TP1",
ID: "2",
SharedGroups: []*utils.TPSharedGroup{
{
Account: "test",
Strategy: "*lowest_cost",
RatingSubject: "test",
},
},
},
}
if err := storDB.SetTPSharedGroups(snd); err != nil {
t.Error(err)
}
// READ
if rcv, err := storDB.GetTPSharedGroups("TP1", ""); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); !slices.Equal(rcv, snd) {
t.Errorf("Expected %v, Received %v", utils.ToJSON(rcv), utils.ToJSON(snd))
}
// UPDATE
snd[0].SharedGroups[0].Strategy = "*highest_cost"
if err := storDB.SetTPSharedGroups(snd); err != nil {
t.Error(err)
}
// READ
if rcv, err := storDB.GetTPSharedGroups("TP1", ""); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].ID < rcv[j].ID
}); snd[0].SharedGroups[0].Strategy != rcv[0].SharedGroups[0].Strategy {
t.Errorf("Expected %v, Received %v", utils.ToJSON(rcv[0]), utils.ToJSON(snd[0]))
}
// REMOVE
if err := storDB.RemTpData("", "TP1", nil); err != nil {
t.Error(err)
}
// READ
if _, err := storDB.GetTPSharedGroups("TP1", ""); err != utils.ErrNotFound {
t.Error(err)
}
}
func TestIDBGetTpIdsEmptyCol(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
tpRoutes := []*utils.TPRouteProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "SUPL_1",
FilterIDs: []string{"*string:~*req.Accout:1007"},
Sorting: "*lowest_cost",
Routes: []*utils.TPRoute{
{
ID: "supplier1",
StatIDs: []string{"Stat1"},
Weight: 10,
Blocker: false,
RouteParameters: "SortingParam1",
},
},
Weight: 20,
}}
if err := storDB.SetTPRoutes(tpRoutes); err != nil {
t.Errorf("Unable to set TPRoutes")
}
tpFilters := []*utils.TPFilterProfile{
{
TPid: "TP2",
Tenant: "cgrates.org",
ID: "Filter1",
Filters: []*utils.TPFilter{
{
Type: utils.MetaString,
Element: "Account",
Values: []string{"1001", "1002"},
},
},
},
}
if err := storDB.SetTPFilters(tpFilters); err != nil {
t.Errorf("Unable to set TPFilters")
}
etpIds := []string{"TP1", "TP2"}
if tpIds, err := storDB.GetTpIds(utils.EmptyString); err != nil {
t.Error(err)
} else if sort.Slice(tpIds, func(i, j int) bool {
return tpIds[i] < tpIds[j]
}); !slices.Equal(tpIds, etpIds) {
t.Errorf("Expected %v,Received %v", tpIds, etpIds)
}
}
func TestIDBGetTpTableIds(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
tpAccActions := []*utils.TPAccountActions{
{
TPid: "TP1",
LoadId: "ID",
Tenant: "cgrates.org",
Account: "1001",
ActionPlanId: "PREPAID_10",
AllowNegative: true,
Disabled: false,
},
{
TPid: "TP1",
LoadId: "TEST_LOADID",
Tenant: "cgrates.org",
Account: "1002",
ActionPlanId: "PACKAGE_10_SHARED_A_5",
ActionTriggersId: "STANDARD_TRIGGERS",
AllowNegative: true,
Disabled: true,
},
}
if err := storDB.SetTPAccountActions(tpAccActions); err != nil {
t.Error(err)
}
expIds := []string{"ID", "TEST_LOADID"}
if tpIds, err := storDB.GetTpTableIds("TP1", utils.TBLTPAccountActions, utils.TPDistinctIds{utils.IDCfg}, nil, &utils.PaginatorWithSearch{}); err != nil {
t.Error(err)
} else if sort.Slice(tpIds, func(i, j int) bool {
return tpIds[i] < tpIds[j]
}); !slices.Equal(tpIds, expIds) {
t.Errorf("Expected %v,Received %v", tpIds, expIds)
}
}
func TestIDBGetTPDestinationRatesPaginator(t *testing.T) {
storDB := NewInternalDB(nil, nil, true, config.CgrConfig().StorDbCfg().Items)
destRates := []*utils.TPDestinationRate{
{
TPid: "TEST_TPID",
ID: "TEST_DSTRATE",
DestinationRates: []*utils.DestinationRate{
{
DestinationId: "TEST_DEST1",
RateId: "TEST_RATE1",
RoundingMethod: "*up",
RoundingDecimals: 4},
{
DestinationId: "TEST_DEST2",
RateId: "TEST_RATE2",
RoundingMethod: "*up",
RoundingDecimals: 4},
},
},
{
TPid: "TEST_TPID",
ID: "RT_STD_WEEKEND",
DestinationRates: []*utils.DestinationRate{
{
DestinationId: "GERMANY",
RateId: "R2",
Rate: csvr.rates["R2"],
RoundingMethod: utils.MetaRoundingMiddle,
RoundingDecimals: 4,
},
{
DestinationId: "GERMANY_O2",
RateId: "R3",
Rate: csvr.rates["R3"],
RoundingMethod: utils.MetaRoundingMiddle,
RoundingDecimals: 4,
},
},
},
}
if err := storDB.SetTPDestinationRates(destRates); err != nil {
t.Error(err)
}
if dstRates, err := storDB.GetTPDestinationRates(destRates[0].TPid, "TEST", &utils.Paginator{Limit: utils.IntPointer(1)}); err != nil {
t.Error(err)
} else if !slices.Equal(destRates[:1], dstRates) {
t.Errorf("Expected %v,Received %v", utils.ToJSON(destRates[:1]), utils.ToJSON(dstRates))
}
}
func TestIDBGetTPRatingPlans(t *testing.T) {
storDB := NewInternalDB(nil, nil, true, config.CgrConfig().StorDbCfg().Items)
ratingPlans := []*utils.TPRatingPlan{
{
TPid: "TP1",
ID: "Plan1",
RatingPlanBindings: []*utils.TPRatingPlanBinding{
{
DestinationRatesId: "DR_FREESWITCH_USERS",
TimingId: "ALWAYS",
Weight: 10,
},
},
},
{
TPid: "TP1",
ID: "Plan2",
RatingPlanBindings: []*utils.TPRatingPlanBinding{
{
DestinationRatesId: "1",
TimingId: "ALWAYS",
Weight: 0.0,
},
},
},
{
TPid: "TP1",
ID: "Plan3",
RatingPlanBindings: []*utils.TPRatingPlanBinding{
{
DestinationRatesId: "2",
TimingId: "ALWAYS",
Weight: 2,
},
},
},
}
if err := storDB.SetTPRatingPlans(ratingPlans); err != nil {
t.Error(err)
}
if rPlans, err := storDB.GetTPRatingPlans("TP1", "Plan", &utils.Paginator{Limit: utils.IntPointer(1), Offset: utils.IntPointer(1)}); err != nil {
t.Error(err)
} else if len(rPlans) != 1 {
t.Errorf("Expected slice length: 1 ,Received : %v", len(rPlans))
}
}
func TestIDBRemoveSMCost(t *testing.T) {
storDB := NewInternalDB(nil, nil, true, config.CgrConfig().StorDbCfg().Items)
// READ
if _, err := storDB.GetSMCosts("", "", "", ""); err != utils.ErrNotFound {
t.Error(err)
}
// WRITE
var snd = []*SMCost{
{
CGRID: "88ed9c38005f07576a1e1af293063833b60edcc6",
RunID: "1",
OriginHost: "host2",
OriginID: "2",
CostDetails: NewBareEventCost(),
},
{
CGRID: "88ed9c38005f07576a1e1af293063833b60edcc2",
RunID: "2",
OriginHost: "host2",
OriginID: "2",
CostDetails: NewBareEventCost(),
},
}
for _, smc := range snd {
if err := storDB.SetSMCost(smc); err != nil {
t.Error(err)
}
}
// READ
if rcv, err := storDB.GetSMCosts("", "", "host2", ""); err != nil {
t.Error(err)
} else if sort.Slice(rcv, func(i, j int) bool {
return rcv[i].CGRID < rcv[j].CGRID
}); slices.Equal(snd, rcv) {
t.Errorf("Expected %+v,Received %+v", utils.ToJSON(snd), utils.ToJSON(rcv))
}
// REMOVE
for _, smc := range snd {
if err := storDB.RemoveSMCost(smc); err != nil {
t.Error(err)
}
}
// READ
if _, err := storDB.GetSMCosts("", "", "", ""); err != utils.ErrNotFound {
t.Error(err)
}
}
func TestIDBRemoveSMC(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
// READ
if _, err := storDB.GetSMCosts("", "", "", ""); err != utils.ErrNotFound {
t.Error(err)
}
// WRITE
var snd = []*SMCost{
{
CGRID: "CGRID1",
RunID: "11",
OriginHost: "host22",
OriginID: "O1",
CostDetails: NewBareEventCost(),
},
{
CGRID: "CGRID2",
RunID: "12",
OriginHost: "host22",
OriginID: "O2",
CostDetails: NewBareEventCost(),
},
{
CGRID: "CGRID3",
RunID: "13",
OriginHost: "host23",
OriginID: "O3",
CostDetails: NewBareEventCost(),
},
}
for _, smc := range snd {
if err := storDB.SetSMCost(smc); err != nil {
t.Error(err)
}
}
// READ
if rcv, err := storDB.GetSMCosts("", "", "host22", ""); err != nil {
t.Fatal(err)
} else if len(rcv) != 2 {
t.Errorf("Expected 2 results received %v ", len(rcv))
}
// REMOVE
if err := storDB.RemoveSMCosts(&utils.SMCostFilter{
RunIDs: []string{"12", "13"},
NotRunIDs: []string{"11"},
OriginHosts: []string{"host22", "host23"},
NotOriginHosts: []string{"host21"},
}); err != nil {
t.Error(err)
}
// READ
if rcv, err := storDB.GetSMCosts("", "", "", ""); err != nil {
t.Error(err)
} else if len(rcv) != 1 {
t.Errorf("Expected 1 result received %v ", len(rcv))
}
// REMOVE
if err := storDB.RemoveSMCosts(&utils.SMCostFilter{}); err != nil {
t.Error(err)
}
// READ
if _, err := storDB.GetSMCosts("", "", "", ""); err != utils.ErrNotFound {
t.Error(err)
}
}
func TestIDBVersions(t *testing.T) {
dataDB := NewInternalDB(nil, nil, true, config.CgrConfig().DataDbCfg().Items)
if _, err := dataDB.GetVersions(utils.Accounts); err != utils.ErrNotFound {
t.Error(err)
}
vrs := Versions{
utils.Accounts: 3,
utils.Actions: 2,
utils.ActionTriggers: 2,
utils.ActionPlans: 2,
utils.SharedGroups: 2,
utils.CostDetails: 1,
}
if err := dataDB.SetVersions(vrs, false); err != nil {
t.Error(err)
}
if rcv, err := dataDB.GetVersions(""); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(vrs, rcv) {
t.Errorf("Expecting: %v, received: %v", vrs, rcv)
}
delete(vrs, utils.SharedGroups)
if err := dataDB.SetVersions(vrs, true); err != nil { // overwrite
t.Error(err)
}
if rcv, err := dataDB.GetVersions(""); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(vrs, rcv) {
t.Errorf("Expecting: %v, received: %v", vrs, rcv)
}
eAcnts := Versions{utils.Accounts: vrs[utils.Accounts]}
if rcv, err := dataDB.GetVersions(utils.Accounts); err != nil { //query one element
t.Error(err)
} else if !reflect.DeepEqual(eAcnts, rcv) {
t.Errorf("Expecting: %v, received: %v", eAcnts, rcv)
}
if _, err := dataDB.GetVersions(utils.NotAvailable); err != utils.ErrNotFound { //query non-existent
t.Error(err)
}
eAcnts[utils.Accounts] = 2
vrs[utils.Accounts] = eAcnts[utils.Accounts]
if err := dataDB.SetVersions(eAcnts, false); err != nil { // change one element
t.Error(err)
}
if rcv, err := dataDB.GetVersions(""); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(vrs, rcv) {
t.Errorf("Expecting: %v, received: %v", vrs, rcv)
}
if err = dataDB.RemoveVersions(eAcnts); err != nil { // remove one element
t.Error(err)
}
delete(vrs, utils.Accounts)
if rcv, err := dataDB.GetVersions(""); err != nil {
t.Error(err)
} else if !reflect.DeepEqual(vrs, rcv) {
t.Errorf("Expecting: %v, received: %v", vrs, rcv)
}
if err = dataDB.RemoveVersions(nil); err != nil { // remove one element
t.Error(err)
}
if _, err := dataDB.GetVersions(""); err != utils.ErrNotFound { //query non-existent
t.Error(err)
}
}
func TestIDBGetCDR(t *testing.T) {
storDB := NewInternalDB([]string{utils.AccountField, utils.CGRID, utils.OriginID, utils.RequestType, utils.Tenant, utils.Category, utils.RunID, utils.Source, utils.ToR, utils.Subject, utils.OriginHost, "ExtraHeader1", "ExtraHeader2"}, []string{"Destination", "Header2"}, false, config.CgrConfig().StorDbCfg().Items)
cdr := &CDR{
CGRID: "CGR1",
RunID: utils.MetaRaw,
OrderID: time.Now().UnixNano(),
OriginHost: "127.0.0.1",
Source: "testSetCDRs",
OriginID: "testevent1",
ToR: utils.MetaVoice,
RequestType: utils.MetaPrepaid,
Tenant: "cgrates.org",
Category: "call",
Account: "1004",
Subject: "1004",
Destination: "1007",
SetupTime: time.Date(2015, 12, 12, 14, 52, 0, 0, time.UTC),
AnswerTime: time.Date(2015, 12, 12, 14, 52, 20, 0, time.UTC),
Usage: 35 * time.Second,
ExtraFields: map[string]string{"ExtraHeader1": "ExtraVal1", "Header2": "Val2", "ExtraHeader2": "Val"},
Cost: -1,
}
if err := storDB.SetCDR(cdr, false); err != nil {
t.Error(err)
}
if cdrs, _, err := storDB.GetCDRs(&utils.CDRsFilter{CGRIDs: []string{cdr.CGRID}, OriginIDs: []string{"testevent1"}, RequestTypes: []string{utils.MetaPrepaid}, Tenants: []string{"cgrates.org"}, Categories: []string{"call"}, Subjects: []string{"1004"}, Sources: []string{"testSetCDRs"}, ToRs: []string{utils.MetaVoice}, RunIDs: []string{utils.MetaRaw}, OriginHosts: []string{"127.0.0.1"}, DestinationPrefixes: []string{"100"}, ExtraFields: map[string]string{"ExtraHeader1": "ExtraVal1", "Header2": "Val2"}, NotExtraFields: map[string]string{"ExtraHeader2": "ExtraVal2", "Header2": "Hdr"}}, false); err != nil {
t.Error(err)
} else if len(cdrs) != 1 {
t.Errorf("cdr %+v, Unexpected number of CDRs returned: %d", cdr, len(cdrs))
}
}
func TestIDBGeTps(t *testing.T) {
storDB := NewInternalDB(nil, nil, false, config.CgrConfig().StorDbCfg().Items)
resources := []*utils.TPResourceProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "ResGroup21",
FilterIDs: []string{"*string:~*req.Account:1001"},
UsageTTL: "1s",
AllocationMessage: "call",
Weight: 10,
Limit: "2",
Blocker: true,
Stored: true,
},
}
if err := storDB.SetTPResources(resources); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPResources("TP1", "cgrates.org", resources[0].ID); err != nil {
t.Error(err)
}
stats := []*utils.TPStatProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "Stats1",
FilterIDs: []string{"FLTR_1"},
QueueLength: 100,
TTL: "1s",
Metrics: []*utils.MetricWithFilters{
{
MetricID: utils.MetaASR,
},
},
ThresholdIDs: []string{"*none"},
Weight: 20.0,
Stored: true,
MinItems: 1,
},
}
if err := storDB.SetTPStats(stats); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPStats("TP1", "cgrates.org", stats[0].ID); err != nil {
t.Error(err)
}
thresholds := []*utils.TPThresholdProfile{
{TPid: "TP1",
Tenant: "cgrates.org",
ID: "TH_1",
FilterIDs: []string{"FLTR_1", "FLTR_2"},
MaxHits: 12,
MinHits: 10,
MinSleep: "1s",
Blocker: false,
Weight: 20.0,
ActionIDs: []string{"WARN3"},
}}
if err := storDB.SetTPThresholds(thresholds); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPThresholds("TP1", "cgrates.org", thresholds[0].ID); err != nil {
t.Error(err)
}
filters := []*utils.TPFilterProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "Filter2",
Filters: []*utils.TPFilter{
{
Type: utils.MetaPrefix,
Element: "Destination",
Values: []string{"10"},
},
},
},
}
if err := storDB.SetTPFilters(filters); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPFilters("TP1", "cgrates.org", filters[0].ID); err != nil {
t.Error(err)
}
routes := []*utils.TPRouteProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "SUPL_1",
FilterIDs: []string{"*string:~*req.Accout:1007"},
Sorting: "*lowest_cost",
Routes: []*utils.TPRoute{
{
ID: "supplier1",
StatIDs: []string{"Stat1"},
Weight: 10,
Blocker: false,
RouteParameters: "SortingParam1",
},
},
Weight: 20,
},
}
if err := storDB.SetTPRoutes(routes); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPRoutes("TP1", "cgrates.org", routes[0].ID); err != nil {
t.Error(err)
}
attributes := []*utils.TPAttributeProfile{
{TPid: "TP1",
Tenant: "cgrates.org",
ID: "ALS1",
Contexts: []string{"con1"},
FilterIDs: []string{"FLTR_ACNT_dan", "FLTR_DST_DE"},
Attributes: []*utils.TPAttribute{
{
Path: utils.MetaReq + utils.NestingSep + "FL1",
Value: "Al1",
},
},
Weight: 20,
}}
if err := storDB.SetTPAttributes(attributes); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPAttributes("TP1", "cgrates.org", attributes[0].ID); err != nil {
t.Error(err)
}
chargers := []*utils.TPChargerProfile{
{TPid: "TP1",
Tenant: "cgrates.org",
ID: "Chrgs",
FilterIDs: []string{"FLTR_ACNT_dan", "FLTR_DST_DE"},
RunID: utils.MetaDefault,
AttributeIDs: []string{"Attr1", "Attr2"},
Weight: 20},
}
if err := storDB.SetTPChargers(chargers); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPChargers("TP1", "cgrates.org", chargers[0].ID); err != nil {
t.Error(err)
}
dispatcherProfiles := []*utils.TPDispatcherProfile{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "Dsp1",
FilterIDs: []string{"*string:~*req.Account:1002"},
ActivationInterval: &utils.TPActivationInterval{
ActivationTime: "2014-07-29T15:00:00Z",
ExpiryTime: "",
},
Strategy: utils.MetaFirst,
Weight: 10,
},
}
if err := storDB.SetTPDispatcherProfiles(dispatcherProfiles); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPDispatcherProfiles("TP1", "cgrates.org", dispatcherProfiles[0].ID); err != nil {
t.Error(err)
}
dispatcherHosts := []*utils.TPDispatcherHost{
{
TPid: "TP1",
Tenant: "cgrates.org",
ID: "ALL",
Conn: &utils.TPDispatcherHostConn{
Address: "127.0.0.1:2012",
Transport: utils.MetaJSON,
TLS: true,
},
},
}
if err := storDB.SetTPDispatcherHosts(dispatcherHosts); err != nil {
t.Error(err)
}
if _, err := storDB.GetTPDispatcherHosts("TP1", "cgrates.org", dispatcherHosts[0].ID); err != nil {
t.Error(err)
}
}
func TestIDBGetAllActionPlanDrv(t *testing.T) {
dataDB := NewInternalDB(nil, nil, true, config.CgrConfig().DataDbCfg().Items)
acPln := []struct {
key string
apl *ActionPlan
}{
{
key: " MORE_MINUTES",
apl: &ActionPlan{
Id: "MORE_MINUTES",
AccountIDs: utils.StringMap{"1001": true},
},
},
{
key: "PACKAGE_10_SHARED_A_5",
apl: &ActionPlan{
Id: "PACKAGE_10_SHARED_A_5",
AccountIDs: utils.StringMap{
"cgrates.org:1001": true,
},
},
},
}
for _, tt := range acPln {
t.Run(tt.key, func(t *testing.T) {
if err := dataDB.SetActionPlanDrv(tt.key, tt.apl); err != nil {
t.Error(err)
}
})
}
if apl, err := dataDB.GetAllActionPlansDrv(); err != nil {
t.Error(err)
} else if len(apl) != 2 {
t.Errorf("Expected : 2,Received %v", len(apl))
}
}
func TestComposeURI(t *testing.T) {
tests := []struct {
name string
scheme string
host string
port string
db string
user string
pass string
expected string
parseErr bool
}{
{
name: "multiple nodes",
scheme: "mongodb",
host: "clusternode1:1230,clusternode2:1231,clusternode3",
port: "1232",
db: "cgrates",
user: "user",
pass: "pass",
expected: "mongodb://user:pass@clusternode1:1230,clusternode2:1231,clusternode3:1232/cgrates",
},
{
name: "no port",
scheme: "mongodb",
host: "localhost:1234",
port: "0",
db: "cgrates",
user: "user",
pass: "pass",
expected: "mongodb://user:pass@localhost:1234/cgrates",
},
{
name: "with port",
scheme: "mongodb",
host: "localhost",
port: "1234",
db: "cgrates",
user: "user",
pass: "pass",
expected: "mongodb://user:pass@localhost:1234/cgrates",
},
{
name: "no password",
scheme: "mongodb",
host: "localhost",
port: "1234",
db: "cgrates",
user: "user",
pass: "",
expected: "mongodb://localhost:1234/cgrates",
},
{
name: "no db",
scheme: "mongodb",
host: "localhost",
port: "1234",
db: "",
user: "user",
pass: "pass",
expected: "mongodb://user:pass@localhost:1234",
},
{
name: "different scheme",
scheme: "mongodb+srv",
host: "cgr.abcdef.mongodb.net",
port: "0",
db: "?retryWrites=true&w=majority",
user: "user",
pass: "pass",
expected: "mongodb+srv://user:pass@cgr.abcdef.mongodb.net/?retryWrites=true&w=majority",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
url := composeMongoURI(tt.scheme, tt.host, tt.port, tt.db, tt.user, tt.pass)
if url != tt.expected {
t.Errorf("expected %v,\nreceived %v", tt.expected, url)
}
})
}
}