Files
cgrates/rates/rates.go
2020-06-17 09:50:36 +02:00

170 lines
5.1 KiB
Go

/*
Real-time Online/Offline Charging System (OerS) 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 rates
import (
"fmt"
"sort"
"time"
"github.com/cgrates/cgrates/config"
"github.com/cgrates/cgrates/engine"
"github.com/cgrates/cgrates/utils"
)
// NewRateS instantiates the RateS
func NewRateS(cfg *config.CGRConfig, filterS *engine.FilterS, dm *engine.DataManager) *RateS {
return &RateS{
cfg: cfg,
filterS: filterS,
dm: dm,
}
}
// RateS calculates costs for events
type RateS struct {
cfg *config.CGRConfig
filterS *engine.FilterS
dm *engine.DataManager
}
// ListenAndServe keeps the service alive
func (rS *RateS) ListenAndServe(exitChan chan bool, cfgRld chan struct{}) (err error) {
utils.Logger.Info(fmt.Sprintf("<%s> starting <%s>",
utils.CoreS, utils.RateS))
for {
select {
case e := <-exitChan: // global exit
rS.Shutdown()
exitChan <- e // put back for the others listening for shutdown request
return
case rld := <-cfgRld: // configuration was reloaded
cfgRld <- rld
}
}
}
// Shutdown is called to shutdown the service
func (rS *RateS) Shutdown() (err error) {
utils.Logger.Info(fmt.Sprintf("<%s> shutdown <%s>", utils.CoreS, utils.RateS))
return
}
// Call implements rpcclient.ClientConnector interface for internal RPC
func (rS *RateS) Call(serviceMethod string, args interface{}, reply interface{}) error {
return utils.RPCCall(rS, serviceMethod, args, reply)
}
// matchingRateProfileForEvent returns the matched RateProfile for the given event
func (rS *RateS) matchingRateProfileForEvent(args *ArgsCostForEvent) (rtPfl *engine.RateProfile, err error) {
rPfIDs := args.RateProfileIDs
if len(rPfIDs) == 0 {
var rPfIDMp utils.StringSet
if rPfIDMp, err = engine.MatchingItemIDsForEvent(
args.CGREvent.Event,
rS.cfg.RateSCfg().StringIndexedFields,
rS.cfg.RateSCfg().PrefixIndexedFields,
rS.dm,
utils.CacheRateProfilesFilterIndexes,
args.CGREvent.Tenant,
rS.cfg.RateSCfg().IndexedSelects,
rS.cfg.RateSCfg().NestedFields,
); err != nil {
return
}
rPfIDs = rPfIDMp.AsSlice()
}
matchingRPfs := make([]*engine.RateProfile, 0, len(rPfIDs))
evNm := utils.MapStorage{utils.MetaReq: args.CGREvent.Event}
for _, rPfID := range rPfIDs {
var rPf *engine.RateProfile
if rPf, err = rS.dm.GetRateProfile(args.CGREvent.Tenant, rPfID, true, true, utils.NonTransactional); err != nil {
if err == utils.ErrNotFound {
err = nil
continue
}
return
}
if rPf.ActivationInterval != nil && args.CGREvent.Time != nil &&
!rPf.ActivationInterval.IsActiveAtTime(*args.CGREvent.Time) { // not active
continue
}
var pass bool
if pass, err = rS.filterS.Pass(args.CGREvent.Tenant, rPf.FilterIDs, evNm); err != nil {
return
} else if !pass {
continue
}
matchingRPfs = append(matchingRPfs, rPf)
}
if len(matchingRPfs) == 0 {
return nil, utils.ErrNotFound
}
sort.Slice(matchingRPfs, func(i, j int) bool { return matchingRPfs[i].Weight > matchingRPfs[j].Weight })
rtPfl = matchingRPfs[0]
return
}
// matchingRatesForEvent returns the matched Rate out of a RateProfile for the given event
// indexed based on intervalStart, there will be one winner per interval start
// returned in order of intervalStart
func (rS *RateS) matchingRatesForEvent(rtPfl *engine.RateProfile, cgrEv *utils.CGREvent) (rts []*engine.Rate, err error) {
var rtIDs utils.StringSet
if rtIDs, err = engine.MatchingItemIDsForEvent(
cgrEv.Event,
rS.cfg.RateSCfg().RateStringIndexedFields,
rS.cfg.RateSCfg().RatePrefixIndexedFields,
rS.dm,
utils.CacheRateFilterIndexes,
utils.ConcatenatedKey(cgrEv.Tenant, rtPfl.ID),
rS.cfg.RateSCfg().RateIndexedSelects,
rS.cfg.RateSCfg().RateNestedFields,
); err != nil {
return
}
rtsWrk := make(map[time.Duration][]*engine.Rate)
evNm := utils.MapStorage{utils.MetaReq: cgrEv.Event}
for rtID := range rtIDs {
rt := rtPfl.Rates[rtID] // pick the rate directly from map based on matched ID
var pass bool
if pass, err = rS.filterS.Pass(cgrEv.Tenant, rt.FilterIDs, evNm); err != nil {
return
} else if !pass {
continue
}
rtsWrk[rt.IntervalStart] = append(rtsWrk[rt.IntervalStart], rt)
}
rts = orderRatesOnIntervals(rtsWrk)
return
}
// AttrArgsProcessEvent arguments used for proccess event
type ArgsCostForEvent struct {
RateProfileIDs []string
Opts map[string]interface{}
*utils.CGREvent
*utils.ArgDispatcher
}
// V1CostForEvent will be called to calculate the cost for an event
func (rS *RateS) V1CostForEvent(args *ArgsCostForEvent, cC *utils.ChargedCost) (err error) {
return
}