mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-11 18:16:24 +05:00
164 lines
4.1 KiB
Go
164 lines
4.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 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 engine
|
|
|
|
import (
|
|
"math"
|
|
)
|
|
|
|
/*
|
|
The struture that is saved to storage.
|
|
*/
|
|
type RatingPlan struct {
|
|
Id string
|
|
Timings map[string]*RITiming
|
|
Ratings map[string]*RIRate
|
|
DestinationRates map[string]RPRateList
|
|
}
|
|
|
|
type RPRate struct {
|
|
Timing string
|
|
Rating string
|
|
Weight float64
|
|
}
|
|
|
|
func (rpr *RPRate) Equal(orpr *RPRate) bool {
|
|
return rpr.Timing == orpr.Timing && rpr.Rating == orpr.Rating && rpr.Weight == orpr.Weight
|
|
}
|
|
|
|
type RPRateList []*RPRate
|
|
|
|
func (rp *RatingPlan) RateIntervalList(dId string) RateIntervalList {
|
|
ril := make(RateIntervalList, len(rp.DestinationRates[dId]))
|
|
for i, rpr := range rp.DestinationRates[dId] {
|
|
ril[i] = &RateInterval{
|
|
Timing: rp.Timings[rpr.Timing],
|
|
Rating: rp.Ratings[rpr.Rating],
|
|
Weight: rpr.Weight,
|
|
}
|
|
}
|
|
return ril
|
|
}
|
|
|
|
// no sorter because it's sorted with RateIntervalTimeSorter
|
|
|
|
/*
|
|
Adds one ore more intervals to the internal interval list only if it is not already in the list.
|
|
*/
|
|
func (rp *RatingPlan) AddRateInterval(dId string, ris ...*RateInterval) {
|
|
if rp.DestinationRates == nil {
|
|
rp.Timings = make(map[string]*RITiming)
|
|
rp.Ratings = make(map[string]*RIRate)
|
|
rp.DestinationRates = make(map[string]RPRateList, 1)
|
|
}
|
|
for _, ri := range ris {
|
|
rpr := &RPRate{Weight: ri.Weight}
|
|
if ri.Timing != nil {
|
|
timingTag := ri.Timing.Stringify()
|
|
rp.Timings[timingTag] = ri.Timing
|
|
rpr.Timing = timingTag
|
|
}
|
|
if ri.Rating != nil {
|
|
ratingTag := ri.Rating.Stringify()
|
|
rp.Ratings[ratingTag] = ri.Rating
|
|
rpr.Rating = ratingTag
|
|
}
|
|
found := false
|
|
for _, erpr := range rp.DestinationRates[dId] {
|
|
if erpr.Equal(rpr) {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
rp.DestinationRates[dId] = append(rp.DestinationRates[dId], rpr)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (rp *RatingPlan) Equal(o *RatingPlan) bool {
|
|
return rp.Id == o.Id
|
|
}
|
|
|
|
// IsValid determines if the rating plan covers a continuous period of time
|
|
func (rp *RatingPlan) isContinous() bool {
|
|
weekdays := make([]int, 7)
|
|
for _, tm := range rp.Timings {
|
|
// if it is a blank timing than it will match all
|
|
if tm.IsBlank() {
|
|
return true
|
|
}
|
|
// skip the special timings (for specific dates)
|
|
if len(tm.Years) != 0 || len(tm.Months) != 0 || len(tm.MonthDays) != 0 {
|
|
continue
|
|
}
|
|
// if the startime is not midnight than is an extra time
|
|
if tm.StartTime != "00:00:00" {
|
|
continue
|
|
}
|
|
//check if all weekdays are covered
|
|
for _, wd := range tm.WeekDays {
|
|
weekdays[wd] = 1
|
|
}
|
|
allWeekdaysCovered := true
|
|
for _, wd := range weekdays {
|
|
if wd != 1 {
|
|
allWeekdaysCovered = false
|
|
break
|
|
}
|
|
}
|
|
if allWeekdaysCovered {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (rp *RatingPlan) getFirstUnsaneRating() string {
|
|
for _, rating := range rp.Ratings {
|
|
rating.Rates.Sort()
|
|
for i, rate := range rating.Rates {
|
|
if i < (len(rating.Rates) - 1) {
|
|
nextRate := rating.Rates[i+1]
|
|
if nextRate.GroupIntervalStart <= rate.GroupIntervalStart {
|
|
return rating.tag
|
|
}
|
|
if math.Mod(float64(nextRate.GroupIntervalStart.Nanoseconds()),
|
|
float64(rate.RateIncrement.Nanoseconds())) != 0 {
|
|
return rating.tag
|
|
}
|
|
if rate.RateUnit == 0 || rate.RateIncrement == 0 {
|
|
return rating.tag
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func (rp *RatingPlan) getFirstUnsaneTiming() string {
|
|
for _, timing := range rp.Timings {
|
|
if (len(timing.Years) != 0 || len(timing.Months) != 0 || len(timing.MonthDays) != 0) &&
|
|
len(timing.WeekDays) != 0 {
|
|
return timing.tag
|
|
}
|
|
}
|
|
return ""
|
|
}
|