/* 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 */ 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, utils.EmptyString) 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 }