Files
cgrates/config/ips_test.go
2025-10-13 09:57:41 +02:00

393 lines
12 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 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 config
import (
"errors"
"reflect"
"slices"
"testing"
"time"
"github.com/cgrates/birpc/context"
"github.com/cgrates/cgrates/utils"
)
func TestIPsCfgLoad(t *testing.T) {
t.Run("successful load", func(t *testing.T) {
jsnCfg := &IPsJsonCfg{
Enabled: utils.BoolPointer(false),
IndexedSelects: utils.BoolPointer(true),
StoreInterval: utils.StringPointer("72h"),
PrefixIndexedFields: &[]string{"*req.prefix1", "*req.prefix2"},
SuffixIndexedFields: &[]string{"*req.suffix1"},
ExistsIndexedFields: &[]string{"*req.exists1", "*req.exists2"},
NotExistsIndexedFields: &[]string{"*req.notexists1"},
NestedFields: utils.BoolPointer(false),
}
expected := &IPsCfg{
Enabled: false,
IndexedSelects: true,
StoreInterval: 72 * time.Hour,
PrefixIndexedFields: &[]string{"*req.prefix1", "*req.prefix2"},
SuffixIndexedFields: &[]string{"*req.suffix1"},
ExistsIndexedFields: &[]string{"*req.exists1", "*req.exists2"},
NotExistsIndexedFields: &[]string{"*req.notexists1"},
NestedFields: false,
Opts: &IPsOpts{AllocationID: nil, TTL: nil},
}
cfg := &IPsCfg{Opts: &IPsOpts{}}
ctx := &context.Context{}
db := &mockDb{
GetSectionF: func(_ *context.Context, section string, out any) error {
if section != IPsJSON {
return errors.New("unexpected section")
}
*out.(*IPsJsonCfg) = *jsnCfg
return nil
},
}
if err := cfg.Load(ctx, db, nil); err != nil {
t.Errorf("unexpected error: %v", err)
} else if !reflect.DeepEqual(expected, cfg) {
t.Errorf("expected:\n%v\ngot:\n%v", utils.ToJSON(expected), utils.ToJSON(cfg))
}
})
t.Run("GetSection returns error", func(t *testing.T) {
cfg := &IPsCfg{Opts: &IPsOpts{}}
ctx := &context.Context{}
db := &mockDb{
GetSectionF: func(_ *context.Context, _ string, _ any) error {
return errors.New("error")
},
}
err := cfg.Load(ctx, db, nil)
if err == nil {
t.Error("expected error, got nil")
}
})
}
func TestIPsCfgSName(t *testing.T) {
var cfg IPsCfg
expected := IPsJSON
if got := cfg.SName(); got != expected {
t.Errorf("SName() = %v; want %v", got, expected)
}
}
func TestIPsCfgloadFromJSONCfg(t *testing.T) {
t.Run("full JSON config loads correctly", func(t *testing.T) {
jsonCfg := &IPsJsonCfg{
Enabled: utils.BoolPointer(true),
IndexedSelects: utils.BoolPointer(false),
StoreInterval: utils.StringPointer("1h30m"),
StringIndexedFields: &[]string{"*req.s1"},
PrefixIndexedFields: &[]string{"*req.p1"},
SuffixIndexedFields: &[]string{"*req.sfx"},
ExistsIndexedFields: &[]string{"*req.e1"},
NotExistsIndexedFields: &[]string{"*req.ne1"},
NestedFields: utils.BoolPointer(true),
Opts: &IPsOptsJson{},
}
expected := &IPsCfg{
Enabled: true,
IndexedSelects: false,
StoreInterval: time.Hour + 30*time.Minute,
StringIndexedFields: &[]string{"*req.s1"},
PrefixIndexedFields: &[]string{"*req.p1"},
SuffixIndexedFields: &[]string{"*req.sfx"},
ExistsIndexedFields: &[]string{"*req.e1"},
NotExistsIndexedFields: &[]string{"*req.ne1"},
NestedFields: true,
Opts: &IPsOpts{},
}
cfg := &IPsCfg{Opts: &IPsOpts{}}
err := cfg.loadFromJSONCfg(jsonCfg)
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
if !reflect.DeepEqual(cfg, expected) {
t.Errorf("expected:\n%v\ngot:\n%v", utils.ToJSON(expected), utils.ToJSON(cfg))
}
})
t.Run("nil input returns nil error", func(t *testing.T) {
cfg := &IPsCfg{Opts: &IPsOpts{}}
if err := cfg.loadFromJSONCfg(nil); err != nil {
t.Errorf("expected nil error, got: %v", err)
}
})
t.Run("invalid store interval returns error", func(t *testing.T) {
jsonCfg := &IPsJsonCfg{
StoreInterval: utils.StringPointer("invalidDuration"),
}
cfg := &IPsCfg{Opts: &IPsOpts{}}
err := cfg.loadFromJSONCfg(jsonCfg)
if err == nil {
t.Error("expected error for invalid duration, got nil")
}
})
}
func TestIPsCfgClone(t *testing.T) {
orig := IPsCfg{
Enabled: true,
IndexedSelects: false,
StoreInterval: 42,
NestedFields: true,
Opts: &IPsOpts{
AllocationID: []*DynamicStringOpt{{value: "alloc1"}},
TTL: []*DynamicDurationOpt{{value: 123}},
},
StringIndexedFields: &[]string{"c", "g"},
PrefixIndexedFields: &[]string{"r"},
SuffixIndexedFields: &[]string{"a", "t"},
ExistsIndexedFields: &[]string{"e"},
NotExistsIndexedFields: &[]string{"s", "t"},
}
clone := orig.Clone()
if !reflect.DeepEqual(clone, &orig) {
t.Errorf("Clone() = %+v; want %+v", clone, orig)
}
*clone.StringIndexedFields = append(*clone.StringIndexedFields, "z")
*clone.PrefixIndexedFields = append(*clone.PrefixIndexedFields, "y")
clone.Enabled = false
clone.Opts.AllocationID[0].value = "changed"
if reflect.DeepEqual(clone, &orig) {
t.Errorf("Clone() is not a deep copy; modifications to clone affected original")
}
if slices.Contains(*clone.StringIndexedFields, "z") && !slices.Contains(*orig.StringIndexedFields, "z") {
} else {
t.Errorf("StringIndexedFields slice not cloned properly")
}
if slices.Contains(*clone.PrefixIndexedFields, "y") && !slices.Contains(*orig.PrefixIndexedFields, "y") {
} else {
t.Errorf("PrefixIndexedFields slice not cloned properly")
}
if clone.Opts == orig.Opts {
t.Errorf("Opts.Clone() did not create a new instance")
}
}
func TestIPsCfgCloneSection(t *testing.T) {
orig := IPsCfg{
Enabled: true,
IndexedSelects: false,
StoreInterval: 42,
NestedFields: true,
Opts: &IPsOpts{},
}
cloneSection := orig.CloneSection()
clone, ok := cloneSection.(*IPsCfg)
if !ok {
t.Fatalf("CloneSection() did not return *IPsCfg, got %T", cloneSection)
}
expected := orig.Clone()
if !reflect.DeepEqual(clone, expected) {
t.Errorf("CloneSection() = %+v; want %+v", clone, expected)
}
}
func TestDiffIPsJsonCfg(t *testing.T) {
v1 := &IPsCfg{
Enabled: true,
IndexedSelects: false,
StoreInterval: 24 * time.Hour,
StringIndexedFields: &[]string{"field1", "field2"},
PrefixIndexedFields: &[]string{"prefix1"},
SuffixIndexedFields: &[]string{"suffix1"},
ExistsIndexedFields: &[]string{"exists1"},
NotExistsIndexedFields: &[]string{"notexists1"},
NestedFields: true,
Opts: &IPsOpts{},
}
v2 := &IPsCfg{
Enabled: false,
IndexedSelects: true,
StoreInterval: 48 * time.Hour,
StringIndexedFields: &[]string{"field2", "field3"},
PrefixIndexedFields: &[]string{"prefix2"},
SuffixIndexedFields: &[]string{"suffix2"},
ExistsIndexedFields: &[]string{"exists2"},
NotExistsIndexedFields: &[]string{"notexists2"},
NestedFields: false,
Opts: &IPsOpts{},
}
diff := diffIPsJsonCfg(nil, v1, v2)
if diff.Enabled == nil || *diff.Enabled != v2.Enabled {
t.Errorf("Expected Enabled = %v, got %+v", v2.Enabled, diff.Enabled)
}
if diff.IndexedSelects == nil || *diff.IndexedSelects != v2.IndexedSelects {
t.Errorf("Expected IndexedSelects = %v, got %+v", v2.IndexedSelects, diff.IndexedSelects)
}
if diff.NestedFields == nil || *diff.NestedFields != v2.NestedFields {
t.Errorf("Expected NestedFields = %v, got %+v", v2.NestedFields, diff.NestedFields)
}
expectedInterval := v2.StoreInterval.String()
if diff.StoreInterval == nil || *diff.StoreInterval != expectedInterval {
t.Errorf("Expected StoreInterval = %v, got %+v", expectedInterval, diff.StoreInterval)
}
if diff.StringIndexedFields == nil || !reflect.DeepEqual(*diff.StringIndexedFields, *v2.StringIndexedFields) {
t.Errorf("Expected StringIndexedFields = %+v, got %+v", *v2.StringIndexedFields, diff.StringIndexedFields)
}
if diff.PrefixIndexedFields == nil || !reflect.DeepEqual(*diff.PrefixIndexedFields, *v2.PrefixIndexedFields) {
t.Errorf("Expected PrefixIndexedFields = %+v, got %+v", *v2.PrefixIndexedFields, diff.PrefixIndexedFields)
}
if diff.SuffixIndexedFields == nil || !reflect.DeepEqual(*diff.SuffixIndexedFields, *v2.SuffixIndexedFields) {
t.Errorf("Expected SuffixIndexedFields = %+v, got %+v", *v2.SuffixIndexedFields, diff.SuffixIndexedFields)
}
if diff.ExistsIndexedFields == nil || !reflect.DeepEqual(*diff.ExistsIndexedFields, *v2.ExistsIndexedFields) {
t.Errorf("Expected ExistsIndexedFields = %+v, got %+v", *v2.ExistsIndexedFields, diff.ExistsIndexedFields)
}
if diff.NotExistsIndexedFields == nil || !reflect.DeepEqual(*diff.NotExistsIndexedFields, *v2.NotExistsIndexedFields) {
t.Errorf("Expected NotExistsIndexedFields = %+v, got %+v", *v2.NotExistsIndexedFields, diff.NotExistsIndexedFields)
}
if diff.Opts != nil && (diff.Opts.AllocationID != nil || diff.Opts.TTL != nil) {
t.Errorf("Expected Opts to be nil or empty, got: %+v", diff.Opts)
}
}
func TestDiffIPsOptsJsonCfg(t *testing.T) {
ttl1 := []*DynamicDurationOpt{{value: 72 * time.Hour}}
ttl2 := []*DynamicDurationOpt{{value: 24 * time.Hour}}
allocID1 := []*DynamicStringOpt{{value: "id1"}}
allocID2 := []*DynamicStringOpt{{value: "id2"}}
tests := []struct {
name string
v1, v2 *IPsOpts
expectDiff bool
expectedTTL []*DynamicInterfaceOpt
expectedID []*DynamicInterfaceOpt
}{
{
name: "No difference",
v1: &IPsOpts{AllocationID: allocID1, TTL: ttl1},
v2: &IPsOpts{AllocationID: allocID1, TTL: ttl1},
expectDiff: false,
},
{
name: "TTL differs",
v1: &IPsOpts{AllocationID: allocID1, TTL: ttl1},
v2: &IPsOpts{AllocationID: allocID1, TTL: ttl2},
expectDiff: true,
expectedTTL: DurationToIfaceDynamicOpts(ttl2),
},
{
name: "AllocationID differs",
v1: &IPsOpts{AllocationID: allocID1, TTL: ttl1},
v2: &IPsOpts{AllocationID: allocID2, TTL: ttl1},
expectDiff: true,
expectedID: DynamicStringToInterfaceOpts(allocID2),
},
{
name: "Both differ",
v1: &IPsOpts{AllocationID: allocID1, TTL: ttl1},
v2: &IPsOpts{AllocationID: allocID2, TTL: ttl2},
expectDiff: true,
expectedID: DynamicStringToInterfaceOpts(allocID2),
expectedTTL: DurationToIfaceDynamicOpts(ttl2),
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
diff := diffIPsOptsJsonCfg(nil, tt.v1, tt.v2)
if tt.expectDiff {
if tt.expectedID != nil && !reflect.DeepEqual(diff.AllocationID, tt.expectedID) {
t.Errorf("Expected AllocationID diff: %+v, got: %+v", tt.expectedID, diff.AllocationID)
}
if tt.expectedTTL != nil && !reflect.DeepEqual(diff.TTL, tt.expectedTTL) {
t.Errorf("Expected TTL diff: %+v, got: %+v", tt.expectedTTL, diff.TTL)
}
} else {
if diff.AllocationID != nil || diff.TTL != nil {
t.Errorf("Expected no differences, but got AllocationID: %+v, TTL: %+v", diff.AllocationID, diff.TTL)
}
}
})
}
}
func TestIPsOptsloadFromJSONCfg(t *testing.T) {
opts := &IPsOpts{}
if err := opts.loadFromJSONCfg(nil); err != nil {
t.Errorf("Expected nil error on nil input, got: %v", err)
}
jCfg := &IPsOptsJson{
AllocationID: []*DynamicInterfaceOpt{
{Value: "alloc1"},
{Value: "alloc2"},
},
TTL: []*DynamicInterfaceOpt{
{Value: "48h"},
{Value: "72h"},
},
}
expected := &IPsOpts{
AllocationID: []*DynamicStringOpt{
{value: "alloc1"},
{value: "alloc2"},
},
TTL: []*DynamicDurationOpt{
{value: 48 * time.Hour},
{value: 72 * time.Hour},
},
}
opts = &IPsOpts{}
if err := opts.loadFromJSONCfg(jCfg); err != nil {
t.Errorf("Unexpected error: %v", err)
}
if !reflect.DeepEqual(opts, expected) {
t.Errorf("Expected %+v, got %+v", utils.ToJSON(expected), utils.ToJSON(opts))
}
}