Files
cgrates/engine/storage_internal_datadb.go
2025-06-03 14:23:29 +02:00

793 lines
26 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 engine
import (
"errors"
"fmt"
"strings"
"sync"
"time"
"github.com/cgrates/birpc/context"
"github.com/cgrates/cgrates/config"
"github.com/cgrates/ltcache"
"github.com/cgrates/cgrates/utils"
)
// InternalDB is used as a DataDB and/or StorDB
type InternalDB struct {
stringIndexedFields []string
prefixIndexedFields []string
indexedFieldsMutex sync.RWMutex // used for reload
cnter *utils.Counter // used for OrderID for cdr
ms utils.Marshaler
db *ltcache.TransCache
}
// NewInternalDB constructs an InternalDB
func NewInternalDB(stringIndexedFields, prefixIndexedFields []string,
transCacheOpts *ltcache.TransCacheOpts, itmsCfg map[string]*config.ItemOpts) (iDB *InternalDB,
err error) {
tcCfg := make(map[string]*ltcache.CacheConfig, len(itmsCfg))
for k, cPcfg := range itmsCfg {
tcCfg[k] = &ltcache.CacheConfig{
MaxItems: cPcfg.Limit,
TTL: cPcfg.TTL,
StaticTTL: cPcfg.StaticTTL,
Clone: true,
}
}
if transCacheOpts != nil && transCacheOpts.DumpInterval == 0 && transCacheOpts.RewriteInterval == 0 {
transCacheOpts = nil // create TransCache without offline collector if neither
// DumpInterval or RewriteInterval are provided
}
tc, err := ltcache.NewTransCacheWithOfflineCollector(transCacheOpts, tcCfg, utils.Logger)
if err != nil {
return nil, err
}
ms, _ := utils.NewMarshaler(config.CgrConfig().GeneralCfg().DBDataEncoding)
return &InternalDB{
stringIndexedFields: stringIndexedFields,
prefixIndexedFields: prefixIndexedFields,
cnter: utils.NewCounter(time.Now().UnixNano(), 0),
ms: ms,
db: tc,
}, nil
}
// SetStringIndexedFields set the stringIndexedFields, used at StorDB reload (is thread safe)
func (iDB *InternalDB) SetStringIndexedFields(stringIndexedFields []string) {
iDB.indexedFieldsMutex.Lock()
iDB.stringIndexedFields = stringIndexedFields
iDB.indexedFieldsMutex.Unlock()
}
// SetPrefixIndexedFields set the prefixIndexedFields, used at StorDB reload (is thread safe)
func (iDB *InternalDB) SetPrefixIndexedFields(prefixIndexedFields []string) {
iDB.indexedFieldsMutex.Lock()
iDB.prefixIndexedFields = prefixIndexedFields
iDB.indexedFieldsMutex.Unlock()
}
// Close depending on dump and rewrite intervals, will dump all thats left in
// cache collector to file and/or rewrite files, and close all files
func (iDB *InternalDB) Close() {
iDB.db.Shutdown()
}
// Flush clears the cache
func (iDB *InternalDB) Flush(string) error {
iDB.db.Clear(nil)
return nil
}
// SelectDatabase only to implement Storage interface
func (iDB *InternalDB) SelectDatabase(string) (err error) {
return nil
}
// GetKeysForPrefix returns the keys from cache that have the given prefix
func (iDB *InternalDB) GetKeysForPrefix(_ *context.Context, prefix string) (ids []string, err error) {
keyLen := len(utils.AccountPrefix)
if len(prefix) < keyLen {
err = fmt.Errorf("unsupported prefix in GetKeysForPrefix: %s", prefix)
return
}
category := prefix[:keyLen] // prefix length
queryPrefix := prefix[keyLen:]
ids = iDB.db.GetItemIDs(utils.CachePrefixToInstance[category], queryPrefix)
for i := range ids {
ids[i] = category + ids[i]
}
return
}
func (iDB *InternalDB) GetVersions(itm string) (vrs Versions, err error) {
x, ok := iDB.db.Get(utils.CacheVersions, utils.VersionName)
if !ok || x == nil {
return nil, utils.ErrNotFound
}
provVrs := x.(Versions)
if itm != "" {
if _, has := provVrs[itm]; !has {
return nil, utils.ErrNotFound
}
return Versions{itm: provVrs[itm]}, nil
}
return provVrs, nil
}
func (iDB *InternalDB) SetVersions(vrs Versions, overwrite bool) (err error) {
if overwrite {
if err = iDB.RemoveVersions(nil); err != nil {
return
}
}
x, ok := iDB.db.Get(utils.CacheVersions, utils.VersionName)
if !ok || x == nil {
iDB.db.Set(utils.CacheVersions, utils.VersionName, vrs, nil,
true, utils.NonTransactional)
return
}
provVrs := x.(Versions)
for key, val := range vrs {
provVrs[key] = val
}
iDB.db.Set(utils.CacheVersions, utils.VersionName, provVrs, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveVersions(vrs Versions) (err error) {
if len(vrs) != 0 {
var internalVersions Versions
x, ok := iDB.db.Get(utils.CacheVersions, utils.VersionName)
if !ok || x == nil {
return utils.ErrNotFound
}
internalVersions = x.(Versions)
for key := range vrs {
delete(internalVersions, key)
}
iDB.db.Set(utils.CacheVersions, utils.VersionName, internalVersions, nil,
true, utils.NonTransactional)
return
}
iDB.db.Remove(utils.CacheVersions, utils.VersionName,
true, utils.NonTransactional)
return
}
// GetStorageType returns the storage type
func (iDB *InternalDB) GetStorageType() string {
return utils.MetaInternal
}
// IsDBEmpty returns true if the cache is empty
func (iDB *InternalDB) IsDBEmpty() (isEmpty bool, err error) {
for cacheInstance := range utils.DataDBPartitions {
if len(iDB.db.GetItemIDs(cacheInstance, utils.EmptyString)) != 0 {
return
}
}
isEmpty = true
return
}
func (iDB *InternalDB) HasDataDrv(_ *context.Context, category, subject, tenant string) (bool, error) {
switch category {
case utils.ResourcesPrefix, utils.ResourceProfilesPrefix, utils.StatQueuePrefix,
utils.StatQueueProfilePrefix, utils.ThresholdPrefix, utils.ThresholdProfilePrefix,
utils.FilterPrefix, utils.RouteProfilePrefix, utils.AttributeProfilePrefix,
utils.ChargerProfilePrefix:
return iDB.db.HasItem(utils.CachePrefixToInstance[category], utils.ConcatenatedKey(tenant, subject)), nil
}
return false, errors.New("Unsupported HasData category")
}
func (iDB *InternalDB) GetResourceProfileDrv(_ *context.Context, tenant, id string) (rp *utils.ResourceProfile, err error) {
if x, ok := iDB.db.Get(utils.CacheResourceProfiles, utils.ConcatenatedKey(tenant, id)); ok && x != nil {
return x.(*utils.ResourceProfile), nil
}
return nil, utils.ErrNotFound
}
func (iDB *InternalDB) SetResourceProfileDrv(_ *context.Context, rp *utils.ResourceProfile) (err error) {
iDB.db.Set(utils.CacheResourceProfiles, rp.TenantID(), rp, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveResourceProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheResourceProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetResourceDrv(_ *context.Context, tenant, id string) (r *utils.Resource, err error) {
if x, ok := iDB.db.Get(utils.CacheResources, utils.ConcatenatedKey(tenant, id)); ok && x != nil {
return x.(*utils.Resource), nil
}
return nil, utils.ErrNotFound
}
func (iDB *InternalDB) SetResourceDrv(_ *context.Context, r *utils.Resource) (err error) {
iDB.db.Set(utils.CacheResources, r.TenantID(), r, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveResourceDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheResources, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetIPProfileDrv(_ *context.Context, tenant, id string) (*utils.IPProfile, error) {
if x, ok := iDB.db.Get(utils.CacheIPProfiles, utils.ConcatenatedKey(tenant, id)); ok && x != nil {
return x.(*utils.IPProfile), nil
}
return nil, utils.ErrNotFound
}
func (iDB *InternalDB) SetIPProfileDrv(_ *context.Context, ipp *utils.IPProfile) error {
iDB.db.Set(utils.CacheIPProfiles, ipp.TenantID(), ipp, nil,
true, utils.NonTransactional)
return nil
}
func (iDB *InternalDB) RemoveIPProfileDrv(_ *context.Context, tenant, id string) error {
iDB.db.Remove(utils.CacheIPProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return nil
}
func (iDB *InternalDB) GetIPAllocationsDrv(_ *context.Context, tenant, id string) (*utils.IPAllocations, error) {
if x, ok := iDB.db.Get(utils.CacheIPAllocations, utils.ConcatenatedKey(tenant, id)); ok && x != nil {
return x.(*utils.IPAllocations), nil
}
return nil, utils.ErrNotFound
}
func (iDB *InternalDB) SetIPAllocationsDrv(_ *context.Context, ip *utils.IPAllocations) error {
iDB.db.Set(utils.CacheIPAllocations, ip.TenantID(), ip, nil,
true, utils.NonTransactional)
return nil
}
func (iDB *InternalDB) RemoveIPAllocationsDrv(_ *context.Context, tenant, id string) error {
iDB.db.Remove(utils.CacheIPAllocations, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return nil
}
func (iDB *InternalDB) GetLoadHistory(int, bool, string) ([]*utils.LoadInstance, error) {
return nil, nil
}
func (iDB *InternalDB) AddLoadHistory(*utils.LoadInstance, int, string) error {
return nil
}
func (iDB *InternalDB) GetStatQueueProfileDrv(_ *context.Context, tenant string, id string) (sq *StatQueueProfile, err error) {
x, ok := iDB.db.Get(utils.CacheStatQueueProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*StatQueueProfile), nil
}
func (iDB *InternalDB) SetStatQueueProfileDrv(_ *context.Context, sq *StatQueueProfile) (err error) {
iDB.db.Set(utils.CacheStatQueueProfiles, sq.TenantID(), sq, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemStatQueueProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheStatQueueProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetRankingProfileDrv(_ *context.Context, tenant, id string) (sg *utils.RankingProfile, err error) {
x, ok := iDB.db.Get(utils.CacheRankingProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.RankingProfile), nil
}
func (iDB *InternalDB) SetRankingProfileDrv(_ *context.Context, sgp *utils.RankingProfile) (err error) {
iDB.db.Set(utils.CacheRankingProfiles, sgp.TenantID(), sgp, nil, true, utils.NonTransactional)
return nil
}
func (iDB *InternalDB) RemRankingProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheRankingProfiles, utils.ConcatenatedKey(tenant, id), true, utils.NonTransactional)
return nil
}
func (iDB *InternalDB) GetRankingDrv(_ *context.Context, tenant, id string) (rn *utils.Ranking, err error) {
x, ok := iDB.db.Get(utils.CacheRankings, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.Ranking), nil
}
func (iDB *InternalDB) SetRankingDrv(_ *context.Context, rn *utils.Ranking) (err error) {
iDB.db.Set(utils.CacheRankings, rn.TenantID(), rn, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveRankingDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheRankings, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetStatQueueDrv(_ *context.Context, tenant, id string) (sq *StatQueue, err error) {
x, ok := iDB.db.Get(utils.CacheStatQueues, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*StatQueue), nil
}
func (iDB *InternalDB) SetStatQueueDrv(_ *context.Context, ssq *StoredStatQueue, sq *StatQueue) (err error) {
if sq == nil {
sq, err = ssq.AsStatQueue(iDB.ms)
if err != nil {
return
}
}
iDB.db.Set(utils.CacheStatQueues, utils.ConcatenatedKey(sq.Tenant, sq.ID), sq, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemStatQueueDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheStatQueues, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetThresholdProfileDrv(_ *context.Context, tenant, id string) (tp *ThresholdProfile, err error) {
x, ok := iDB.db.Get(utils.CacheThresholdProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*ThresholdProfile), nil
}
func (iDB *InternalDB) SetTrendProfileDrv(_ *context.Context, srp *utils.TrendProfile) (err error) {
iDB.db.Set(utils.CacheTrendProfiles, srp.TenantID(), srp, nil, true, utils.NonTransactional)
return nil
}
func (iDB *InternalDB) RemTrendProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheTrendProfiles, utils.ConcatenatedKey(tenant, id), true, utils.NonTransactional)
return nil
}
func (iDB *InternalDB) GetTrendProfileDrv(_ *context.Context, tenant, id string) (sg *utils.TrendProfile, err error) {
x, ok := iDB.db.Get(utils.CacheTrendProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.TrendProfile), nil
}
func (iDB *InternalDB) GetTrendDrv(_ *context.Context, tenant, id string) (th *utils.Trend, err error) {
x, ok := iDB.db.Get(utils.CacheTrends, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.Trend), nil
}
func (iDB *InternalDB) SetTrendDrv(_ *context.Context, tr *utils.Trend) (err error) {
iDB.db.Set(utils.CacheTrends, tr.TenantID(), tr, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveTrendDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheTrends, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) SetThresholdProfileDrv(_ *context.Context, tp *ThresholdProfile) (err error) {
iDB.db.Set(utils.CacheThresholdProfiles, tp.TenantID(), tp, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemThresholdProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheThresholdProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetThresholdDrv(_ *context.Context, tenant, id string) (th *Threshold, err error) {
x, ok := iDB.db.Get(utils.CacheThresholds, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*Threshold), nil
}
func (iDB *InternalDB) SetThresholdDrv(_ *context.Context, th *Threshold) (err error) {
iDB.db.Set(utils.CacheThresholds, th.TenantID(), th, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveThresholdDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheThresholds, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetFilterDrv(_ *context.Context, tenant, id string) (fltr *Filter, err error) {
x, ok := iDB.db.Get(utils.CacheFilters, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*Filter), nil
}
func (iDB *InternalDB) SetFilterDrv(_ *context.Context, fltr *Filter) (err error) {
if err = fltr.Compile(); err != nil {
return
}
iDB.db.Set(utils.CacheFilters, fltr.TenantID(), fltr, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveFilterDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheFilters, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetRouteProfileDrv(_ *context.Context, tenant, id string) (spp *utils.RouteProfile, err error) {
x, ok := iDB.db.Get(utils.CacheRouteProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.RouteProfile), nil
}
func (iDB *InternalDB) SetRouteProfileDrv(_ *context.Context, spp *utils.RouteProfile) (err error) {
if err = spp.Compile(); err != nil {
return
}
iDB.db.Set(utils.CacheRouteProfiles, spp.TenantID(), spp, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveRouteProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheRouteProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetAttributeProfileDrv(_ *context.Context, tenant, id string) (attr *utils.AttributeProfile, err error) {
x, ok := iDB.db.Get(utils.CacheAttributeProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.AttributeProfile), nil
}
func (iDB *InternalDB) SetAttributeProfileDrv(_ *context.Context, attr *utils.AttributeProfile) (err error) {
if err = attr.Compile(); err != nil {
return
}
iDB.db.Set(utils.CacheAttributeProfiles, attr.TenantID(), attr, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveAttributeProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheAttributeProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetChargerProfileDrv(_ *context.Context, tenant, id string) (ch *utils.ChargerProfile, err error) {
x, ok := iDB.db.Get(utils.CacheChargerProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.ChargerProfile), nil
}
func (iDB *InternalDB) SetChargerProfileDrv(_ *context.Context, chr *utils.ChargerProfile) (err error) {
iDB.db.Set(utils.CacheChargerProfiles, chr.TenantID(), chr, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveChargerProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheChargerProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetItemLoadIDsDrv(_ *context.Context, itemIDPrefix string) (loadIDs map[string]int64, err error) {
x, ok := iDB.db.Get(utils.CacheLoadIDs, utils.LoadIDs)
if !ok || x == nil {
return nil, utils.ErrNotFound
}
loadIDs = x.(map[string]int64)
if itemIDPrefix != utils.EmptyString {
return map[string]int64{itemIDPrefix: loadIDs[itemIDPrefix]}, nil
}
return
}
func (iDB *InternalDB) SetLoadIDsDrv(_ *context.Context, loadIDs map[string]int64) (err error) {
iDB.db.Set(utils.CacheLoadIDs, utils.LoadIDs, loadIDs, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetRateProfileDrv(_ *context.Context, tenant, id string) (rpp *utils.RateProfile, err error) {
x, ok := iDB.db.Get(utils.CacheRateProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.RateProfile), nil
}
func (iDB *InternalDB) GetRateProfileRatesDrv(ctx *context.Context, tenant, profileID, rtPrfx string, needIDs bool) (rateIDs []string, rates []*utils.Rate, err error) {
x, ok := iDB.db.Get(utils.CacheRateProfiles, utils.ConcatenatedKey(tenant, profileID))
if !ok || x == nil {
return nil, nil, utils.ErrNotFound
}
for key, rt := range x.(*utils.RateProfile).Rates {
if key[:len(rtPrfx)] == rtPrfx {
if needIDs {
rateIDs = append(rateIDs, key)
continue
}
rates = append(rates, rt)
}
}
return
}
func (iDB *InternalDB) SetRateProfileDrv(_ *context.Context, rpp *utils.RateProfile, optOverwrite bool) (err error) {
if err = rpp.Compile(); err != nil {
return
}
if !optOverwrite {
// in case of add new rates into our profile
x, ok := iDB.db.Get(utils.CacheRateProfiles, utils.ConcatenatedKey(rpp.Tenant, rpp.ID))
if ok || x != nil {
// mix the old and new rates, in order to add new rates into our profile
oldRp := x.(*utils.RateProfile)
for key, rate := range oldRp.Rates {
if _, has := rpp.Rates[key]; !has {
rpp.Rates[key] = rate
}
}
}
}
iDB.db.Set(utils.CacheRateProfiles, rpp.TenantID(), rpp, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveRateProfileDrv(_ *context.Context, tenant, id string, rateIDs *[]string) (err error) {
// if we want to remove just some rates from our profile, we will remove by their key Rates:rateID, but firstly we have to get the obejct from cache
if rateIDs != nil {
x, ok := iDB.db.Get(utils.CacheRateProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return utils.ErrNotFound
}
rpfl := *x.(*utils.RateProfile)
for _, rateID := range *rateIDs {
delete(rpfl.Rates, rateID)
}
iDB.db.Set(utils.CacheRateProfiles, rpfl.TenantID(), &rpfl, nil,
true, utils.NonTransactional)
return
}
iDB.db.Remove(utils.CacheRateProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetActionProfileDrv(_ *context.Context, tenant, id string) (ap *utils.ActionProfile, err error) {
x, ok := iDB.db.Get(utils.CacheActionProfiles, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.ActionProfile), nil
}
func (iDB *InternalDB) SetActionProfileDrv(_ *context.Context, ap *utils.ActionProfile) (err error) {
iDB.db.Set(utils.CacheActionProfiles, ap.TenantID(), ap, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveActionProfileDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheActionProfiles, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveLoadIDsDrv() (err error) {
return utils.ErrNotImplemented
}
func (iDB *InternalDB) GetIndexesDrv(_ *context.Context, idxItmType, tntCtx, idxKey, transactionID string) (indexes map[string]utils.StringSet, err error) {
if idxKey == utils.EmptyString { // return all
indexes = make(map[string]utils.StringSet)
for _, dbKey := range iDB.db.GetGroupItemIDs(idxItmType, tntCtx) {
x, ok := iDB.db.Get(idxItmType, dbKey)
if !ok || x == nil {
continue
}
dbKey = strings.TrimPrefix(dbKey, tntCtx+utils.ConcatenatedKeySep)
indexes[dbKey] = x.(utils.StringSet).Clone()
}
if len(indexes) == 0 {
return nil, utils.ErrNotFound
}
return
}
dbKey := utils.ConcatenatedKey(tntCtx, idxKey)
x, ok := iDB.db.Get(idxItmType, dbKey)
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return map[string]utils.StringSet{
idxKey: x.(utils.StringSet).Clone(),
}, nil
}
func (iDB *InternalDB) SetIndexesDrv(_ *context.Context, idxItmType, tntCtx string,
indexes map[string]utils.StringSet, commit bool, transactionID string) (err error) {
if commit && transactionID != utils.EmptyString {
for _, dbKey := range iDB.db.GetGroupItemIDs(idxItmType, tntCtx) {
if !strings.HasPrefix(dbKey, "tmp_") || !strings.HasSuffix(dbKey, transactionID) {
continue
}
x, ok := iDB.db.Get(idxItmType, dbKey)
if !ok || x == nil {
continue
}
iDB.db.Remove(idxItmType, dbKey,
true, utils.NonTransactional)
key := strings.TrimSuffix(strings.TrimPrefix(dbKey, "tmp_"), utils.ConcatenatedKeySep+transactionID)
iDB.db.Set(idxItmType, key, x, []string{tntCtx},
true, utils.NonTransactional)
}
return
}
for idxKey, indx := range indexes {
dbKey := utils.ConcatenatedKey(tntCtx, idxKey)
if transactionID != utils.EmptyString {
dbKey = "tmp_" + utils.ConcatenatedKey(dbKey, transactionID)
}
if len(indx) == 0 {
iDB.db.Remove(idxItmType, dbKey,
true, utils.NonTransactional)
continue
}
//to be the same as HMSET
if transactionID != utils.EmptyString {
if x, ok := iDB.db.Get(idxItmType, dbKey); ok && x != nil {
indx = utils.JoinStringSet(indx, x.(utils.StringSet))
}
}
iDB.db.Set(idxItmType, dbKey, indx, []string{tntCtx},
true, utils.NonTransactional)
}
return
}
func (iDB *InternalDB) RemoveIndexesDrv(_ *context.Context, idxItmType, tntCtx, idxKey string) (err error) {
if idxKey == utils.EmptyString {
iDB.db.RemoveGroup(idxItmType, tntCtx, true, utils.EmptyString)
return
}
iDB.db.Remove(idxItmType, utils.ConcatenatedKey(tntCtx, idxKey), true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetAccountDrv(_ *context.Context, tenant, id string) (ap *utils.Account, err error) {
x, ok := iDB.db.Get(utils.CacheAccounts, utils.ConcatenatedKey(tenant, id))
if !ok || x == nil {
return nil, utils.ErrNotFound
}
return x.(*utils.Account).Clone(), nil
}
func (iDB *InternalDB) SetAccountDrv(_ *context.Context, ap *utils.Account) (err error) {
iDB.db.Set(utils.CacheAccounts, ap.TenantID(), ap, nil,
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) RemoveAccountDrv(_ *context.Context, tenant, id string) (err error) {
iDB.db.Remove(utils.CacheAccounts, utils.ConcatenatedKey(tenant, id),
true, utils.NonTransactional)
return
}
func (iDB *InternalDB) GetConfigSectionsDrv(ctx *context.Context, nodeID string, sectionIDs []string) (sectionMap map[string][]byte, err error) {
sectionMap = make(map[string][]byte)
for _, sectionID := range sectionIDs {
x, ok := iDB.db.Get(utils.CacheConfig, utils.ConcatenatedKey(nodeID, sectionID))
if !ok || x == nil {
utils.Logger.Warning(fmt.Sprintf("CGRateS<%+v> Could not find any data for section <%+v>",
nodeID, sectionID))
continue
}
sectionMap[sectionID] = x.([]byte)
}
if len(sectionMap) == 0 {
err = utils.ErrNotFound
return
}
return
}
func (iDB *InternalDB) SetConfigSectionsDrv(ctx *context.Context, nodeID string, sectionsData map[string][]byte) (err error) {
for sectionID, sectionData := range sectionsData {
iDB.db.Set(utils.CacheConfig, utils.ConcatenatedKey(nodeID, sectionID),
sectionData, nil, true, utils.NonTransactional)
}
return
}
func (iDB *InternalDB) RemoveConfigSectionsDrv(ctx *context.Context, nodeID string, sectionIDs []string) (err error) {
for _, sectionID := range sectionIDs {
iDB.db.Remove(utils.CacheConfig, utils.ConcatenatedKey(nodeID, sectionID), true, utils.NonTransactional)
}
return
}
// Will dump everything inside datadb to files
func (iDB *InternalDB) DumpDataDB() (err error) {
return iDB.db.DumpAll()
}
// Will rewrite every dump file of DataDB
func (iDB *InternalDB) RewriteDataDB() (err error) {
return iDB.db.RewriteAll()
}
// BackupDataDB will momentarely stop any dumping and rewriting until all dump folder is backed up in folder path backupFolderPath, making zip true will create a zip file in the path instead
func (iDB *InternalDB) BackupDataDB(backupFolderPath string, zip bool) (err error) {
return iDB.db.BackupDumpFolder(backupFolderPath, zip)
}