Files
cgrates/utils/dynamicweight.go
2021-12-21 15:05:44 +01:00

115 lines
3.1 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 utils
import (
"fmt"
"strconv"
"strings"
)
// NewDynamicWeightsFromString creates a DynamicWeight list based on the string received from .csv/StorDB
func NewDynamicWeightsFromString(s, dWSep, fltrSep string) (dWs DynamicWeights, err error) {
if len(s) == 0 {
return DynamicWeights{{}}, nil
}
dwStrs := strings.Split(s, dWSep)
lnDwStrs := len(dwStrs)
if lnDwStrs%2 != 0 { // need to have multiples of number of fields in one DynamicWeight
return nil, fmt.Errorf("invalid DynamicWeight format for string <%s>", s)
}
dWs = make([]*DynamicWeight, 0, lnDwStrs/2)
for i := 0; i < lnDwStrs; i += 2 {
var fltrIDs []string
if len(dwStrs[i]) != 0 {
fltrIDs = strings.Split(dwStrs[i], fltrSep)
}
var weight float64
if len(dwStrs[i+1]) != 0 {
if weight, err = strconv.ParseFloat(dwStrs[i+1], 64); err != nil {
return nil, fmt.Errorf("invalid Weight <%s> in string: <%s>", dwStrs[i+1], s)
}
}
dWs = append(dWs, &DynamicWeight{FilterIDs: fltrIDs, Weight: weight})
}
return
}
type DynamicWeights []*DynamicWeight
func (dWS DynamicWeights) String(dWSep, fltrsep string) (out string) {
if len(dWS) == 0 {
return
}
dwToString := make([]string, len(dWS))
for i, value := range dWS {
dwToString[i] = value.String(dWSep, fltrsep)
}
return strings.Join(dwToString, dWSep)
}
func (dW DynamicWeight) String(dWSep, fltrsep string) (out string) {
return strings.Join(dW.FilterIDs, fltrsep) + dWSep + strconv.FormatFloat(dW.Weight, 'f', -1, 64)
}
func (dW *DynamicWeight) Equals(dnWg *DynamicWeight) (eq bool) {
if dW.FilterIDs == nil && dnWg.FilterIDs != nil ||
dW.FilterIDs != nil && dnWg.FilterIDs == nil ||
len(dW.FilterIDs) != len(dnWg.FilterIDs) ||
dW.Weight != dnWg.Weight {
return
}
for i := range dW.FilterIDs {
if dW.FilterIDs[i] != dnWg.FilterIDs[i] {
return
}
}
return true
}
// DynamicWeight returns Weight based on Filters
type DynamicWeight struct {
FilterIDs []string
Weight float64
}
func (dW *DynamicWeight) Clone() (dinWeight *DynamicWeight) {
dinWeight = &DynamicWeight{
Weight: dW.Weight,
}
if dW.FilterIDs != nil {
dinWeight.FilterIDs = make([]string, len(dW.FilterIDs))
for i, value := range dW.FilterIDs {
dinWeight.FilterIDs[i] = value
}
}
return dinWeight
}
func (dW DynamicWeights) Clone() (dinWeight DynamicWeights) {
if dW == nil {
return
}
dinWeight = make(DynamicWeights, len(dW))
for i, value := range dW {
dinWeight[i] = value.Clone()
}
return
}