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

297 lines
8.9 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 rates
import (
"fmt"
"sort"
"time"
"github.com/ericlagergren/decimal"
"github.com/cgrates/cgrates/utils"
)
type rpWithWeight struct {
*utils.RateProfile
weight float64
}
func newRatesWithWinner(rIt *rateWithTimes) *ratesWithWinner {
return &ratesWithWinner{
rts: map[string]*rateWithTimes{
rIt.id(): rIt,
},
wnr: rIt,
}
}
func initRatesWithWinner() *ratesWithWinner {
return &ratesWithWinner{
rts: make(map[string]*rateWithTimes),
}
}
// ratesWithWinner computes always the winner based on highest Weight
type ratesWithWinner struct {
rts map[string]*rateWithTimes
wnr *rateWithTimes
}
// add will add the rate to the rates
func (rs *ratesWithWinner) add(rWt *rateWithTimes) {
rs.rts[rWt.id()] = rWt
if rs.wnr == nil || rs.wnr.weight < rWt.weight {
rs.wnr = rWt
}
}
// winner returns the rate with the highest Weight
func (rs *ratesWithWinner) winner() *rateWithTimes {
return rs.wnr
}
// has tests if the rateID is present in rates
func (rs *ratesWithWinner) has(rtID string) (has bool) {
_, has = rs.rts[rtID]
return
}
// rateWithTimes activates a rate on an interval
type rateWithTimes struct {
uId string
rt *utils.Rate
aTime,
iTime time.Time
weight float64
}
// id is used to provide an unique identifier for a rateWithTimes
func (rWt *rateWithTimes) id() string {
if rWt.uId == "" {
rWt.uId = fmt.Sprintf("%s_%d", rWt.rt.ID, rWt.aTime.Unix())
}
return rWt.uId
}
type orderedRate struct {
*decimal.Big
*utils.Rate
}
// orderRatesOnIntervals will order the rates based on ActivationInterval and intervalStart of each Rate
// there can be only one winning Rate for each interval, prioritized by the Weight
func orderRatesOnIntervals(aRts []*utils.Rate, wghts []float64, sTime time.Time, usage *decimal.Big,
isDuration bool, verbosity int) (ordRts []*orderedRate, err error) {
usageInt, ok := usage.Int64()
if !ok {
return nil, fmt.Errorf("<%s> cannot convert <%+v> increment to Int64", utils.RateS, usage)
}
endTime := sTime.Add(time.Duration(usageInt))
// index the received rates based on unique times they run
rtIdx := make(map[time.Time]*ratesWithWinner) // map[ActivationTimes]*ratesWithWinner
allRates := make(map[string]*rateWithTimes)
for i, rt := range aRts {
var rTimes [][]time.Time
if rTimes, err = rt.RunTimes(sTime, endTime, verbosity); err != nil {
return
}
for _, rTimeSet := range rTimes {
rIt := &rateWithTimes{
rt: rt,
aTime: rTimeSet[0],
iTime: rTimeSet[1],
weight: wghts[i], // weights are in order of aRts
}
allRates[rIt.id()] = rIt
if _, hasKey := rtIdx[rTimeSet[0]]; !hasKey {
rtIdx[rTimeSet[0]] = initRatesWithWinner()
}
rtIdx[rTimeSet[0]].add(rIt)
if rTimeSet[1].IsZero() { // the rate will always be active
continue
}
if _, hasKey := rtIdx[rTimeSet[1]]; !hasKey {
rtIdx[rTimeSet[1]] = initRatesWithWinner()
}
}
}
// add the active rates to all time samples
for tm, rWw := range rtIdx {
for _, rIt := range allRates {
if rWw.has(rIt.id()) ||
rIt.aTime.After(tm) ||
!rIt.iTime.IsZero() && !tm.Before(rIt.iTime) {
continue
}
rWw.add(rIt)
}
}
// sort the activation times
sortedATimes := make([]time.Time, len(rtIdx))
idxATimes := 0
for aTime := range rtIdx {
sortedATimes[idxATimes] = aTime
idxATimes++
}
sort.Slice(sortedATimes, func(i, j int) bool {
return sortedATimes[i].Before(sortedATimes[j])
})
// start with most recent activationTime lower or equal to sTime
for i, aT := range sortedATimes {
if !aT.After(sTime) || i == 0 {
continue
}
sortedATimes = sortedATimes[i-1:]
break
}
// compute the list of returned rates together with their index interval
if isDuration {
// add all the possible ActivationTimes from cron expressions
usageIndx := decimal.New(0, 0) // the difference between setup and activation time of the rate
for _, aTime := range sortedATimes {
if !endTime.After(aTime) {
break // we are not interested about further rates
}
wnr := rtIdx[aTime].winner()
if wnr == nil {
continue
}
if sTime.Before(aTime) {
usageIndx = decimal.New(int64(aTime.Sub(sTime)), 0)
}
if len(ordRts) == 0 || wnr.rt.ID != ordRts[len(ordRts)-1].Rate.ID { // only add the winner if not already active
ordRts = append(ordRts, &orderedRate{usageIndx, rtIdx[aTime].winner().rt})
}
}
} else { // only first rate is considered for units
ordRts = []*orderedRate{{decimal.New(0, 0), rtIdx[sortedATimes[0]].winner().rt}}
}
return
}
func getRateIntervalIDFromIncrement(cstRts map[string]*utils.IntervalRate, intRt *utils.IntervalRate) string {
for key, val := range cstRts {
if val.Equals(intRt) {
return key
}
}
key := utils.UUIDSha1Prefix()
cstRts[key] = intRt
return key
}
// computeRateSIntervals will give out the cost projection for the given orderedRates and usage
func computeRateSIntervals(rts []*orderedRate, intervalStart, usage *decimal.Big,
cstRts map[string]*utils.IntervalRate) (rtIvls []*utils.RateSInterval, err error) {
totalUsage := usage
if intervalStart.Cmp(decimal.New(0, 0)) != 0 {
totalUsage = utils.SumBig(usage, intervalStart)
}
for i, rt := range rts {
isLastRt := i == len(rts)-1
var rtUsageEIdx *decimal.Big
if !isLastRt {
rtUsageEIdx = rts[i+1].Big
} else {
rtUsageEIdx = totalUsage
}
var rIcmts []*utils.RateSIncrement
iRtUsageSIdx := intervalStart
iRtUsageEIdx := rtUsageEIdx
for j, iRt := range rt.IntervalRates {
//fmt.Printf("ivalStart: %v, ivalEnd: %v, rtID: %s, increment idx: %d, iRtUsageSIdx: %+v, iRtUsageEIdx: %+v, iRt: %s\n",
// intervalStart, rtUsageEIdx, rt.ID, j, iRtUsageSIdx, iRtUsageEIdx, utils.ToIJSON(iRt))
if iRtUsageSIdx.Cmp(rtUsageEIdx) >= 0 { // charged enough for interval
break
}
// make sure we bill from start
if iRt.IntervalStart.Cmp(iRtUsageSIdx) > 0 && j == 0 {
return nil, fmt.Errorf("intervalStart for rate: <%s> higher than usage: %v",
rt.UID(), iRtUsageSIdx)
}
isLastIRt := j == len(rt.IntervalRates)-1
if !isLastIRt && rt.IntervalRates[j+1].IntervalStart.Cmp(iRtUsageSIdx) <= 0 {
continue // the next interval changes the rating
}
if iRt.Increment.Cmp(decimal.New(0, 0)) == 0 {
return nil, fmt.Errorf("zero increment to be charged within rate: <%s>", rt.UID())
}
if rt.IntervalRates[j].FixedFee != nil && rt.IntervalRates[j].FixedFee.Cmp(decimal.New(0, 0)) != 0 { // Add FixedFee
rIcmts = append(rIcmts, &utils.RateSIncrement{
IncrementStart: &utils.Decimal{Big: iRtUsageSIdx},
RateIntervalIndex: j,
RateID: getRateIntervalIDFromIncrement(cstRts, rt.IntervalRates[j]),
CompressFactor: 1,
Usage: utils.NewDecimal(utils.InvalidUsage, 0),
})
}
if rt.IntervalRates[j].RecurrentFee == nil { // RecurrentFee not defined, go to the next increment
continue
}
if isLastIRt {
iRtUsageEIdx = rtUsageEIdx
} else if rt.IntervalRates[j+1].IntervalStart.Cmp(rtUsageEIdx) > 0 {
iRtUsageEIdx = rtUsageEIdx
} else {
iRtUsageEIdx = rt.IntervalRates[j+1].IntervalStart.Big
}
iRtUsage := utils.SubstractBig(iRtUsageEIdx, iRtUsageSIdx)
intUsage := iRtUsage
intIncrm := iRt.Increment.Big
cmpFactor, rem := utils.DivideBigWithReminder(intUsage, intIncrm)
if rem.Cmp(decimal.New(0, 0)) != 0 {
cmpFactor = utils.SumBig(cmpFactor, decimal.New(1, 0)) // int division has used math.Floor, need Ceil
}
cmpFactorInt, ok := cmpFactor.Int64()
if !ok {
return nil, fmt.Errorf("<%s> cannot convert <%+v> increment to Int64", utils.RateS, cmpFactor)
}
rIcmts = append(rIcmts, &utils.RateSIncrement{
IncrementStart: &utils.Decimal{Big: iRtUsageSIdx},
RateID: getRateIntervalIDFromIncrement(cstRts, rt.IntervalRates[j]),
CompressFactor: cmpFactorInt,
RateIntervalIndex: j,
Usage: &utils.Decimal{Big: iRtUsage},
})
iRtUsageSIdx = utils.SumBig(iRtUsageSIdx, iRtUsage)
}
usageStart := intervalStart
intervalStart = rtUsageEIdx // continue for the next interval
if len(rIcmts) == 0 { // no match found
continue
}
rtIvls = append(rtIvls, &utils.RateSInterval{
IntervalStart: &utils.Decimal{Big: usageStart},
Increments: rIcmts,
CompressFactor: 1,
})
if iRtUsageSIdx.Cmp(totalUsage) >= 0 { // charged enough for the usage
break
}
}
return
}