Files
cgrates/apier/tpdestinations.go
Radu Ioan Fericean a2cae467fe updated docs
2013-06-18 12:28:44 +03:00

228 lines
5.6 KiB
Go

/*
Rating system designed to be used in VoIP Carriers World
Copyright (C) 2013 ITsysCOM
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 apier
import (
"errors"
"fmt"
"github.com/cgrates/cgrates/rater"
"github.com/cgrates/cgrates/utils"
)
type AttrGetTPDestinations struct {
TPid string
DestinationsTag string
}
// Return destinations profile for a destination tag received as parameter
func (self *Apier) GetTPDestinations(attrs AttrGetTPDestinations, reply *rater.Destination) error {
if missing := utils.MissingStructFields(&attrs, []string{"TPid", "DestinationsTag"}); len(missing) != 0 { //Params missing
return fmt.Errorf("%s:%v", utils.ERR_MANDATORY_IE_MISSING, missing)
}
if dst, err := self.StorDb.GetTPDestination(attrs.TPid, attrs.DestinationsTag); err != nil {
return fmt.Errorf("%s:%s", utils.ERR_SERVER_ERROR, err.Error())
} else if dst == nil {
return errors.New(utils.ERR_NOT_FOUND)
} else {
*reply = *dst
}
return nil
}
type AttrDestination struct {
Id string
Prefixes []string
}
func (self *Apier) GetDestination(attr *AttrDestination, reply *AttrDestination) error {
if dst, err := self.Getter.GetDestination(attr.Id); err != nil {
return errors.New(utils.ERR_NOT_FOUND)
} else {
reply.Id = dst.Id
reply.Prefixes = dst.Prefixes
}
return nil
}
func (self *Apier) SetDestination(attr *AttrDestination, reply *float64) error {
d := &rater.Destination{
Id: attr.Id,
Prefixes: attr.Prefixes,
}
if err := self.Getter.SetDestination(d); err != nil {
return err
}
return nil
}
type AttrGetBalance struct {
Tenant string
Account string
BalanceId string
Direction string
}
// Get balance
func (self *Apier) GetBalance(attr *AttrGetBalance, reply *float64) error {
tag := fmt.Sprintf("%s:%s:%s", attr.Direction, attr.Tenant, attr.Account)
userBalance, err := self.Getter.GetUserBalance(tag)
if err != nil {
return err
}
if attr.Direction == "" {
attr.Direction = rater.OUTBOUND
}
if balance, balExists := userBalance.BalanceMap[attr.BalanceId+attr.Direction]; !balExists {
// No match, balanceId not found
return errors.New(utils.ERR_NOT_FOUND)
} else {
*reply = balance
}
return nil
}
type AttrAddBalance struct {
Tenant string
Account string
BalanceId string
Direction string
Value float64
}
func (self *Apier) AddBalance(attr *AttrAddBalance, reply *float64) error {
// what storage instance do we use?
tag := fmt.Sprintf("%s:%s:%s", attr.Direction, attr.Tenant, attr.Account)
at := &rater.ActionTiming{
UserBalanceIds: []string{tag},
}
if attr.Direction == "" {
attr.Direction = rater.OUTBOUND
}
at.SetActions(rater.Actions{&rater.Action{BalanceId: attr.BalanceId, Direction: attr.Direction, Units: attr.Value}})
if err := at.Execute(); err != nil {
return err
}
// what to put in replay?
return nil
}
type AttrExecuteAction struct {
Direction string
Tenant string
Account string
BalanceId string
ActionsId string
}
func (self *Apier) ExecuteAction(attr *AttrExecuteAction, reply *float64) error {
tag := fmt.Sprintf("%s:%s:%s", attr.Direction, attr.Tenant, attr.Account)
at := &rater.ActionTiming{
UserBalanceIds: []string{tag},
ActionsId: attr.ActionsId,
}
if err := at.Execute(); err != nil {
return err
}
// what to put in replay
return nil
}
type AttrSetRatingProfile struct {
Direction string
Tenant string
TOR string
Subject string
RatingProfileId string
}
func (self *Apier) SetRatingProfile(attr *AttrSetRatingProfile, reply *float64) error {
cd := &rater.CallDescriptor{
Direction: attr.Direction,
Tenant: attr.Tenant,
TOR: attr.TOR,
Subject: attr.Subject,
}
subject, err := self.Getter.GetRatingProfile(cd.GetKey())
if err != nil {
return err
}
rp, err := self.Getter.GetRatingProfile(attr.RatingProfileId)
if err != nil {
return err
}
subject.DestinationMap = rp.DestinationMap
err = self.Getter.SetRatingProfile(subject)
return err
}
type AttrActionTrigger struct {
Tenant string
Account string
Direction string
BalanceId string
ThresholdValue float64
DestinationId string
Weight float64
ActionsId string
}
func (self *Apier) AddTriggeredAction(attr *AttrActionTrigger, reply *float64) error {
if attr.Direction == "" {
attr.Direction = rater.OUTBOUND
}
at := &rater.ActionTrigger{
Id: utils.GenUUID(),
BalanceId: attr.BalanceId,
Direction: attr.Direction,
ThresholdValue: attr.ThresholdValue,
DestinationId: attr.DestinationId,
Weight: attr.Weight,
ActionsId: attr.ActionsId,
Executed: false,
}
tag := fmt.Sprintf("%s:%s:%s", attr.Direction, attr.Tenant, attr.Account)
var dbErr error
rater.AccLock.Guard(tag, func() (float64, error) {
userBalance, err := self.Getter.GetUserBalance(tag)
if err != nil {
dbErr = err
return 0, err
}
userBalance.ActionTriggers = append(userBalance.ActionTriggers, at)
if err = self.Getter.SetUserBalance(userBalance); err != nil {
dbErr = err
return 0, err
}
return 0, nil
})
return dbErr
}