mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-11 18:16:24 +05:00
166 lines
4.8 KiB
Go
166 lines
4.8 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 actions
|
|
|
|
import (
|
|
"cmp"
|
|
"fmt"
|
|
"slices"
|
|
|
|
"github.com/cgrates/birpc/context"
|
|
"github.com/cgrates/cgrates/config"
|
|
"github.com/cgrates/cgrates/engine"
|
|
"github.com/cgrates/cgrates/utils"
|
|
)
|
|
|
|
// actSetBalance will update the account
|
|
type actSetBalance struct {
|
|
config *config.CGRConfig
|
|
connMgr *engine.ConnManager
|
|
fltrS *engine.FilterS
|
|
aCfg *utils.APAction
|
|
tnt string
|
|
reset bool
|
|
}
|
|
|
|
func (aL *actSetBalance) id() string {
|
|
return aL.aCfg.ID
|
|
}
|
|
|
|
func (aL *actSetBalance) cfg() *utils.APAction {
|
|
return aL.aCfg
|
|
}
|
|
|
|
// execute implements actioner interface
|
|
func (aL *actSetBalance) execute(ctx *context.Context, data utils.MapStorage, trgID string) (err error) {
|
|
if len(aL.config.ActionSCfg().AccountSConns) == 0 {
|
|
return fmt.Errorf("no connection with AccountS")
|
|
}
|
|
|
|
weights := make(map[string]float64) // stores sorting weights by Diktat ID
|
|
diktats := make([]*utils.APDiktat, 0) // list of diktats which have *balancePath in opts, will be weight sorted later
|
|
for _, diktat := range aL.cfg().Diktats {
|
|
if pass, err := aL.fltrS.Pass(ctx, aL.tnt, diktat.FilterIDs, data); err != nil {
|
|
return err
|
|
} else if !pass {
|
|
continue
|
|
}
|
|
weight, err := engine.WeightFromDynamics(ctx, diktat.Weights, aL.fltrS, aL.tnt, data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
weights[diktat.ID] = weight
|
|
diktats = append(diktats, diktat)
|
|
}
|
|
// Sort by weight (higher values first).
|
|
slices.SortFunc(diktats, func(a, b *utils.APDiktat) int {
|
|
return cmp.Compare(weights[b.ID], weights[a.ID])
|
|
})
|
|
args := &utils.ArgsActSetBalance{
|
|
Tenant: aL.tnt,
|
|
AccountID: trgID,
|
|
Reset: aL.reset,
|
|
Diktats: make([]*utils.BalDiktat, 0),
|
|
APIOpts: aL.cfg().Opts,
|
|
}
|
|
for _, actD := range diktats {
|
|
var rsr utils.RSRParsers
|
|
if rsr, err = actD.RSRValues(); err != nil {
|
|
return
|
|
}
|
|
var val string
|
|
if val, err = rsr.ParseDataProvider(data); err != nil {
|
|
return
|
|
}
|
|
args.Diktats = append(args.Diktats, &utils.BalDiktat{
|
|
Path: utils.IfaceAsString(actD.Opts[utils.MetaBalancePath]),
|
|
Value: val,
|
|
})
|
|
if blocker, err := engine.BlockerFromDynamics(ctx, actD.Blockers, aL.fltrS, aL.tnt, data); err != nil {
|
|
return err
|
|
} else if blocker {
|
|
break
|
|
}
|
|
}
|
|
var rply string
|
|
return aL.connMgr.Call(ctx, aL.config.ActionSCfg().AccountSConns,
|
|
utils.AccountSv1ActionSetBalance, args, &rply)
|
|
}
|
|
|
|
// actRemBalance will remove multiple balances from account
|
|
type actRemBalance struct {
|
|
config *config.CGRConfig
|
|
connMgr *engine.ConnManager
|
|
fltrS *engine.FilterS
|
|
aCfg *utils.APAction
|
|
tnt string
|
|
}
|
|
|
|
func (aL *actRemBalance) id() string {
|
|
return aL.aCfg.ID
|
|
}
|
|
|
|
func (aL *actRemBalance) cfg() *utils.APAction {
|
|
return aL.aCfg
|
|
}
|
|
|
|
// execute implements actioner interface
|
|
func (aL *actRemBalance) execute(ctx *context.Context, data utils.MapStorage, trgID string) (err error) {
|
|
if len(aL.config.ActionSCfg().AccountSConns) == 0 {
|
|
return fmt.Errorf("no connection with AccountS")
|
|
}
|
|
|
|
weights := make(map[string]float64) // stores sorting weights by Diktat ID
|
|
diktats := make([]*utils.APDiktat, 0) // list of diktats which have *balancePath in opts, will be weight sorted later
|
|
for _, diktat := range aL.cfg().Diktats {
|
|
if pass, err := aL.fltrS.Pass(ctx, aL.tnt, diktat.FilterIDs, data); err != nil {
|
|
return err
|
|
} else if !pass {
|
|
continue
|
|
}
|
|
weight, err := engine.WeightFromDynamics(ctx, diktat.Weights, aL.fltrS, aL.tnt, data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
weights[diktat.ID] = weight
|
|
diktats = append(diktats, diktat)
|
|
}
|
|
// Sort by weight (higher values first).
|
|
slices.SortFunc(diktats, func(a, b *utils.APDiktat) int {
|
|
return cmp.Compare(weights[b.ID], weights[a.ID])
|
|
})
|
|
args := &utils.ArgsActRemoveBalances{
|
|
Tenant: aL.tnt,
|
|
AccountID: trgID,
|
|
BalanceIDs: make([]string, len(diktats)),
|
|
APIOpts: aL.cfg().Opts,
|
|
}
|
|
for i, actD := range diktats {
|
|
args.BalanceIDs[i] = utils.IfaceAsString(actD.Opts[utils.MetaBalancePath])
|
|
if blocker, err := engine.BlockerFromDynamics(ctx, actD.Blockers, aL.fltrS, aL.tnt, data); err != nil {
|
|
return err
|
|
} else if blocker {
|
|
break
|
|
}
|
|
}
|
|
var rply string
|
|
return aL.connMgr.Call(ctx, aL.config.ActionSCfg().AccountSConns,
|
|
utils.AccountSv1ActionRemoveBalance, args, &rply)
|
|
}
|