mirror of
https://github.com/cgrates/cgrates.git
synced 2026-02-11 18:16:24 +05:00
183 lines
4.9 KiB
Go
183 lines
4.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 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 loaders
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"sync"
|
|
|
|
"github.com/cgrates/cgrates/config"
|
|
"github.com/cgrates/cgrates/engine"
|
|
"github.com/cgrates/cgrates/utils"
|
|
)
|
|
|
|
func NewLoaderService(dm *engine.DataManager, ldrsCfg []*config.LoaderSCfg,
|
|
timezone string, exitChan chan bool, filterS *engine.FilterS,
|
|
connMgr *engine.ConnManager) (ldrS *LoaderService) {
|
|
ldrS = &LoaderService{ldrs: make(map[string]*Loader)}
|
|
for _, ldrCfg := range ldrsCfg {
|
|
if !ldrCfg.Enabled {
|
|
continue
|
|
}
|
|
ldrS.ldrs[ldrCfg.Id] = NewLoader(dm, ldrCfg, timezone, exitChan, filterS, connMgr, ldrCfg.CacheSConns)
|
|
}
|
|
return
|
|
}
|
|
|
|
// LoaderS is the Loader service handling independent Loaders
|
|
type LoaderService struct {
|
|
sync.RWMutex
|
|
ldrs map[string]*Loader
|
|
}
|
|
|
|
// IsEnabled returns true if at least one loader is enabled
|
|
func (ldrS *LoaderService) Enabled() bool {
|
|
for _, ldr := range ldrS.ldrs {
|
|
if ldr.enabled {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (ldrS *LoaderService) ListenAndServe(exitChan chan bool) (err error) {
|
|
ldrExitChan := make(chan struct{})
|
|
for _, ldr := range ldrS.ldrs {
|
|
go ldr.ListenAndServe(ldrExitChan)
|
|
}
|
|
select { // exit on errors coming from server or any loader
|
|
case e := <-exitChan:
|
|
close(ldrExitChan)
|
|
exitChan <- e // put back for the others listening for shutdown request
|
|
case <-ldrExitChan:
|
|
exitChan <- true
|
|
}
|
|
return
|
|
}
|
|
|
|
type ArgsProcessFolder struct {
|
|
LoaderID string
|
|
ForceLock bool
|
|
Caching *string
|
|
}
|
|
|
|
func (ldrS *LoaderService) V1Load(args *ArgsProcessFolder,
|
|
rply *string) (err error) {
|
|
ldrS.RLock()
|
|
defer ldrS.RUnlock()
|
|
if args.LoaderID == "" {
|
|
args.LoaderID = utils.MetaDefault
|
|
}
|
|
ldr, has := ldrS.ldrs[args.LoaderID]
|
|
if !has {
|
|
return fmt.Errorf("UNKNOWN_LOADER: %s", args.LoaderID)
|
|
}
|
|
if locked, err := ldr.isFolderLocked(); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
} else if locked {
|
|
if !args.ForceLock {
|
|
return errors.New("ANOTHER_LOADER_RUNNING")
|
|
}
|
|
if err := ldr.unlockFolder(); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
}
|
|
}
|
|
//verify If Caching is present in arguments
|
|
caching := config.CgrConfig().GeneralCfg().DefaultCaching
|
|
if args.Caching != nil {
|
|
caching = *args.Caching
|
|
}
|
|
if err := ldr.ProcessFolder(caching, utils.MetaStore); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
}
|
|
*rply = utils.OK
|
|
return
|
|
}
|
|
|
|
func (ldrS *LoaderService) V1Remove(args *ArgsProcessFolder,
|
|
rply *string) (err error) {
|
|
ldrS.RLock()
|
|
defer ldrS.RUnlock()
|
|
if args.LoaderID == "" {
|
|
args.LoaderID = utils.MetaDefault
|
|
}
|
|
ldr, has := ldrS.ldrs[args.LoaderID]
|
|
if !has {
|
|
return fmt.Errorf("UNKNOWN_LOADER: %s", args.LoaderID)
|
|
}
|
|
if locked, err := ldr.isFolderLocked(); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
} else if locked {
|
|
if args.ForceLock {
|
|
if err := ldr.unlockFolder(); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
}
|
|
}
|
|
return errors.New("ANOTHER_LOADER_RUNNING")
|
|
}
|
|
//verify If Caching is present in arguments
|
|
caching := config.CgrConfig().GeneralCfg().DefaultCaching
|
|
if args.Caching != nil {
|
|
caching = *args.Caching
|
|
}
|
|
if err := ldr.ProcessFolder(caching, utils.MetaRemove); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
}
|
|
*rply = utils.OK
|
|
return
|
|
}
|
|
|
|
// Reload recreates the loaders map thread safe
|
|
func (ldrS *LoaderService) Reload(dm *engine.DataManager, ldrsCfg []*config.LoaderSCfg,
|
|
timezone string, exitChan chan bool, filterS *engine.FilterS, connMgr *engine.ConnManager) {
|
|
ldrS.Lock()
|
|
ldrS.ldrs = make(map[string]*Loader)
|
|
for _, ldrCfg := range ldrsCfg {
|
|
if !ldrCfg.Enabled {
|
|
continue
|
|
}
|
|
ldrS.ldrs[ldrCfg.Id] = NewLoader(dm, ldrCfg, timezone, exitChan, filterS, connMgr, ldrCfg.CacheSConns)
|
|
}
|
|
ldrS.Unlock()
|
|
}
|
|
|
|
func (ldrS *LoaderService) Preload(loaderIDs []string) (err error) {
|
|
ldrS.RLock()
|
|
defer ldrS.RUnlock()
|
|
for _, loaderID := range loaderIDs {
|
|
ldr, has := ldrS.ldrs[loaderID]
|
|
if !has {
|
|
return fmt.Errorf("UNKNOWN_LOADER: %s", loaderID)
|
|
}
|
|
if locked, err := ldr.isFolderLocked(); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
} else if locked {
|
|
if err := ldr.unlockFolder(); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
}
|
|
return errors.New("ANOTHER_LOADER_RUNNING")
|
|
}
|
|
if err := ldr.ProcessFolder(config.CgrConfig().GeneralCfg().DefaultCaching, utils.MetaStore); err != nil {
|
|
return utils.NewErrServerError(err)
|
|
}
|
|
}
|
|
return
|
|
}
|