mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-15 05:09:54 +05:00
170 lines
5.1 KiB
Go
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
|
|
}
|