mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-11 18:16:24 +05:00
240 lines
6.3 KiB
Go
240 lines
6.3 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 (
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/cgrates/cgrates/engine"
|
|
"github.com/cgrates/cgrates/utils"
|
|
)
|
|
|
|
var (
|
|
SupplierProfilePrefix = "spp_"
|
|
ColSpp = "supplier_profiles"
|
|
)
|
|
|
|
type Supplier struct {
|
|
ID string // SupplierID
|
|
FilterIDs []string
|
|
AccountIDs []string
|
|
RatingPlanIDs []string // used when computing price
|
|
ResourceIDs []string // queried in some strategies
|
|
StatIDs []string // queried in some strategies
|
|
Weight float64
|
|
Blocker bool // do not process further supplier after this one
|
|
SupplierParameters string
|
|
}
|
|
|
|
type SupplierProfile struct {
|
|
Tenant string
|
|
ID string // LCR Profile ID
|
|
FilterIDs []string
|
|
ActivationInterval *utils.ActivationInterval // Activation interval
|
|
Sorting string // Sorting strategy
|
|
SortingParameters []string
|
|
Suppliers []*Supplier
|
|
Weight float64
|
|
}
|
|
|
|
func (m *Migrator) removeSupplier() (err error) {
|
|
for {
|
|
var spp *SupplierProfile
|
|
spp, err = m.dmIN.getSupplier()
|
|
if err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
if err != nil {
|
|
return
|
|
}
|
|
if err = m.dmIN.remSupplier(spp.Tenant, spp.ID); err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
|
|
}
|
|
|
|
func (m *Migrator) migrateFromSupplierToRoute() (err error) {
|
|
for {
|
|
var spp *SupplierProfile
|
|
spp, err = m.dmIN.getSupplier()
|
|
if err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if spp == nil || m.dryRun {
|
|
continue
|
|
}
|
|
if err := m.dmOut.DataManager().SetRouteProfile(convertSupplierToRoute(spp), true); err != nil {
|
|
return err
|
|
}
|
|
m.stats[utils.Routes]++
|
|
}
|
|
if m.dryRun {
|
|
return
|
|
}
|
|
if err = m.removeSupplier(); err != nil && err != utils.ErrNoMoreData {
|
|
return
|
|
}
|
|
// All done, update version with current one
|
|
vrs := engine.Versions{utils.Routes: 1}
|
|
if err = m.dmOut.DataManager().DataDB().SetVersions(vrs, false); err != nil {
|
|
return utils.NewCGRError(utils.Migrator,
|
|
utils.ServerErrorCaps,
|
|
err.Error(),
|
|
fmt.Sprintf("error: <%s> when updating RouteProfiles version into dataDB", err.Error()))
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateCurrentRouteProfile() (err error) {
|
|
var ids []string
|
|
ids, err = m.dmIN.DataManager().DataDB().GetKeysForPrefix(utils.RouteProfilePrefix)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, id := range ids {
|
|
tntID := strings.SplitN(strings.TrimPrefix(id, utils.RouteProfilePrefix), utils.InInFieldSep, 2)
|
|
if len(tntID) < 2 {
|
|
return fmt.Errorf("invalid key <%s> when migrating route profiles", id)
|
|
}
|
|
rPrf, err := m.dmIN.DataManager().GetRouteProfile(tntID[0], tntID[1], false, false, utils.NonTransactional)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if rPrf == nil || m.dryRun {
|
|
continue
|
|
}
|
|
if err := m.dmOut.DataManager().SetRouteProfile(rPrf, true); err != nil {
|
|
return err
|
|
}
|
|
if err := m.dmIN.DataManager().RemoveRouteProfile(tntID[0], tntID[1], true); err != nil {
|
|
return err
|
|
}
|
|
m.stats[utils.Routes]++
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateRouteProfiles() (err error) {
|
|
var vrs engine.Versions
|
|
current := engine.CurrentDataDBVersions()
|
|
if vrs, err = m.getVersions(utils.ActionTriggers); err != nil {
|
|
return
|
|
}
|
|
routeVersion, has := vrs[utils.Routes]
|
|
if !has {
|
|
if vrs[utils.RQF] != current[utils.RQF] {
|
|
return fmt.Errorf("please migrate the filters before migrating the routes")
|
|
}
|
|
if err = m.migrateFromSupplierToRoute(); err != nil {
|
|
return
|
|
}
|
|
}
|
|
migrated := true
|
|
var v2 *engine.RouteProfile
|
|
for {
|
|
version := routeVersion
|
|
for {
|
|
switch version {
|
|
default:
|
|
return fmt.Errorf("Unsupported version %v", version)
|
|
case current[utils.Routes]:
|
|
migrated = false
|
|
if m.sameDataDB {
|
|
break
|
|
}
|
|
if err = m.migrateCurrentRouteProfile(); err != nil {
|
|
return err
|
|
}
|
|
case 1:
|
|
if v2, err = m.migrateV1ToV2Routes(); err != nil && err != utils.ErrNoMoreData {
|
|
return
|
|
} else if err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
version = 2
|
|
}
|
|
if version == current[utils.Routes] || err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
}
|
|
if err == utils.ErrNoMoreData || !migrated {
|
|
break
|
|
}
|
|
if !m.dryRun {
|
|
if err = m.dmIN.DataManager().SetRouteProfile(v2, true); err != nil {
|
|
return
|
|
}
|
|
}
|
|
m.stats[utils.Routes]++
|
|
}
|
|
// All done, update version wtih current one
|
|
if err = m.setVersions(utils.Routes); err != nil {
|
|
return
|
|
}
|
|
|
|
return m.ensureIndexesDataDB(engine.ColRts)
|
|
}
|
|
|
|
func convertSupplierToRoute(spp *SupplierProfile) (route *engine.RouteProfile) {
|
|
route = &engine.RouteProfile{
|
|
Tenant: spp.Tenant,
|
|
ID: spp.ID,
|
|
FilterIDs: spp.FilterIDs,
|
|
ActivationInterval: spp.ActivationInterval,
|
|
Sorting: spp.Sorting,
|
|
SortingParameters: spp.SortingParameters,
|
|
Weight: spp.Weight,
|
|
}
|
|
route.Routes = make([]*engine.Route, len(spp.Suppliers))
|
|
for i, supl := range spp.Suppliers {
|
|
route.Routes[i] = &engine.Route{
|
|
ID: supl.ID,
|
|
FilterIDs: supl.FilterIDs,
|
|
AccountIDs: supl.AccountIDs,
|
|
RatingPlanIDs: supl.RatingPlanIDs,
|
|
ResourceIDs: supl.ResourceIDs,
|
|
StatIDs: supl.StatIDs,
|
|
Weight: supl.Weight,
|
|
Blocker: supl.Blocker,
|
|
RouteParameters: supl.SupplierParameters,
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateV1ToV2Routes() (v4Cpp *engine.RouteProfile, err error) {
|
|
v4Cpp, err = m.dmIN.getV1RouteProfile()
|
|
if err != nil {
|
|
return nil, err
|
|
} else if v4Cpp == nil {
|
|
return nil, errors.New("Dispatcher NIL")
|
|
}
|
|
if v4Cpp.FilterIDs, err = migrateInlineFilterV4(v4Cpp.FilterIDs); err != nil {
|
|
return nil, err
|
|
}
|
|
return
|
|
}
|