mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-11 18:16:24 +05:00
654 lines
16 KiB
Go
654 lines
16 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 engine
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/cgrates/cgrates/utils"
|
|
)
|
|
|
|
/*
|
|
Defines a time interval for which a certain set of prices will apply
|
|
*/
|
|
type RateInterval struct {
|
|
Timing *RITiming
|
|
Rating *RIRate
|
|
Weight float64
|
|
}
|
|
|
|
// Separate structure used for rating plan size optimization
|
|
type RITiming struct {
|
|
ID string
|
|
Years utils.Years
|
|
Months utils.Months
|
|
MonthDays utils.MonthDays
|
|
WeekDays utils.WeekDays
|
|
StartTime string // ##:##:## format
|
|
EndTime string // ##:##:## format
|
|
cronString string
|
|
tag string // loading validation only
|
|
}
|
|
|
|
func (rit *RITiming) CronString() string {
|
|
if rit.cronString != "" && rit.ID != utils.MetaMonthlyEstimated {
|
|
return rit.cronString
|
|
}
|
|
var sec, min, hour, monthday, month, weekday, year string
|
|
switch rit.StartTime {
|
|
case "":
|
|
hour, min, sec = "*", "*", "*"
|
|
case utils.MetaEveryMinute:
|
|
hour, min, sec = "*", "*", "0"
|
|
case utils.MetaHourly:
|
|
hour, min, sec = "*", "0", "0"
|
|
default:
|
|
hms := strings.Split(rit.StartTime, ":")
|
|
if len(hms) == 3 {
|
|
hour, min, sec = hms[0], hms[1], hms[2]
|
|
} else {
|
|
hour, min, sec = "*", "*", "*"
|
|
}
|
|
hour = strings.TrimPrefix(hour, "0")
|
|
min = strings.TrimPrefix(min, "0")
|
|
sec = strings.TrimPrefix(sec, "0")
|
|
}
|
|
if len(rit.MonthDays) == 0 {
|
|
monthday = "*"
|
|
} else {
|
|
for i, md := range rit.MonthDays {
|
|
if i > 0 {
|
|
monthday += ","
|
|
}
|
|
monthday += strconv.Itoa(md)
|
|
}
|
|
}
|
|
if len(rit.Months) == 0 {
|
|
month = "*"
|
|
} else {
|
|
for i, md := range rit.Months {
|
|
if i > 0 {
|
|
month += ","
|
|
}
|
|
month += strconv.Itoa(int(md))
|
|
}
|
|
}
|
|
if len(rit.WeekDays) == 0 {
|
|
weekday = "*"
|
|
} else {
|
|
for i, md := range rit.WeekDays {
|
|
if i > 0 {
|
|
weekday += ","
|
|
}
|
|
weekday += strconv.Itoa(int(md))
|
|
}
|
|
}
|
|
if len(rit.Years) == 0 {
|
|
year = "*"
|
|
} else {
|
|
for i, md := range rit.Years {
|
|
if i > 0 {
|
|
year += ","
|
|
}
|
|
year += strconv.Itoa(int(md))
|
|
}
|
|
}
|
|
if monthday == "-1" { // in case we receive -1 we send to cron special character L ( Last )
|
|
monthday = "L"
|
|
}
|
|
rit.cronString = fmt.Sprintf("%s %s %s %s %s %s %s", sec, min, hour, monthday, month, weekday, year)
|
|
return rit.cronString
|
|
}
|
|
|
|
/*
|
|
Returns a time object that represents the end of the interval realtive to the received time
|
|
*/
|
|
func (rit *RITiming) getRightMargin(t time.Time) (rigthtTime time.Time) {
|
|
year, month, day := t.Year(), t.Month(), t.Day()
|
|
hour, min, sec, nsec := 23, 59, 59, 0
|
|
loc := t.Location()
|
|
if rit.EndTime != "" {
|
|
split := strings.Split(rit.EndTime, ":")
|
|
hour, _ = strconv.Atoi(split[0])
|
|
min, _ = strconv.Atoi(split[1])
|
|
sec, _ = strconv.Atoi(split[2])
|
|
//log.Print("RIGHT1: ", time.Date(year, month, day, hour, min, sec, nsec, loc))
|
|
return time.Date(year, month, day, hour, min, sec, nsec, loc)
|
|
}
|
|
//log.Print("RIGHT2: ", time.Date(year, month, day, hour, min, sec, nsec, loc).Add(time.Second))
|
|
return time.Date(year, month, day, hour, min, sec, nsec, loc).Add(time.Second)
|
|
}
|
|
|
|
//Returns a time object that represents the start of the interval realtive to the received time
|
|
func (rit *RITiming) getLeftMargin(t time.Time) (rigthtTime time.Time) {
|
|
year, month, day := t.Year(), t.Month(), t.Day()
|
|
hour, min, sec, nsec := 0, 0, 0, 0
|
|
loc := t.Location()
|
|
if rit.StartTime != "" {
|
|
split := strings.Split(rit.StartTime, ":")
|
|
hour, _ = strconv.Atoi(split[0])
|
|
min, _ = strconv.Atoi(split[1])
|
|
sec, _ = strconv.Atoi(split[2])
|
|
}
|
|
//log.Print("LEFT: ", time.Date(year, month, day, hour, min, sec, nsec, loc))
|
|
return time.Date(year, month, day, hour, min, sec, nsec, loc)
|
|
}
|
|
|
|
// Returns wheter the Timing is active at the specified time
|
|
func (rit *RITiming) IsActiveAt(t time.Time) bool {
|
|
// check for years
|
|
if len(rit.Years) > 0 && !rit.Years.Contains(t.Year()) {
|
|
return false
|
|
}
|
|
// check for months
|
|
if len(rit.Months) > 0 && !rit.Months.Contains(t.Month()) {
|
|
return false
|
|
}
|
|
// check for month days
|
|
if len(rit.MonthDays) > 0 && !rit.MonthDays.Contains(t.Day()) {
|
|
return false
|
|
}
|
|
// check for weekdays
|
|
if len(rit.WeekDays) > 0 && !rit.WeekDays.Contains(t.Weekday()) {
|
|
return false
|
|
}
|
|
//log.Print("Time: ", t)
|
|
|
|
//log.Print("Left Margin: ", rit.getLeftMargin(t))
|
|
// check for start hour
|
|
if t.Before(rit.getLeftMargin(t)) {
|
|
return false
|
|
}
|
|
|
|
//log.Print("Right Margin: ", rit.getRightMargin(t))
|
|
// check for end hour
|
|
if t.After(rit.getRightMargin(t)) {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
// IsActive returns wheter the Timing is active now
|
|
func (rit *RITiming) IsActive() bool {
|
|
return rit.IsActiveAt(time.Now())
|
|
}
|
|
|
|
func (rit *RITiming) IsBlank() bool {
|
|
return len(rit.Years) == 0 &&
|
|
len(rit.Months) == 0 &&
|
|
len(rit.MonthDays) == 0 &&
|
|
len(rit.WeekDays) == 0 &&
|
|
rit.StartTime == "00:00:00"
|
|
}
|
|
|
|
func (rit *RITiming) Stringify() string {
|
|
return utils.Sha1(fmt.Sprintf("%v", rit))[:8]
|
|
}
|
|
|
|
// Separate structure used for rating plan size optimization
|
|
type RIRate struct {
|
|
ConnectFee float64
|
|
RoundingMethod string
|
|
RoundingDecimals int
|
|
MaxCost float64
|
|
MaxCostStrategy string
|
|
Rates RateGroups // GroupRateInterval (start time): RGRate
|
|
tag string // loading validation only
|
|
}
|
|
|
|
func (rir *RIRate) Stringify() string {
|
|
str := fmt.Sprintf("%v %v %v %v %v", rir.ConnectFee, rir.RoundingMethod, rir.RoundingDecimals, rir.MaxCost, rir.MaxCostStrategy)
|
|
for _, r := range rir.Rates {
|
|
str += r.Stringify()
|
|
}
|
|
return utils.Sha1(str)[:8]
|
|
}
|
|
|
|
type RGRate struct {
|
|
GroupIntervalStart time.Duration
|
|
Value float64
|
|
RateIncrement time.Duration
|
|
RateUnit time.Duration
|
|
}
|
|
|
|
// FieldAsInterface func to help EventCost FieldAsInterface
|
|
func (r *RGRate) FieldAsInterface(fldPath []string) (val interface{}, err error) {
|
|
if r == nil || len(fldPath) != 1 {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
switch fldPath[0] {
|
|
default:
|
|
return nil, fmt.Errorf("unsupported field prefix: <%s>", fldPath[0])
|
|
case utils.GroupIntervalStart:
|
|
return r.GroupIntervalStart, nil
|
|
case utils.Value:
|
|
return r.Value, nil
|
|
case utils.RateIncrement:
|
|
return r.RateIncrement, nil
|
|
case utils.RateUnit:
|
|
return r.RateUnit, nil
|
|
}
|
|
}
|
|
|
|
func (r *RGRate) Stringify() string {
|
|
return utils.Sha1(fmt.Sprintf("%v", r))[:8]
|
|
}
|
|
|
|
func (p *RGRate) Equal(o *RGRate) bool {
|
|
return p.GroupIntervalStart == o.GroupIntervalStart &&
|
|
p.Value == o.Value &&
|
|
p.RateIncrement == o.RateIncrement &&
|
|
p.RateUnit == o.RateUnit
|
|
}
|
|
|
|
type RateGroups []*RGRate
|
|
|
|
func (pg RateGroups) Len() int {
|
|
return len(pg)
|
|
}
|
|
|
|
func (pg RateGroups) Swap(i, j int) {
|
|
pg[i], pg[j] = pg[j], pg[i]
|
|
}
|
|
|
|
func (pg RateGroups) Less(i, j int) bool {
|
|
return pg[i].GroupIntervalStart < pg[j].GroupIntervalStart
|
|
}
|
|
|
|
func (pg RateGroups) Sort() {
|
|
sort.Sort(pg)
|
|
}
|
|
|
|
func (pg RateGroups) Equal(og RateGroups) bool {
|
|
if len(pg) != len(og) {
|
|
return false
|
|
}
|
|
for i := 0; i < len(pg); i++ {
|
|
if !pg[i].Equal(og[i]) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (pg *RateGroups) AddRate(ps ...*RGRate) {
|
|
for _, p := range ps {
|
|
found := false
|
|
for _, op := range *pg {
|
|
if op.Equal(p) {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
*pg = append(*pg, p)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (pg RateGroups) Equals(oRG RateGroups) bool {
|
|
if len(pg) != len(oRG) {
|
|
return false
|
|
}
|
|
for i := range pg {
|
|
if !pg[i].Equal(oRG[i]) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (pg RateGroups) Clone() (cln RateGroups) {
|
|
cln = make(RateGroups, len(pg))
|
|
for i, rt := range pg {
|
|
cln[i] = new(RGRate)
|
|
*cln[i] = *rt
|
|
}
|
|
return
|
|
}
|
|
|
|
/*
|
|
Returns true if the received time result inside the interval
|
|
*/
|
|
func (i *RateInterval) Contains(t time.Time, endTime bool) bool {
|
|
if endTime {
|
|
if utils.TimeIs0h(t) { // back one second to 23:59:59
|
|
t = t.Add(-1 * time.Second)
|
|
}
|
|
}
|
|
return i.Timing.IsActiveAt(t)
|
|
}
|
|
|
|
func (i *RateInterval) String_DISABLED() string {
|
|
return fmt.Sprintf("%v %v %v %v %v %v", i.Timing.Years, i.Timing.Months, i.Timing.MonthDays, i.Timing.WeekDays, i.Timing.StartTime, i.Timing.EndTime)
|
|
}
|
|
|
|
func (i *RateInterval) Equal(o *RateInterval) bool {
|
|
if i == nil && o == nil {
|
|
return true
|
|
}
|
|
if i == nil || o == nil {
|
|
return false // considering the earlier test
|
|
}
|
|
if i.Weight != o.Weight {
|
|
return false
|
|
}
|
|
if i.Timing == nil && o.Timing == nil {
|
|
return true
|
|
}
|
|
return reflect.DeepEqual(i.Timing.Years, o.Timing.Years) &&
|
|
reflect.DeepEqual(i.Timing.Months, o.Timing.Months) &&
|
|
reflect.DeepEqual(i.Timing.MonthDays, o.Timing.MonthDays) &&
|
|
reflect.DeepEqual(i.Timing.WeekDays, o.Timing.WeekDays) &&
|
|
i.Timing.StartTime == o.Timing.StartTime &&
|
|
i.Timing.EndTime == o.Timing.EndTime
|
|
}
|
|
|
|
func (i *RateInterval) GetCost(duration, startSecond time.Duration) float64 {
|
|
price, _, rateUnit := i.GetRateParameters(startSecond)
|
|
price /= float64(rateUnit.Nanoseconds())
|
|
d := float64(duration.Nanoseconds())
|
|
return utils.Round(d*price, globalRoundingDecimals, utils.MetaRoundingMiddle)
|
|
}
|
|
|
|
// Gets the price for a the provided start second
|
|
func (i *RateInterval) GetRateParameters(startSecond time.Duration) (rate float64, rateIncrement, rateUnit time.Duration) {
|
|
if i.Rating == nil {
|
|
return -1, -1, -1
|
|
}
|
|
i.Rating.Rates.Sort()
|
|
for index, price := range i.Rating.Rates {
|
|
if price.GroupIntervalStart <= startSecond && (index == len(i.Rating.Rates)-1 ||
|
|
i.Rating.Rates[index+1].GroupIntervalStart > startSecond) {
|
|
if price.RateIncrement == 0 {
|
|
price.RateIncrement = 1 * time.Second
|
|
}
|
|
if price.RateUnit == 0 {
|
|
price.RateUnit = 1 * time.Second
|
|
}
|
|
return price.Value, price.RateIncrement, price.RateUnit
|
|
}
|
|
}
|
|
return -1, -1, -1
|
|
}
|
|
|
|
func (ri *RateInterval) GetMaxCost() (float64, string) {
|
|
if ri.Rating == nil {
|
|
return 0.0, ""
|
|
}
|
|
return ri.Rating.MaxCost, ri.Rating.MaxCostStrategy
|
|
}
|
|
|
|
// Structure to store intervals according to weight
|
|
type RateIntervalList []*RateInterval
|
|
|
|
func (rl RateIntervalList) GetWeight() float64 {
|
|
// all reates should have the same weight
|
|
// just in case get the max
|
|
var maxWeight float64
|
|
for _, r := range rl {
|
|
if r.Weight > maxWeight {
|
|
maxWeight = r.Weight
|
|
}
|
|
}
|
|
return maxWeight
|
|
}
|
|
|
|
// Structure to store intervals according to weight
|
|
type RateIntervalTimeSorter struct {
|
|
referenceTime time.Time
|
|
ris []*RateInterval
|
|
}
|
|
|
|
func (il *RateIntervalTimeSorter) Len() int {
|
|
return len(il.ris)
|
|
}
|
|
|
|
func (il *RateIntervalTimeSorter) Swap(i, j int) {
|
|
il.ris[i], il.ris[j] = il.ris[j], il.ris[i]
|
|
}
|
|
|
|
// we need higher weights earlyer in the list
|
|
func (il *RateIntervalTimeSorter) Less(j, i int) bool {
|
|
if il.ris[i].Weight < il.ris[j].Weight {
|
|
return il.ris[i].Weight < il.ris[j].Weight
|
|
}
|
|
t1 := il.ris[i].Timing.getLeftMargin(il.referenceTime)
|
|
t2 := il.ris[j].Timing.getLeftMargin(il.referenceTime)
|
|
return t1.After(t2)
|
|
}
|
|
|
|
func (il *RateIntervalTimeSorter) Sort() []*RateInterval {
|
|
sort.Sort(il)
|
|
return il.ris
|
|
}
|
|
|
|
// Clone clones RateInterval
|
|
func (i *RateInterval) Clone() (cln *RateInterval) {
|
|
if i == nil {
|
|
return
|
|
}
|
|
cln = &RateInterval{
|
|
Timing: i.Timing.Clone(),
|
|
Rating: i.Rating.Clone(),
|
|
Weight: i.Weight,
|
|
}
|
|
return
|
|
}
|
|
|
|
// Clone clones RITiming
|
|
func (rit *RITiming) Clone() (cln *RITiming) {
|
|
if rit == nil {
|
|
return
|
|
}
|
|
cln = &RITiming{
|
|
ID: rit.ID,
|
|
StartTime: rit.StartTime,
|
|
EndTime: rit.EndTime,
|
|
}
|
|
if len(rit.Years) != 0 {
|
|
cln.Years = make(utils.Years, len(rit.Years))
|
|
for i, year := range rit.Years {
|
|
cln.Years[i] = year
|
|
}
|
|
}
|
|
if len(rit.Months) != 0 {
|
|
cln.Months = make(utils.Months, len(rit.Months))
|
|
for i, month := range rit.Months {
|
|
cln.Months[i] = month
|
|
}
|
|
}
|
|
if len(rit.MonthDays) != 0 {
|
|
cln.MonthDays = make(utils.MonthDays, len(rit.MonthDays))
|
|
for i, monthDay := range rit.MonthDays {
|
|
cln.MonthDays[i] = monthDay
|
|
}
|
|
}
|
|
if len(rit.WeekDays) != 0 {
|
|
cln.WeekDays = make(utils.WeekDays, len(rit.WeekDays))
|
|
for i, weekDay := range rit.WeekDays {
|
|
cln.WeekDays[i] = weekDay
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// Clone clones RIRate
|
|
func (rit *RIRate) Clone() (cln *RIRate) {
|
|
if rit == nil {
|
|
return
|
|
}
|
|
cln = &RIRate{
|
|
ConnectFee: rit.ConnectFee,
|
|
RoundingMethod: rit.RoundingMethod,
|
|
RoundingDecimals: rit.RoundingDecimals,
|
|
MaxCost: rit.MaxCost,
|
|
MaxCostStrategy: rit.MaxCostStrategy,
|
|
}
|
|
if rit.Rates != nil {
|
|
cln.Rates = make([]*RGRate, len(rit.Rates))
|
|
for i, rate := range rit.Rates {
|
|
cln.Rates[i] = rate.Clone()
|
|
}
|
|
}
|
|
return cln
|
|
}
|
|
|
|
// Clone clones Rates
|
|
func (r *RGRate) Clone() (cln *RGRate) {
|
|
if r == nil {
|
|
return
|
|
}
|
|
cln = &RGRate{
|
|
GroupIntervalStart: r.GroupIntervalStart,
|
|
Value: r.Value,
|
|
RateIncrement: r.RateIncrement,
|
|
RateUnit: r.RateUnit,
|
|
}
|
|
return
|
|
}
|
|
|
|
func (rit *RITiming) FieldAsInterface(fldPath []string) (val interface{}, err error) {
|
|
if rit == nil || len(fldPath) == 0 {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
switch fldPath[0] {
|
|
default:
|
|
opath, indx := utils.GetPathIndex(fldPath[0])
|
|
if indx != nil {
|
|
switch opath {
|
|
case utils.YearsFieldName:
|
|
if len(fldPath) != 1 || len(rit.Years) <= *indx {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.Years[*indx], nil
|
|
case utils.MonthsFieldName:
|
|
if len(fldPath) != 1 || len(rit.Months) <= *indx {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.Months[*indx], nil
|
|
case utils.MonthDaysFieldName:
|
|
if len(fldPath) != 1 || len(rit.MonthDays) <= *indx {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.MonthDays[*indx], nil
|
|
case utils.WeekDaysFieldName:
|
|
if len(fldPath) != 1 || len(rit.WeekDays) <= *indx {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.WeekDays[*indx], nil
|
|
}
|
|
}
|
|
return nil, fmt.Errorf("unsupported field prefix: <%s>", fldPath[0])
|
|
case utils.ID:
|
|
if len(fldPath) != 1 {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.ID, nil
|
|
case utils.StartTime:
|
|
if len(fldPath) != 1 {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.StartTime, nil
|
|
case utils.EndTime:
|
|
if len(fldPath) != 1 {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.EndTime, nil
|
|
case utils.YearsFieldName:
|
|
switch len(fldPath) {
|
|
case 1:
|
|
return rit.Years, nil
|
|
case 2:
|
|
var idx int
|
|
if idx, err = strconv.Atoi(fldPath[1]); err != nil {
|
|
return
|
|
}
|
|
if len(rit.Years) <= idx {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.Years[idx], nil
|
|
default:
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
case utils.MonthsFieldName:
|
|
switch len(fldPath) {
|
|
case 1:
|
|
return rit.Months, nil
|
|
case 2:
|
|
var idx int
|
|
if idx, err = strconv.Atoi(fldPath[1]); err != nil {
|
|
return
|
|
}
|
|
if len(rit.Months) <= idx {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.Months[idx], nil
|
|
default:
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
case utils.MonthDaysFieldName:
|
|
switch len(fldPath) {
|
|
case 1:
|
|
return rit.MonthDays, nil
|
|
case 2:
|
|
var idx int
|
|
if idx, err = strconv.Atoi(fldPath[1]); err != nil {
|
|
return
|
|
}
|
|
if len(rit.MonthDays) <= idx {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.MonthDays[idx], nil
|
|
default:
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
case utils.WeekDaysFieldName:
|
|
switch len(fldPath) {
|
|
case 1:
|
|
return rit.WeekDays, nil
|
|
case 2:
|
|
var idx int
|
|
if idx, err = strconv.Atoi(fldPath[1]); err != nil {
|
|
return
|
|
}
|
|
if len(rit.WeekDays) <= idx {
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
return rit.WeekDays[idx], nil
|
|
default:
|
|
return nil, utils.ErrNotFound
|
|
}
|
|
}
|
|
}
|
|
|
|
func (rit *RITiming) FieldAsString(fldPath []string) (val string, err error) {
|
|
var iface interface{}
|
|
iface, err = rit.FieldAsInterface(fldPath)
|
|
if err != nil {
|
|
return
|
|
}
|
|
return utils.IfaceAsString(iface), nil
|
|
}
|