mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-11 18:16:24 +05:00
216 lines
5.9 KiB
Go
216 lines
5.9 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 (
|
|
|
|
//"log"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/cgrates/cgrates/config"
|
|
"github.com/cgrates/cgrates/engine"
|
|
"github.com/cgrates/cgrates/utils"
|
|
)
|
|
|
|
type v1Alias struct {
|
|
Direction string
|
|
Tenant string
|
|
Category string
|
|
Account string
|
|
Subject string
|
|
Context string
|
|
Values v1AliasValues
|
|
}
|
|
|
|
var (
|
|
AliasesPrefix = "als_"
|
|
Alias = "Alias"
|
|
)
|
|
|
|
type v1AliasValues []*v1AliasValue
|
|
|
|
type v1AliasValue struct {
|
|
DestinationId string //=Destination
|
|
Pairs v1AliasPairs
|
|
Weight float64
|
|
}
|
|
|
|
type v1AliasPairs map[string]map[string]string
|
|
|
|
func (al *v1Alias) SetId(id string) error {
|
|
vals := strings.Split(id, utils.ConcatenatedKeySep)
|
|
if len(vals) != 6 {
|
|
return utils.ErrInvalidKey
|
|
}
|
|
al.Direction = vals[0]
|
|
al.Tenant = vals[1]
|
|
al.Category = vals[2]
|
|
al.Account = vals[3]
|
|
al.Subject = vals[4]
|
|
al.Context = vals[5]
|
|
return nil
|
|
}
|
|
|
|
func (al *v1Alias) GetId() string {
|
|
return utils.ConcatenatedKey(al.Direction, al.Tenant, al.Category, al.Account, al.Subject, al.Context)
|
|
}
|
|
|
|
func alias2AtttributeProfile(alias *v1Alias, defaultTenant string) *engine.AttributeProfile {
|
|
out := &engine.AttributeProfile{
|
|
Tenant: alias.Tenant,
|
|
ID: alias.GetId(),
|
|
Contexts: []string{utils.MetaAny},
|
|
FilterIDs: make([]string, 0),
|
|
ActivationInterval: nil,
|
|
Attributes: make([]*engine.Attribute, 0),
|
|
Blocker: false,
|
|
Weight: 20, // should have prio against attributes out of users
|
|
}
|
|
if len(out.Tenant) == 0 || out.Tenant == utils.MetaAny {
|
|
out.Tenant = defaultTenant
|
|
}
|
|
if len(alias.Category) != 0 && alias.Category != utils.MetaAny {
|
|
out.FilterIDs = append(out.FilterIDs,
|
|
fmt.Sprintf("%s:~%s:%s", utils.MetaString, utils.MetaReq+utils.NestingSep+utils.Category, alias.Category))
|
|
}
|
|
if len(alias.Account) != 0 && alias.Account != utils.MetaAny {
|
|
out.FilterIDs = append(out.FilterIDs,
|
|
fmt.Sprintf("%s:~%s:%s", utils.MetaString, utils.MetaReq+utils.NestingSep+utils.AccountField, alias.Account))
|
|
}
|
|
if len(alias.Subject) != 0 && alias.Subject != utils.MetaAny {
|
|
out.FilterIDs = append(out.FilterIDs,
|
|
fmt.Sprintf("%s:~%s:%s", utils.MetaString, utils.MetaReq+utils.NestingSep+utils.Subject, alias.Subject))
|
|
}
|
|
var destination string
|
|
for _, av := range alias.Values {
|
|
if len(destination) == 0 || destination == utils.MetaAny {
|
|
destination = av.DestinationId
|
|
}
|
|
for fieldName, vals := range av.Pairs {
|
|
for initial, substitute := range vals {
|
|
var fld string
|
|
if fieldName == utils.Tenant {
|
|
fieldName = utils.MetaTenant
|
|
fld = utils.MetaTenant
|
|
} else if fieldName != utils.EmptyString {
|
|
fld = utils.MetaReq + utils.NestingSep + fieldName
|
|
} else {
|
|
continue // ignore empty fieldNames
|
|
}
|
|
attr := &engine.Attribute{
|
|
Path: fld,
|
|
Type: utils.MetaVariable, //default type for Attribute
|
|
Value: config.NewRSRParsersMustCompile(substitute, utils.InfieldSep),
|
|
}
|
|
out.Attributes = append(out.Attributes, attr)
|
|
// Add attribute filters if needed
|
|
if initial == "" || initial == utils.MetaAny {
|
|
continue
|
|
}
|
|
if fieldName == utils.MetaTenant { // no filter for tenant
|
|
continue
|
|
}
|
|
if fieldName == utils.Category && alias.Category == initial {
|
|
continue
|
|
}
|
|
if fieldName == utils.AccountField && alias.Account == initial {
|
|
continue
|
|
}
|
|
if fieldName == utils.Subject && alias.Subject == initial {
|
|
continue
|
|
}
|
|
attr.FilterIDs = append(attr.FilterIDs,
|
|
fmt.Sprintf("%s:~%s:%s", utils.MetaString, utils.MetaReq+utils.NestingSep+fieldName, initial))
|
|
}
|
|
}
|
|
}
|
|
if len(destination) != 0 && destination != utils.MetaAny {
|
|
out.FilterIDs = append(out.FilterIDs,
|
|
fmt.Sprintf("%s:~%s:%s", utils.MetaDestinations, utils.MetaReq+utils.NestingSep+utils.Destination, destination))
|
|
}
|
|
return out
|
|
}
|
|
|
|
func (m *Migrator) removeAlias2Attributes() (err error) {
|
|
for {
|
|
alias, err := m.dmIN.getV1Alias()
|
|
if err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := m.dmIN.remV1Alias(alias.GetId()); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateAlias2Attributes() (err error) {
|
|
cfg := config.CgrConfig()
|
|
|
|
defaultTenant := cfg.GeneralCfg().DefaultTenant
|
|
for {
|
|
alias, err := m.dmIN.getV1Alias()
|
|
if err == utils.ErrNoMoreData {
|
|
break
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if alias == nil || m.dryRun {
|
|
continue
|
|
}
|
|
attr := alias2AtttributeProfile(alias, defaultTenant)
|
|
if len(attr.Attributes) == 0 {
|
|
continue
|
|
}
|
|
if err := m.dmOut.DataManager().SetAttributeProfile(attr, true); err != nil {
|
|
return err
|
|
}
|
|
m.stats[Alias]++
|
|
}
|
|
if m.dryRun {
|
|
return
|
|
}
|
|
if !m.sameDataDB {
|
|
if err = m.removeAlias2Attributes(); err != nil {
|
|
return
|
|
}
|
|
}
|
|
// All done, update version wtih current one
|
|
vrs := engine.Versions{Alias: 0} //engine.CurrentDataDBVersions()[utils.Alias]}
|
|
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 Alias version into dataDB", err.Error()))
|
|
}
|
|
return
|
|
}
|
|
|
|
func (m *Migrator) migrateAlias() (err error) {
|
|
if err = m.migrateAlias2Attributes(); err != nil {
|
|
return
|
|
}
|
|
return m.ensureIndexesDataDB(engine.ColAttr)
|
|
}
|