mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-16 21:59:53 +05:00
287 lines
9.9 KiB
Go
287 lines
9.9 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 mediator
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/csv"
|
|
"errors"
|
|
"flag"
|
|
"fmt"
|
|
"github.com/cgrates/cgrates/config"
|
|
"github.com/cgrates/cgrates/engine"
|
|
"github.com/cgrates/cgrates/utils"
|
|
"github.com/howeyc/fsnotify"
|
|
"os"
|
|
"path"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
func NewMediator(connector engine.Connector, storDb engine.DataStorage, cfg *config.CGRConfig) (m *Mediator, err error) {
|
|
m = &Mediator{
|
|
connector: connector,
|
|
storDb: storDb,
|
|
cgrCfg: cfg,
|
|
}
|
|
m.fieldNames = make(map[string][]string)
|
|
m.fieldIdxs = make(map[string][]int)
|
|
// Load config fields
|
|
if errLoad := m.loadConfig(); errLoad != nil {
|
|
return nil, errLoad
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
type Mediator struct {
|
|
connector engine.Connector
|
|
storDb engine.DataStorage
|
|
cgrCfg *config.CGRConfig
|
|
cdrInDir, cdrOutDir string
|
|
accIdField string
|
|
accIdIdx int // Populated only for csv files where we have no names but indexes for the fields
|
|
fieldNames map[string][]string
|
|
fieldIdxs map[string][]int // Populated only for csv files where we have no names but indexes for the fields
|
|
}
|
|
|
|
/*
|
|
Responsible for loading configuration fields out of CGRConfig instance, doing the necessary pre-checks.
|
|
@param fieldKeys: stores the keys which will be referenced inside fieldNames/fieldIdxs
|
|
@param cfgVals: keep ordered references to configuration fields from CGRConfig instance.
|
|
fieldKeys and cfgVals are directly related through index.
|
|
Method logic:
|
|
* Make sure the field used as reference in mediation process loop is not empty.
|
|
* All other fields should match the length of reference field.
|
|
* Accounting id field should not be empty.
|
|
* If we run mediation on csv file:
|
|
* Make sure cdrInDir and cdrOutDir are valid paths.
|
|
* Populate accIdIdx by converting accIdField into integer.
|
|
* Populate fieldIdxs by converting fieldNames into integers
|
|
*/
|
|
func (self *Mediator) loadConfig() error {
|
|
fieldKeys := []string{"subject", "reqtype", "direction", "tenant", "tor", "account", "destination", "time_start", "duration"}
|
|
cfgVals := [][]string{self.cgrCfg.MediatorSubjectFields, self.cgrCfg.MediatorReqTypeFields, self.cgrCfg.MediatorDirectionFields,
|
|
self.cgrCfg.MediatorTenantFields, self.cgrCfg.MediatorTORFields, self.cgrCfg.MediatorAccountFields, self.cgrCfg.MediatorDestFields,
|
|
self.cgrCfg.MediatorTimeAnswerFields, self.cgrCfg.MediatorDurationFields}
|
|
|
|
refIdx := 0 // Subject becomes reference for our checks
|
|
if len(cfgVals[refIdx]) == 0 {
|
|
return fmt.Errorf("Unconfigured %s fields", fieldKeys[refIdx])
|
|
}
|
|
// All other configured fields must match the length of reference fields
|
|
for iCfgVal := range cfgVals {
|
|
if len(cfgVals[refIdx]) != len(cfgVals[iCfgVal]) {
|
|
// Make sure we have everywhere the length of reference key (subject)
|
|
return errors.New("Inconsistent lenght of mediator fields.")
|
|
}
|
|
}
|
|
|
|
// AccIdField has no special requirements, should just exist
|
|
if self.cgrCfg.MediatorAccIdField == "" {
|
|
return errors.New("Undefined mediator accid field")
|
|
}
|
|
self.accIdField = self.cgrCfg.MediatorAccIdField
|
|
|
|
var errConv error
|
|
// Specific settings of CSV style CDRS
|
|
if self.cgrCfg.MediatorCDRType == utils.FSCDR_FILE_CSV {
|
|
// Check paths to be valid before adding as configuration
|
|
if _, err := os.Stat(self.cgrCfg.MediatorCDRInDir); err != nil {
|
|
return fmt.Errorf("The input path for mediator does not exist: %v", self.cgrCfg.MediatorCDRInDir)
|
|
} else {
|
|
self.cdrInDir = self.cgrCfg.MediatorCDRInDir
|
|
}
|
|
if _, err := os.Stat(self.cgrCfg.MediatorCDROutDir); err != nil {
|
|
return fmt.Errorf("The output path for mediator does not exist: %v", self.cgrCfg.MediatorCDROutDir)
|
|
} else {
|
|
self.cdrOutDir = self.cgrCfg.MediatorCDROutDir
|
|
}
|
|
if self.accIdIdx, errConv = strconv.Atoi(self.cgrCfg.MediatorAccIdField); errConv != nil {
|
|
return errors.New("AccIdIndex must be integer.")
|
|
}
|
|
}
|
|
|
|
// Load here field names and convert to integers in case of unamed cdrs like CSV
|
|
for idx, key := range fieldKeys {
|
|
self.fieldNames[key] = cfgVals[idx]
|
|
if self.cgrCfg.MediatorCDRType == utils.FSCDR_FILE_CSV { // Special case when field names represent indexes of their location in file
|
|
self.fieldIdxs[key] = make([]int, len(cfgVals[idx]))
|
|
for iStr, cfgStr := range cfgVals[idx] {
|
|
if self.fieldIdxs[key][iStr], errConv = strconv.Atoi(cfgStr); errConv != nil {
|
|
return fmt.Errorf("All mediator index members (%s) must be ints", key)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Watch the specified folder for file moves and parse the files on events
|
|
func (self *Mediator) TrackCDRFiles() (err error) {
|
|
watcher, err := fsnotify.NewWatcher()
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer watcher.Close()
|
|
err = watcher.Watch(self.cdrInDir)
|
|
if err != nil {
|
|
return
|
|
}
|
|
engine.Logger.Info(fmt.Sprintf("Monitoring %s for file moves.", self.cdrInDir))
|
|
for {
|
|
select {
|
|
case ev := <-watcher.Event:
|
|
if ev.IsCreate() && path.Ext(ev.Name) != ".csv" {
|
|
engine.Logger.Info(fmt.Sprintf("Parsing: %s", ev.Name))
|
|
err = self.MediateCSVCDR(ev.Name)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
case err := <-watcher.Error:
|
|
engine.Logger.Err(fmt.Sprintf("Inotify error: %s", err.Error()))
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// Retrive the cost from logging database
|
|
func (self *Mediator) getCostsFromDB(cdr utils.CDR) (cc *engine.CallCost, err error) {
|
|
return self.storDb.GetCallCostLog(cdr.GetCgrId(), engine.SESSION_MANAGER_SOURCE)
|
|
}
|
|
|
|
// Retrive the cost from engine
|
|
func (self *Mediator) getCostsFromRater(cdr utils.CDR) (*engine.CallCost, error) {
|
|
cc := &engine.CallCost{}
|
|
d, err := time.ParseDuration(strconv.FormatInt(cdr.GetDuration(), 10) + "s")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if d.Seconds() == 0 { // failed call, returning empty callcost, no error
|
|
return cc, nil
|
|
}
|
|
t1, err := cdr.GetAnswerTime()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
cd := engine.CallDescriptor{
|
|
Direction: "*out", //record[m.directionFields[runIdx]] TODO: fix me
|
|
Tenant: cdr.GetTenant(),
|
|
TOR: cdr.GetTOR(),
|
|
Subject: cdr.GetSubject(),
|
|
Account: cdr.GetAccount(),
|
|
Destination: cdr.GetDestination(),
|
|
TimeStart: t1,
|
|
TimeEnd: t1.Add(d)}
|
|
if cdr.GetReqType() == utils.PSEUDOPREPAID {
|
|
err = self.connector.Debit(cd, cc)
|
|
} else {
|
|
err = self.connector.GetCost(cd, cc)
|
|
}
|
|
if err != nil {
|
|
self.storDb.LogError(cdr.GetCgrId(), engine.MEDIATOR_SOURCE, err.Error())
|
|
} else {
|
|
// If the mediator calculated a price it will write it to logdb
|
|
self.storDb.LogCallCost(cdr.GetCgrId(), engine.MEDIATOR_SOURCE, cc)
|
|
}
|
|
return cc, err
|
|
}
|
|
|
|
// Parse the files and get cost for every record
|
|
func (self *Mediator) MediateCSVCDR(cdrfn string) (err error) {
|
|
flag.Parse()
|
|
file, err := os.Open(cdrfn)
|
|
defer file.Close()
|
|
if err != nil {
|
|
engine.Logger.Crit(err.Error())
|
|
os.Exit(1)
|
|
}
|
|
csvReader := csv.NewReader(bufio.NewReader(file))
|
|
|
|
_, fn := path.Split(cdrfn)
|
|
fout, err := os.Create(path.Join(self.cdrOutDir, fn))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer fout.Close()
|
|
|
|
w := bufio.NewWriter(fout)
|
|
for record, ok := csvReader.Read(); ok == nil; record, ok = csvReader.Read() {
|
|
//t, _ := time.Parse("2006-01-02 15:04:05", record[5])
|
|
var cc *engine.CallCost
|
|
|
|
for runIdx := range self.fieldIdxs["subject"] { // Query costs for every run index given by subject
|
|
csvCDR, errCDR := NewFScsvCDR(record, self.accIdIdx,
|
|
self.fieldIdxs["subject"][runIdx],
|
|
self.fieldIdxs["reqtype"][runIdx],
|
|
self.fieldIdxs["direction"][runIdx],
|
|
self.fieldIdxs["tenant"][runIdx],
|
|
self.fieldIdxs["tor"][runIdx],
|
|
self.fieldIdxs["account"][runIdx],
|
|
self.fieldIdxs["destination"][runIdx],
|
|
self.fieldIdxs["answer_time"][runIdx],
|
|
self.fieldIdxs["duration"][runIdx],
|
|
self.cgrCfg)
|
|
if errCDR != nil {
|
|
engine.Logger.Err(fmt.Sprintf("<Mediator> Could not calculate price for accid: <%s>, err: <%s>",
|
|
record[self.accIdIdx], errCDR.Error()))
|
|
}
|
|
var errCost error
|
|
if csvCDR.GetReqType() == utils.PREPAID || csvCDR.GetReqType() == utils.POSTPAID {
|
|
// Should be previously calculated and stored in DB
|
|
cc, errCost = self.getCostsFromDB(csvCDR)
|
|
} else {
|
|
cc, errCost = self.getCostsFromRater(csvCDR)
|
|
}
|
|
cost := "-1"
|
|
if errCost != nil || cc == nil {
|
|
engine.Logger.Err(fmt.Sprintf("<Mediator> Could not calculate price for accid: <%s>, err: <%s>, cost: <%v>", csvCDR.GetAccId(), err.Error(), cc))
|
|
} else {
|
|
cost = strconv.FormatFloat(cc.ConnectFee+cc.Cost, 'f', -1, 64)
|
|
engine.Logger.Debug(fmt.Sprintf("Calculated for accid:%s, cost: %v", csvCDR.GetAccId(), cost))
|
|
}
|
|
record = append(record, cost)
|
|
}
|
|
w.WriteString(strings.Join(record, ",") + "\n")
|
|
}
|
|
w.Flush()
|
|
return
|
|
}
|
|
|
|
func (self *Mediator) MediateDBCDR(cdr utils.CDR, db engine.DataStorage) error {
|
|
var cc *engine.CallCost
|
|
var errCost error
|
|
if cdr.GetReqType() == utils.PREPAID || cdr.GetReqType() == utils.POSTPAID {
|
|
// Should be previously calculated and stored in DB
|
|
cc, errCost = self.getCostsFromDB(cdr)
|
|
} else {
|
|
cc, errCost = self.getCostsFromRater(cdr)
|
|
}
|
|
cost := "-1"
|
|
if errCost != nil || cc == nil {
|
|
engine.Logger.Err(fmt.Sprintf("<Mediator> Could not calculate price for cgrid: <%s>, err: <%s>, cost: <%v>", cdr.GetCgrId(), errCost.Error(), cc))
|
|
} else {
|
|
cost = strconv.FormatFloat(cc.ConnectFee+cc.Cost, 'f', -1, 64)
|
|
engine.Logger.Debug(fmt.Sprintf("<Mediator> Calculated for cgrid:%s, cost: %v", cdr.GetCgrId(), cost))
|
|
}
|
|
return self.storDb.SetRatedCdr(cdr, cc)
|
|
}
|