mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-12 02:26:26 +05:00
233 lines
6.0 KiB
Go
233 lines
6.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 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 migrator
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/cgrates/birpc/context"
|
|
"github.com/cgrates/cgrates/engine"
|
|
"github.com/cgrates/cgrates/utils"
|
|
)
|
|
|
|
func (m *Migrator) migrateCurrentThresholds() (err error) {
|
|
var ids []string
|
|
//ThresholdProfiles
|
|
ids, err = m.dmIN.DataManager().DataDB().GetKeysForPrefix(context.TODO(), utils.ThresholdProfilePrefix)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, id := range ids {
|
|
tntID := strings.SplitN(strings.TrimPrefix(id, utils.ThresholdProfilePrefix), utils.InInFieldSep, 2)
|
|
if len(tntID) < 2 {
|
|
return fmt.Errorf("Invalid key <%s> when migrating threshold profiles", id)
|
|
}
|
|
thps, err := m.dmIN.DataManager().GetThresholdProfile(context.TODO(), tntID[0], tntID[1], false, false, utils.NonTransactional)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ths, err := m.dmIN.DataManager().GetThreshold(context.TODO(), tntID[0], tntID[1], false, false, utils.NonTransactional)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if thps == nil || m.dryRun {
|
|
continue
|
|
}
|
|
if err := m.dmOut.DataManager().SetThresholdProfile(context.TODO(), thps, true); err != nil {
|
|
return err
|
|
}
|
|
// update the threshold in the new DB
|
|
if ths != nil {
|
|
if err := m.dmOut.DataManager().SetThreshold(context.TODO(), ths); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if err := m.dmIN.DataManager().RemoveThresholdProfile(context.TODO(), tntID[0], tntID[1], false); err != nil {
|
|
return err
|
|
}
|
|
m.stats[utils.Thresholds]++
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) removeV2Thresholds() (err error) {
|
|
var v2T *v2Threshold
|
|
for {
|
|
v2T, err = m.dmIN.getV2ThresholdProfile()
|
|
if err != nil && err != utils.ErrNoMoreData {
|
|
return err
|
|
}
|
|
if err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
if err = m.dmIN.remV2ThresholdProfile(v2T.Tenant, v2T.ID); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateV2Thresholds() (v3 *engine.ThresholdProfile, err error) {
|
|
var v2T *v2Threshold
|
|
if v2T, err = m.dmIN.getV2ThresholdProfile(); err != nil {
|
|
return
|
|
}
|
|
if v2T == nil {
|
|
return
|
|
}
|
|
v3 = v2T.V2toV3Threshold()
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateThresholds() (err error) {
|
|
var vrs engine.Versions
|
|
current := engine.CurrentDataDBVersions()
|
|
if vrs, err = m.getVersions(utils.Thresholds); err != nil {
|
|
return
|
|
}
|
|
migrated := true
|
|
migratedFrom := 0
|
|
var th *engine.Threshold
|
|
var filter *engine.Filter
|
|
var v3 *engine.ThresholdProfile
|
|
var v4 *engine.ThresholdProfile
|
|
for {
|
|
version := vrs[utils.Thresholds]
|
|
migratedFrom = int(version)
|
|
for {
|
|
switch version {
|
|
default:
|
|
return fmt.Errorf("Unsupported version %v", version)
|
|
case current[utils.Thresholds]:
|
|
migrated = false
|
|
if m.sameDataDB {
|
|
break
|
|
}
|
|
if err = m.migrateCurrentThresholds(); err != nil {
|
|
return
|
|
}
|
|
case 1:
|
|
version = 3
|
|
case 2:
|
|
if v3, err = m.migrateV2Thresholds(); err != nil && err != utils.ErrNoMoreData {
|
|
return
|
|
}
|
|
version = 3
|
|
case 3:
|
|
if v4, err = m.migrateV3ToV4Threshold(v3); err != nil && err != utils.ErrNoMoreData {
|
|
return
|
|
} else if err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
version = 4
|
|
}
|
|
if version == current[utils.Thresholds] || err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
}
|
|
if err == utils.ErrNoMoreData || !migrated {
|
|
break
|
|
}
|
|
|
|
if !m.dryRun {
|
|
//set threshond
|
|
if migratedFrom == 1 {
|
|
if err = m.dmOut.DataManager().SetFilter(context.TODO(), filter, true); err != nil {
|
|
return
|
|
}
|
|
}
|
|
if err = m.dmOut.DataManager().SetThresholdProfile(context.TODO(), v4, true); err != nil {
|
|
return
|
|
}
|
|
if migratedFrom == 1 { // do it after SetThresholdProfile to overwrite the created threshold
|
|
if err = m.dmOut.DataManager().SetThreshold(context.TODO(), th); err != nil {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
m.stats[utils.Thresholds]++
|
|
}
|
|
if m.dryRun || !migrated {
|
|
return nil
|
|
}
|
|
// remove old threshonds
|
|
if !m.sameDataDB && migratedFrom == 2 {
|
|
if err = m.removeV2Thresholds(); err != nil && err != utils.ErrNoMoreData {
|
|
return
|
|
}
|
|
}
|
|
// All done, update version wtih current one
|
|
if err = m.setVersions(utils.Thresholds); err != nil {
|
|
return
|
|
}
|
|
return m.ensureIndexesDataDB(engine.ColTps)
|
|
}
|
|
|
|
type v2Threshold struct {
|
|
Tenant string
|
|
ID string
|
|
FilterIDs []string
|
|
ActivationInterval *utils.ActivationInterval // Time when this limit becomes active and expires
|
|
Recurrent bool
|
|
MinHits int
|
|
MinSleep time.Duration
|
|
Blocker bool // blocker flag to stop processing on filters matched
|
|
Weight float64 // Weight to sort the thresholds
|
|
ActionIDs []string
|
|
Async bool
|
|
}
|
|
|
|
func (v2T v2Threshold) V2toV3Threshold() (th *engine.ThresholdProfile) {
|
|
th = &engine.ThresholdProfile{
|
|
Tenant: v2T.Tenant,
|
|
ID: v2T.ID,
|
|
FilterIDs: v2T.FilterIDs,
|
|
MinHits: v2T.MinHits,
|
|
MinSleep: v2T.MinSleep,
|
|
Blocker: v2T.Blocker,
|
|
Weights: utils.DynamicWeights{
|
|
{
|
|
Weight: v2T.Weight,
|
|
},
|
|
},
|
|
ActionProfileIDs: v2T.ActionIDs,
|
|
Async: v2T.Async,
|
|
}
|
|
th.MaxHits = 1
|
|
if v2T.Recurrent {
|
|
th.MaxHits = -1
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateV3ToV4Threshold(v3sts *engine.ThresholdProfile) (v4Cpp *engine.ThresholdProfile, err error) {
|
|
if v3sts == nil {
|
|
// read data from DataDB
|
|
if v3sts, err = m.dmIN.getV3ThresholdProfile(); err != nil {
|
|
return
|
|
}
|
|
}
|
|
if v3sts.FilterIDs, err = migrateInlineFilterV4(v3sts.FilterIDs); err != nil {
|
|
return
|
|
}
|
|
return v3sts, nil
|
|
}
|