mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-11 18:16:24 +05:00
184 lines
5.0 KiB
Go
184 lines
5.0 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 migrator
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/cgrates/cgrates/engine"
|
|
"github.com/cgrates/cgrates/utils"
|
|
)
|
|
|
|
func (m *Migrator) migrateCurrentSessionSCost() (err error) {
|
|
if m.sameStorDB { // no move
|
|
return
|
|
}
|
|
smCosts, err := m.storDBIn.StorDB().GetSMCosts("", "", "", "")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, smCost := range smCosts {
|
|
if err := m.storDBOut.StorDB().SetSMCost(smCost); err != nil {
|
|
return err
|
|
}
|
|
if err := m.storDBIn.StorDB().RemoveSMCost(smCost); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateSessionSCosts() (err error) {
|
|
var vrs engine.Versions
|
|
current := engine.CurrentStorDBVersions()
|
|
vrs, err = m.storDBOut.StorDB().GetVersions("")
|
|
if err != nil {
|
|
return utils.NewCGRError(utils.Migrator,
|
|
utils.ServerErrorCaps,
|
|
err.Error(),
|
|
fmt.Sprintf("error: <%s> when querying OutStorDB for versions", err.Error()))
|
|
} else if len(vrs) == 0 {
|
|
return utils.NewCGRError(utils.Migrator,
|
|
utils.MandatoryIEMissingCaps,
|
|
utils.UndefinedVersion,
|
|
"version number is not defined for SessionsCosts model")
|
|
}
|
|
switch vrs[utils.SessionSCosts] {
|
|
case 0, 1:
|
|
if err = m.migrateV1SessionSCosts(); err != nil {
|
|
return err
|
|
}
|
|
case 2:
|
|
if err = m.migrateV2SessionSCosts(); err != nil {
|
|
return err
|
|
}
|
|
case current[utils.SessionSCosts]:
|
|
if err = m.migrateCurrentSessionSCost(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return m.ensureIndexesStorDB(utils.SessionCostsTBL)
|
|
}
|
|
|
|
func (m *Migrator) migrateV1SessionSCosts() (err error) {
|
|
if err = m.storDBIn.renameV1SMCosts(); err != nil {
|
|
return err
|
|
}
|
|
if m.dryRun {
|
|
return
|
|
}
|
|
vrs := engine.Versions{utils.SessionSCosts: 2}
|
|
if err = m.storDBOut.StorDB().SetVersions(vrs, false); err != nil {
|
|
return utils.NewCGRError(utils.Migrator,
|
|
utils.ServerErrorCaps,
|
|
err.Error(),
|
|
fmt.Sprintf("error: <%s> when updating SessionSCosts version into StorDB", err.Error()))
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateV2SessionSCosts() (err error) {
|
|
var v2Cost *v2SessionsCost
|
|
for {
|
|
v2Cost, err = m.storDBIn.getV2SMCost()
|
|
if err != nil && err != utils.ErrNoMoreData {
|
|
return err
|
|
}
|
|
if err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
if v2Cost == nil || m.dryRun {
|
|
continue
|
|
}
|
|
smCost := v2Cost.V2toV3Cost()
|
|
if err = m.storDBOut.StorDB().SetSMCost(smCost); err != nil {
|
|
return err
|
|
}
|
|
if err = m.storDBIn.remV2SMCost(v2Cost); err != nil {
|
|
return err
|
|
}
|
|
m.stats[utils.SessionSCosts] += 1
|
|
}
|
|
if m.dryRun {
|
|
return
|
|
}
|
|
// All done, update version with current one
|
|
vrs := engine.Versions{utils.SessionSCosts: engine.CurrentStorDBVersions()[utils.SessionSCosts]}
|
|
if err = m.storDBOut.StorDB().SetVersions(vrs, false); err != nil {
|
|
return utils.NewCGRError(utils.Migrator,
|
|
utils.ServerErrorCaps,
|
|
err.Error(),
|
|
fmt.Sprintf("error: <%s> when updating SessionSCosts version into StorDB", err.Error()))
|
|
}
|
|
return
|
|
}
|
|
|
|
type v2SessionsCost struct {
|
|
CGRID string
|
|
RunID string
|
|
OriginHost string
|
|
OriginID string
|
|
CostSource string
|
|
Usage time.Duration
|
|
CostDetails *engine.CallCost
|
|
}
|
|
|
|
func (v2Cost *v2SessionsCost) V2toV3Cost() (cost *engine.SMCost) {
|
|
cost = &engine.SMCost{
|
|
CGRID: v2Cost.CGRID,
|
|
RunID: v2Cost.RunID,
|
|
OriginHost: v2Cost.OriginHost,
|
|
OriginID: v2Cost.OriginID,
|
|
Usage: v2Cost.Usage,
|
|
CostSource: v2Cost.CostSource,
|
|
CostDetails: engine.NewEventCostFromCallCost(v2Cost.CostDetails, v2Cost.CGRID, v2Cost.RunID),
|
|
}
|
|
return
|
|
}
|
|
|
|
func NewV2SessionsCostFromSessionsCostSql(smSql *engine.SessionCostsSQL) (smV2 *v2SessionsCost, err error) {
|
|
smV2 = new(v2SessionsCost)
|
|
smV2.CGRID = smSql.Cgrid
|
|
smV2.RunID = smSql.RunID
|
|
smV2.OriginHost = smSql.OriginHost
|
|
smV2.OriginID = smSql.OriginID
|
|
smV2.CostSource = smSql.CostSource
|
|
smV2.Usage = time.Duration(smSql.Usage)
|
|
smV2.CostDetails = new(engine.CallCost)
|
|
if err := json.Unmarshal([]byte(smSql.CostDetails), smV2.CostDetails); err != nil {
|
|
return nil, err
|
|
}
|
|
return
|
|
}
|
|
|
|
func (v2Cost *v2SessionsCost) AsSessionsCostSql() (smSql *engine.SessionCostsSQL) {
|
|
smSql = new(engine.SessionCostsSQL)
|
|
smSql.Cgrid = v2Cost.CGRID
|
|
smSql.RunID = v2Cost.RunID
|
|
smSql.OriginHost = v2Cost.OriginHost
|
|
smSql.OriginID = v2Cost.OriginID
|
|
smSql.CostSource = v2Cost.CostSource
|
|
smSql.CostDetails = utils.ToJSON(v2Cost.CostDetails)
|
|
smSql.Usage = v2Cost.Usage.Nanoseconds()
|
|
smSql.CreatedAt = time.Now()
|
|
return
|
|
}
|