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

258 lines
7.8 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 (
"slices"
"time"
"github.com/cgrates/birpc/context"
"github.com/cgrates/cgrates/utils"
)
const (
IPsAllocationIDDftOpt = utils.EmptyString
IPsTTLDftOpt = 72 * time.Hour
IPsUnitsDftOpt = 1
)
// IPsJsonCfg holds the unparsed ips section configuration as found in the
// config file.
type IPsJsonCfg struct {
Enabled *bool `json:"enabled"`
IndexedSelects *bool `json:"indexed_selects"`
StoreInterval *string `json:"store_interval"`
StringIndexedFields *[]string `json:"string_indexed_fields"`
PrefixIndexedFields *[]string `json:"prefix_indexed_fields"`
SuffixIndexedFields *[]string `json:"suffix_indexed_fields"`
ExistsIndexedFields *[]string `json:"exists_indexed_fields"`
NotExistsIndexedFields *[]string `json:"not_exists_indexed_fields"`
NestedFields *bool `json:"nested_fields"`
Opts *IPsOptsJson `json:"opts"`
}
// IPsCfg represents the configuration of the IPs module.
type IPsCfg struct {
Enabled bool
IndexedSelects bool
StoreInterval time.Duration
StringIndexedFields *[]string
PrefixIndexedFields *[]string
SuffixIndexedFields *[]string
ExistsIndexedFields *[]string
NotExistsIndexedFields *[]string
NestedFields bool
Opts *IPsOpts
}
// Load loads the IPs section of the configuration.
func (rlcfg *IPsCfg) Load(ctx *context.Context, jsnCfg ConfigDB, _ *CGRConfig) (err error) {
jsnRLSCfg := new(IPsJsonCfg)
if err = jsnCfg.GetSection(ctx, IPsJSON, jsnRLSCfg); err != nil {
return
}
return rlcfg.loadFromJSONCfg(jsnRLSCfg)
}
func (IPsCfg) SName() string { return IPsJSON }
func (c IPsCfg) CloneSection() Section { return c.Clone() }
func (c *IPsCfg) loadFromJSONCfg(jc *IPsJsonCfg) error {
if jc == nil {
return nil
}
if jc.Enabled != nil {
c.Enabled = *jc.Enabled
}
if jc.IndexedSelects != nil {
c.IndexedSelects = *jc.IndexedSelects
}
if jc.StoreInterval != nil {
v, err := utils.ParseDurationWithNanosecs(*jc.StoreInterval)
if err != nil {
return err
}
c.StoreInterval = v
}
if jc.StringIndexedFields != nil {
sif := slices.Clone(*jc.StringIndexedFields)
c.StringIndexedFields = &sif
}
if jc.PrefixIndexedFields != nil {
pif := slices.Clone(*jc.PrefixIndexedFields)
c.PrefixIndexedFields = &pif
}
if jc.SuffixIndexedFields != nil {
sif := slices.Clone(*jc.SuffixIndexedFields)
c.SuffixIndexedFields = &sif
}
if jc.ExistsIndexedFields != nil {
eif := slices.Clone(*jc.ExistsIndexedFields)
c.ExistsIndexedFields = &eif
}
if jc.NotExistsIndexedFields != nil {
c.NotExistsIndexedFields = utils.SliceStringPointer(slices.Clone(*jc.NotExistsIndexedFields))
}
if jc.NestedFields != nil {
c.NestedFields = *jc.NestedFields
}
if jc.Opts != nil {
if err := c.Opts.loadFromJSONCfg(jc.Opts); err != nil {
return err
}
}
return nil
}
// Clone returns a deep copy of IPsCfg.
func (c IPsCfg) Clone() *IPsCfg {
clone := &IPsCfg{
Enabled: c.Enabled,
IndexedSelects: c.IndexedSelects,
StoreInterval: c.StoreInterval,
NestedFields: c.NestedFields,
Opts: c.Opts.Clone(),
}
if c.StringIndexedFields != nil {
idx := slices.Clone(*c.StringIndexedFields)
clone.StringIndexedFields = &idx
}
if c.PrefixIndexedFields != nil {
idx := slices.Clone(*c.PrefixIndexedFields)
clone.PrefixIndexedFields = &idx
}
if c.SuffixIndexedFields != nil {
idx := slices.Clone(*c.SuffixIndexedFields)
clone.SuffixIndexedFields = &idx
}
if c.ExistsIndexedFields != nil {
idx := slices.Clone(*c.ExistsIndexedFields)
clone.ExistsIndexedFields = &idx
}
if c.NotExistsIndexedFields != nil {
idx := slices.Clone(*c.NotExistsIndexedFields)
clone.NotExistsIndexedFields = &idx
}
return clone
}
// AsMapInterface returns the ips config as a map[string]any.
func (c IPsCfg) AsMapInterface() any {
return map[string]any{
utils.EnabledCfg: c.Enabled,
utils.IndexedSelectsCfg: c.IndexedSelects,
utils.NestedFieldsCfg: c.NestedFields,
utils.StoreIntervalCfg: c.StoreInterval.String(),
utils.StringIndexedFieldsCfg: c.StringIndexedFields,
utils.PrefixIndexedFieldsCfg: c.PrefixIndexedFields,
utils.SuffixIndexedFieldsCfg: c.SuffixIndexedFields,
utils.ExistsIndexedFieldsCfg: c.ExistsIndexedFields,
utils.NotExistsIndexedFieldsCfg: c.NotExistsIndexedFields,
utils.OptsCfg: c.Opts.AsMapInterface(),
}
}
type IPsOptsJson struct {
AllocationID []*DynamicInterfaceOpt `json:"*allocationID"`
TTL []*DynamicInterfaceOpt `json:"*ttl"`
}
type IPsOpts struct {
AllocationID []*DynamicStringOpt
TTL []*DynamicDurationOpt
}
func (o *IPsOpts) loadFromJSONCfg(jc *IPsOptsJson) error {
if jc == nil {
return nil
}
// NOTE: prepend to the existing slice to ensure that the default opts that
// always match are at the end.
if jc.AllocationID != nil {
allocID, err := InterfaceToDynamicStringOpts(jc.AllocationID)
if err != nil {
return err
}
o.AllocationID = append(allocID, o.AllocationID...)
}
if jc.TTL != nil {
ttl, err := IfaceToDurationDynamicOpts(jc.TTL)
if err != nil {
return err
}
o.TTL = append(ttl, o.TTL...)
}
return nil
}
// Clone returns a deep copy of IPsOpts.
func (o *IPsOpts) Clone() *IPsOpts {
return &IPsOpts{
AllocationID: CloneDynamicStringOpt(o.AllocationID),
TTL: CloneDynamicDurationOpt(o.TTL),
}
}
// AsMapInterface returns the config as a map[string]any.
func (o *IPsOpts) AsMapInterface() map[string]any {
return map[string]any{
utils.MetaAllocationID: o.AllocationID,
utils.MetaTTLCfg: o.TTL,
}
}
func diffIPsOptsJsonCfg(d *IPsOptsJson, v1, v2 *IPsOpts) *IPsOptsJson {
if d == nil {
d = new(IPsOptsJson)
}
if !DynamicStringOptEqual(v1.AllocationID, v2.AllocationID) {
d.AllocationID = DynamicStringToInterfaceOpts(v2.AllocationID)
}
if !DynamicDurationOptEqual(v1.TTL, v2.TTL) {
d.TTL = DurationToIfaceDynamicOpts(v2.TTL)
}
return d
}
func diffIPsJsonCfg(d *IPsJsonCfg, v1, v2 *IPsCfg) *IPsJsonCfg {
if d == nil {
d = new(IPsJsonCfg)
}
if v1.Enabled != v2.Enabled {
d.Enabled = utils.BoolPointer(v2.Enabled)
}
if v1.IndexedSelects != v2.IndexedSelects {
d.IndexedSelects = utils.BoolPointer(v2.IndexedSelects)
}
if v1.StoreInterval != v2.StoreInterval {
d.StoreInterval = utils.StringPointer(v2.StoreInterval.String())
}
d.StringIndexedFields = diffIndexSlice(d.StringIndexedFields, v1.StringIndexedFields, v2.StringIndexedFields)
d.PrefixIndexedFields = diffIndexSlice(d.PrefixIndexedFields, v1.PrefixIndexedFields, v2.PrefixIndexedFields)
d.SuffixIndexedFields = diffIndexSlice(d.SuffixIndexedFields, v1.SuffixIndexedFields, v2.SuffixIndexedFields)
d.ExistsIndexedFields = diffIndexSlice(d.ExistsIndexedFields, v1.ExistsIndexedFields, v2.ExistsIndexedFields)
d.NotExistsIndexedFields = diffIndexSlice(d.NotExistsIndexedFields, v1.NotExistsIndexedFields, v2.NotExistsIndexedFields)
if v1.NestedFields != v2.NestedFields {
d.NestedFields = utils.BoolPointer(v2.NestedFields)
}
d.Opts = diffIPsOptsJsonCfg(d.Opts, v1.Opts, v2.Opts)
return d
}