mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-14 20:59:53 +05:00
180 lines
5.2 KiB
Go
180 lines
5.2 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"
|
|
"sort"
|
|
"time"
|
|
|
|
"github.com/cgrates/cgrates/config"
|
|
"github.com/cgrates/cgrates/guardian"
|
|
"github.com/cgrates/cgrates/utils"
|
|
)
|
|
|
|
type AliasEntry struct {
|
|
FieldName string
|
|
Initial string
|
|
Alias string
|
|
}
|
|
|
|
type AliasProfile struct {
|
|
Tenant string
|
|
ID string
|
|
FilterIDs []string
|
|
ActivationInterval *utils.ActivationInterval // Activation interval
|
|
Aliases map[string]map[string]string // map[FieldName][InitialValue]AliasValue
|
|
Weight float64
|
|
}
|
|
|
|
func (als *AliasProfile) TenantID() string {
|
|
return utils.ConcatenatedKey(als.Tenant, als.ID)
|
|
}
|
|
|
|
// AliasProfiles is a sortable list of Alias profiles
|
|
type AliasProfiles []*AliasProfile
|
|
|
|
// Sort is part of sort interface, sort based on Weight
|
|
func (aps AliasProfiles) Sort() {
|
|
sort.Slice(aps, func(i, j int) bool { return aps[i].Weight > aps[j].Weight })
|
|
}
|
|
|
|
func NewAliasService(dm *DataManager, filterS *FilterS, indexedFields []string) (*AliasService, error) {
|
|
return &AliasService{dm: dm, filterS: filterS, indexedFields: indexedFields}, nil
|
|
}
|
|
|
|
type AliasService struct {
|
|
dm *DataManager
|
|
filterS *FilterS
|
|
indexedFields []string
|
|
}
|
|
|
|
// ListenAndServe will initialize the service
|
|
func (alS *AliasService) ListenAndServe(exitChan chan bool) (err error) {
|
|
utils.Logger.Info("Starting Alias service")
|
|
e := <-exitChan
|
|
exitChan <- e // put back for the others listening for shutdown request
|
|
return
|
|
}
|
|
|
|
// Shutdown is called to shutdown the service
|
|
func (alS *AliasService) Shutdown() (err error) {
|
|
utils.Logger.Info(fmt.Sprintf("<%s> shutdown initialized", utils.AliasS))
|
|
utils.Logger.Info(fmt.Sprintf("<%s> shutdown complete", utils.AliasS))
|
|
return
|
|
}
|
|
|
|
// matchingSupplierProfilesForEvent returns ordered list of matching resources which are active by the time of the call
|
|
func (alS *AliasService) matchingAliasProfilesForEvent(ev *utils.CGREvent) (aPrfls AliasProfiles, err error) {
|
|
matchingAPs := make(map[string]*AliasProfile)
|
|
aPrflIDs, err := matchingItemIDsForEvent(ev.Event, alS.indexedFields,
|
|
alS.dm, utils.AliasProfilesStringIndex+ev.Tenant)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
lockIDs := utils.PrefixSliceItems(aPrflIDs.Slice(), utils.AliasProfilesStringIndex)
|
|
guardian.Guardian.GuardIDs(config.CgrConfig().LockingTimeout, lockIDs...)
|
|
defer guardian.Guardian.UnguardIDs(lockIDs...)
|
|
for apID := range aPrflIDs {
|
|
aPrfl, err := alS.dm.GetAliasProfile(ev.Tenant, apID, false, utils.NonTransactional)
|
|
if err != nil {
|
|
if err == utils.ErrNotFound {
|
|
continue
|
|
}
|
|
return nil, err
|
|
}
|
|
evTime := time.Now()
|
|
if ev.Time != nil {
|
|
evTime = *ev.Time
|
|
}
|
|
if aPrfl.ActivationInterval != nil &&
|
|
!aPrfl.ActivationInterval.IsActiveAtTime(evTime) { // not active
|
|
continue
|
|
}
|
|
if pass, err := alS.filterS.PassFiltersForEvent(ev.Tenant,
|
|
ev.Event, aPrfl.FilterIDs); err != nil {
|
|
return nil, err
|
|
} else if !pass {
|
|
continue
|
|
}
|
|
matchingAPs[apID] = aPrfl
|
|
}
|
|
// All good, convert from Map to Slice so we can sort
|
|
aPrfls = make(AliasProfiles, len(matchingAPs))
|
|
i := 0
|
|
for _, aPrfl := range matchingAPs {
|
|
aPrfls[i] = aPrfl
|
|
i++
|
|
}
|
|
aPrfls.Sort()
|
|
return
|
|
}
|
|
|
|
func (alS *AliasService) V1ProcessEvent(ev *utils.CGREvent,
|
|
reply *string) (err error) {
|
|
return
|
|
}
|
|
|
|
type ExternalAliasProfile struct {
|
|
Tenant string
|
|
ID string
|
|
FilterIDs []string
|
|
ActivationInterval *utils.ActivationInterval // Activation interval
|
|
Aliases []*AliasEntry
|
|
Weight float64
|
|
}
|
|
|
|
func (eap *ExternalAliasProfile) AsAliasProfile() *AliasProfile {
|
|
alsPrf := &AliasProfile{
|
|
Tenant: eap.Tenant,
|
|
ID: eap.ID,
|
|
Weight: eap.Weight,
|
|
FilterIDs: eap.FilterIDs,
|
|
ActivationInterval: eap.ActivationInterval,
|
|
}
|
|
alsMap := make(map[string]map[string]string)
|
|
for _, als := range eap.Aliases {
|
|
alsMap[als.FieldName] = make(map[string]string)
|
|
alsMap[als.FieldName][als.Initial] = als.Alias
|
|
}
|
|
alsPrf.Aliases = alsMap
|
|
return alsPrf
|
|
}
|
|
|
|
func NewExternalAliasProfileFromAliasProfile(alsPrf *AliasProfile) *ExternalAliasProfile {
|
|
extals := &ExternalAliasProfile{
|
|
Tenant: alsPrf.Tenant,
|
|
ID: alsPrf.ID,
|
|
Weight: alsPrf.Weight,
|
|
ActivationInterval: alsPrf.ActivationInterval,
|
|
FilterIDs: alsPrf.FilterIDs,
|
|
}
|
|
for key, val := range alsPrf.Aliases {
|
|
for key2, val2 := range val {
|
|
extals.Aliases = append(extals.Aliases, &AliasEntry{
|
|
FieldName: key,
|
|
Initial: key2,
|
|
Alias: val2,
|
|
})
|
|
}
|
|
}
|
|
return extals
|
|
|
|
}
|