/* 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 */ package config import ( "reflect" "testing" "time" "github.com/cgrates/cgrates/utils" "github.com/cgrates/ltcache" ) func TestAsTransCacheConfig(t *testing.T) { a := &CacheCfg{ Partitions: map[string]*CacheParamCfg{ "test": { Limit: 50, TTL: 60 * time.Second, StaticTTL: true, Precache: true, }, }, } expected := map[string]*ltcache.CacheConfig{ "test": { MaxItems: 50, TTL: 60 * time.Second, StaticTTL: true, }, } reply := a.AsTransCacheConfig() if !reflect.DeepEqual(expected, reply) { t.Errorf("Expected: %+v, received: %+v", expected, utils.ToJSON(reply)) } } func TestCacheCfgloadFromJsonCfg(t *testing.T) { jsonCfg := &CacheJsonCfg{ Partitions: &map[string]*CacheParamJsonCfg{ utils.MetaDestinations: { Limit: utils.IntPointer(10), Ttl: utils.StringPointer("2"), Static_ttl: utils.BoolPointer(true), Precache: utils.BoolPointer(true), Replicate: utils.BoolPointer(true), }, }, Replication_conns: &[]string{"conn1", "conn2"}, } expected := &CacheCfg{ Partitions: map[string]*CacheParamCfg{ utils.MetaDestinations: {Limit: 10, TTL: 2, StaticTTL: true, Precache: true, Replicate: true}, }, ReplicationConns: []string{"conn1", "conn2"}, } jsnCfg := NewDefaultCGRConfig() if err := jsnCfg.cacheCfg.loadFromJSONCfg(nil); err != nil { t.Error(err) } else if err := jsnCfg.cacheCfg.loadFromJSONCfg(jsonCfg); err != nil { t.Error(err) } else { if !reflect.DeepEqual(expected.Partitions[utils.MetaDestinations], jsnCfg.cacheCfg.Partitions[utils.MetaDestinations]) { t.Errorf("Expected %+v \n, received %+v", utils.ToJSON(expected.Partitions[utils.MetaDestinations]), utils.ToJSON(jsnCfg.cacheCfg.Partitions[utils.MetaDestinations])) } else if !reflect.DeepEqual(jsnCfg.cacheCfg.ReplicationConns, expected.ReplicationConns) { t.Errorf("Expected %+v \n, received %+v", utils.ToJSON(expected.ReplicationConns), utils.ToJSON(jsnCfg.cacheCfg.ReplicationConns)) } } } func TestReplicationConnsLoadFromJsonCfg(t *testing.T) { jsonCfg := &CacheJsonCfg{ Replication_conns: &[]string{utils.MetaInternal}, } expErrMessage := "replication connection ID needs to be different than *internal" jsnCfg := NewDefaultCGRConfig() if err := jsnCfg.cacheCfg.loadFromJSONCfg(jsonCfg); err == nil || err.Error() != expErrMessage { t.Errorf("Expected %+v , recevied %+v", expErrMessage, err) } } func TestCacheParamCfgloadFromJsonCfg1(t *testing.T) { json := &CacheParamJsonCfg{ Limit: utils.IntPointer(5), Ttl: utils.StringPointer("1s"), Static_ttl: utils.BoolPointer(true), Precache: utils.BoolPointer(true), } expected := &CacheParamCfg{ Limit: 5, TTL: time.Second, StaticTTL: true, Precache: true, } rcv := new(CacheParamCfg) if err := rcv.loadFromJSONCfg(nil); err != nil { t.Error(err) } else if err := rcv.loadFromJSONCfg(json); err != nil { t.Error(err) } else if !reflect.DeepEqual(rcv, expected) { t.Errorf("Expected %+v, received %+v", utils.ToJSON(expected), utils.ToJSON(rcv)) } } func TestCacheParamCfgloadFromJsonCfg2(t *testing.T) { jsonCfg := &CacheJsonCfg{ Partitions: &map[string]*CacheParamJsonCfg{ utils.MetaDestinations: { Ttl: utils.StringPointer("1ss"), }, }, } expErrMessage := "time: unknown unit \"ss\" in duration \"1ss\"" jsnCfg := NewDefaultCGRConfig() if err := jsnCfg.cacheCfg.loadFromJSONCfg(jsonCfg); err == nil || err.Error() != expErrMessage { t.Errorf("Expected %+v \n, recevied %+v", expErrMessage, err) } } func TestAddTmpCaches(t *testing.T) { cfgJSON := &CacheJsonCfg{ Partitions: &map[string]*CacheParamJsonCfg{ utils.CacheRatingProfilesTmp: { Limit: utils.IntPointer(-1), Ttl: utils.StringPointer(time.Minute.String()), }, }, } expected := &CacheCfg{ Partitions: map[string]*CacheParamCfg{}, } expected.AddTmpCaches() json := NewDefaultCGRConfig() if err := json.cacheCfg.loadFromJSONCfg(cfgJSON); err != nil { t.Error(err) } else if !reflect.DeepEqual(expected.Partitions[utils.CacheRatingProfilesTmp], json.cacheCfg.Partitions[utils.CacheRatingProfilesTmp]) { t.Errorf("Expected %+v, received %+v", utils.ToJSON(expected.Partitions[utils.CacheRatingProfilesTmp]), utils.ToJSON(json.cacheCfg.Partitions[utils.CacheRatingProfilesTmp])) } } func TestCachesCfgAsMapInterface1(t *testing.T) { cfgJSONStr := `{ "caches":{ "partitions": { "*destinations": {"limit": 10000, "static_ttl": false, "precache": true, "remote": true, "replicate": true}, }, }, }` eMap := map[string]any{ utils.PartitionsCfg: map[string]any{ utils.MetaDestinations: map[string]any{"limit": 10000, "static_ttl": false, "precache": true, "remote": true, "replicate": true}, }, utils.ReplicationConnsCfg: []string{}, } if cgrCfg, err := NewCGRConfigFromJSONStringWithDefaults(cfgJSONStr); err != nil { t.Error(err) } else { newMap := cgrCfg.cacheCfg.AsMapInterface() if !reflect.DeepEqual(newMap[utils.PartitionsCfg].(map[string]any)[utils.MetaDestinations], eMap[utils.PartitionsCfg].(map[string]any)[utils.MetaDestinations]) { t.Errorf("Expected %+v, received %+v", eMap[utils.PartitionsCfg].(map[string]any)[utils.MetaDestinations], newMap[utils.PartitionsCfg].(map[string]any)[utils.MetaDestinations]) } } } func TestCachesCfgAsMapInterface2(t *testing.T) { cfgJSONStr := `{ "caches":{ "partitions": { "*rating_plans": {"limit": 10, "ttl": "", "static_ttl": true, "precache": true, "replicate": false}, }, "replication_conns": ["conn1", "conn2"], }, }` eMap := map[string]any{ utils.PartitionsCfg: map[string]any{ utils.MetaRatingPlans: map[string]any{"limit": 10, "ttl": "", "static_ttl": true, "precache": true}, }, utils.ReplicationConnsCfg: []string{"conn1", "conn2"}, } if cgrCfg, err := NewCGRConfigFromJSONStringWithDefaults(cfgJSONStr); err != nil { t.Error(err) } else { newMap := cgrCfg.cacheCfg.AsMapInterface() if !reflect.DeepEqual(newMap[utils.PartitionsCfg].(map[string]any)[utils.MetaRatingPlans], newMap[utils.PartitionsCfg].(map[string]any)[utils.MetaRatingPlans]) { t.Errorf("Expected %+v, received %+v", eMap[utils.PartitionsCfg].(map[string]any)[utils.MetaRatingPlans], eMap[utils.PartitionsCfg].(map[string]any)[utils.MetaRatingPlans]) } if !reflect.DeepEqual(newMap[utils.ReplicationConnsCfg], eMap[utils.ReplicationConnsCfg]) { t.Errorf("Expected %+v, received %+v", eMap[utils.ReplicationConnsCfg], newMap[utils.ReplicationConnsCfg]) } } } func TestCacheCfgClone(t *testing.T) { cs := &CacheCfg{ Partitions: map[string]*CacheParamCfg{ utils.MetaDestinations: {Limit: 10, TTL: 2, StaticTTL: true, Precache: true, Replicate: true}, }, ReplicationConns: []string{utils.ConcatenatedKey(utils.MetaInternal, utils.MetaSessionS)}, } rcv := cs.Clone() if !reflect.DeepEqual(cs, rcv) { t.Errorf("Expected: %+v\nReceived: %+v", utils.ToJSON(cs), utils.ToJSON(rcv)) } if rcv.Partitions[utils.MetaDestinations].Limit = 0; cs.Partitions[utils.MetaDestinations].Limit != 10 { t.Errorf("Expected clone to not modify the cloned") } if rcv.ReplicationConns[0] = ""; cs.ReplicationConns[0] != utils.ConcatenatedKey(utils.MetaInternal, utils.MetaSessionS) { t.Errorf("Expected clone to not modify the cloned") } } func TestCacheCfgloadFromJSONCfg(t *testing.T) { str := "test" bl := true nm := 1 slc := []string{str} tm := 1 * time.Second tms := "1s" mcp := map[string]*CacheParamCfg{ str: { Limit: nm, TTL: tm, StaticTTL: bl, Precache: bl, Remote: bl, Replicate: bl, }, } cCfg := &CacheCfg{ Partitions: map[string]*CacheParamCfg{}, ReplicationConns: []string{}, RemoteConns: []string{}, } mc := map[string]*CacheParamJsonCfg{ str: { Limit: &nm, Ttl: &tms, Static_ttl: &bl, Precache: &bl, Remote: &bl, Replicate: &bl, }, } jsn := &CacheJsonCfg{ Partitions: &mc, Replication_conns: &slc, Remote_conns: &slc, } exp := &CacheCfg{ Partitions: mcp, ReplicationConns: slc, RemoteConns: slc, } err := cCfg.loadFromJSONCfg(jsn) if err != nil { t.Error(err) } if !reflect.DeepEqual(cCfg, exp) { t.Errorf("\nexpected %s\nreceived %s\n", utils.ToJSON(exp), utils.ToJSON(cCfg)) } }