mirror of
https://github.com/d0zingcat/gocryptotrader.git
synced 2026-05-13 15:09:42 +00:00
* Updates starting and stopping routines to be a bit more parallel with less waiting required * Removes stop, removes debugging output * linting and test fixes * Add extra kill switch for exiting on exchange loading delay * Fixes fun math * breaks loop instead of switch. Moves param warns higher * Removes unceccary gos. passes in cfg to remove data race * Removes os signal processing. Fixes bad master merge
133 lines
3.1 KiB
Go
133 lines
3.1 KiB
Go
package engine
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/thrasher-corp/gocryptotrader/database"
|
|
dbpsql "github.com/thrasher-corp/gocryptotrader/database/drivers/postgres"
|
|
dbsqlite3 "github.com/thrasher-corp/gocryptotrader/database/drivers/sqlite3"
|
|
"github.com/thrasher-corp/gocryptotrader/engine/subsystem"
|
|
"github.com/thrasher-corp/gocryptotrader/log"
|
|
"github.com/thrasher-corp/sqlboiler/boil"
|
|
)
|
|
|
|
var (
|
|
dbConn *database.Instance
|
|
)
|
|
|
|
type databaseManager struct {
|
|
started int32
|
|
shutdown chan struct{}
|
|
}
|
|
|
|
func (a *databaseManager) Started() bool {
|
|
return atomic.LoadInt32(&a.started) == 1
|
|
}
|
|
|
|
func (a *databaseManager) Start(bot *Engine) (err error) {
|
|
if !atomic.CompareAndSwapInt32(&a.started, 0, 1) {
|
|
return fmt.Errorf("database manager %w", subsystem.ErrSubSystemAlreadyStarted)
|
|
}
|
|
|
|
defer func() {
|
|
if err != nil {
|
|
atomic.CompareAndSwapInt32(&a.started, 1, 0)
|
|
}
|
|
}()
|
|
|
|
log.Debugln(log.DatabaseMgr, "Database manager starting...")
|
|
|
|
a.shutdown = make(chan struct{})
|
|
|
|
if bot.Config.Database.Enabled {
|
|
if bot.Config.Database.Driver == database.DBPostgreSQL {
|
|
log.Debugf(log.DatabaseMgr,
|
|
"Attempting to establish database connection to host %s/%s utilising %s driver\n",
|
|
bot.Config.Database.Host,
|
|
bot.Config.Database.Database,
|
|
bot.Config.Database.Driver)
|
|
dbConn, err = dbpsql.Connect()
|
|
} else if bot.Config.Database.Driver == database.DBSQLite ||
|
|
bot.Config.Database.Driver == database.DBSQLite3 {
|
|
log.Debugf(log.DatabaseMgr,
|
|
"Attempting to establish database connection to %s utilising %s driver\n",
|
|
bot.Config.Database.Database,
|
|
bot.Config.Database.Driver)
|
|
dbConn, err = dbsqlite3.Connect()
|
|
}
|
|
if err != nil {
|
|
return fmt.Errorf("database failed to connect: %v Some features that utilise a database will be unavailable", err)
|
|
}
|
|
dbConn.Connected = true
|
|
|
|
DBLogger := database.Logger{}
|
|
if bot.Config.Database.Verbose {
|
|
boil.DebugMode = true
|
|
boil.DebugWriter = DBLogger
|
|
}
|
|
|
|
go a.run(bot)
|
|
return nil
|
|
}
|
|
|
|
return errors.New("database support disabled")
|
|
}
|
|
|
|
func (a *databaseManager) Stop() error {
|
|
if atomic.LoadInt32(&a.started) == 0 {
|
|
return fmt.Errorf("database manager %w", subsystem.ErrSubSystemNotStarted)
|
|
}
|
|
defer func() {
|
|
atomic.CompareAndSwapInt32(&a.started, 1, 0)
|
|
}()
|
|
|
|
err := dbConn.SQL.Close()
|
|
if err != nil {
|
|
log.Errorf(log.DatabaseMgr, "Failed to close database: %v", err)
|
|
}
|
|
|
|
close(a.shutdown)
|
|
return nil
|
|
}
|
|
|
|
func (a *databaseManager) run(bot *Engine) {
|
|
log.Debugln(log.DatabaseMgr, "Database manager started.")
|
|
bot.ServicesWG.Add(1)
|
|
t := time.NewTicker(time.Second * 2)
|
|
|
|
defer func() {
|
|
t.Stop()
|
|
bot.ServicesWG.Done()
|
|
log.Debugln(log.DatabaseMgr, "Database manager shutdown.")
|
|
}()
|
|
|
|
for {
|
|
select {
|
|
case <-a.shutdown:
|
|
return
|
|
case <-t.C:
|
|
go a.checkConnection()
|
|
}
|
|
}
|
|
}
|
|
|
|
func (a *databaseManager) checkConnection() {
|
|
dbConn.Mu.Lock()
|
|
defer dbConn.Mu.Unlock()
|
|
|
|
err := dbConn.SQL.Ping()
|
|
if err != nil {
|
|
log.Errorf(log.DatabaseMgr, "Database connection error: %v\n", err)
|
|
dbConn.Connected = false
|
|
return
|
|
}
|
|
|
|
if !dbConn.Connected {
|
|
log.Info(log.DatabaseMgr, "Database connection reestablished")
|
|
dbConn.Connected = true
|
|
}
|
|
}
|