statping-ng/cmd/main.go

167 lines
3.8 KiB
Go

package main
import (
"fmt"
"github.com/pkg/errors"
"github.com/statping/statping/database"
"github.com/statping/statping/handlers"
"github.com/statping/statping/notifiers"
"github.com/statping/statping/source"
"github.com/statping/statping/types/configs"
"github.com/statping/statping/types/core"
"github.com/statping/statping/types/metrics"
"github.com/statping/statping/types/services"
"github.com/statping/statping/utils"
"os"
"os/signal"
"syscall"
)
var (
// VERSION stores the current version of Statping
VERSION string
// COMMIT stores the git commit hash for this version of Statping
COMMIT string
log = utils.Log.WithField("type", "cmd")
confgs *configs.DbConfig
stopped chan bool
)
func init() {
stopped = make(chan bool, 1)
core.New(VERSION)
utils.InitEnvs()
rootCmd.AddCommand(versionCmd)
rootCmd.AddCommand(updateCmd)
rootCmd.AddCommand(assetsCmd)
rootCmd.AddCommand(exportCmd)
rootCmd.AddCommand(importCmd)
rootCmd.AddCommand(sassCmd)
rootCmd.AddCommand(onceCmd)
rootCmd.AddCommand(envCmd)
rootCmd.AddCommand(systemctlCmd)
rootCmd.AddCommand(resetCmd)
parseFlags(rootCmd)
}
// exit will return an error and return an exit code 1 due to this error
func exit(err error) {
utils.SentryErr(err)
log.Fatalln(err)
os.Exit(1)
}
// Close will gracefully stop the database connection, and log file
func Close() {
utils.CloseLogs()
confgs.Close()
fmt.Println("Shutting down Statping")
}
// main will run the Statping application
func main() {
go Execute()
<-stopped
Close()
}
// main will run the Statping application
func start() {
go sigterm()
var err error
if err := source.Assets(); err != nil {
exit(err)
}
utils.VerboseMode = verboseMode
if err := utils.InitLogs(); err != nil {
log.Errorf("Statping Log Error: %v\n", err)
}
log.Info(fmt.Sprintf("Starting Statping v%s", VERSION))
utils.Params.Set("SERVER_IP", ipAddress)
utils.Params.Set("SERVER_PORT", port)
confgs, err = configs.LoadConfigs(configFile)
if err != nil {
log.Infoln("Starting in Setup Mode")
if err = handlers.RunHTTPServer(); err != nil {
exit(err)
}
}
if err = configs.ConnectConfigs(confgs, true); err != nil {
exit(err)
}
if err = confgs.ResetCore(); err != nil {
exit(err)
}
if err = confgs.DatabaseChanges(); err != nil {
exit(err)
}
if err := confgs.MigrateDatabase(); err != nil {
exit(err)
}
if err := mainProcess(); err != nil {
exit(err)
}
}
// sigterm will attempt to close the database connections gracefully
func sigterm() {
sigs := make(chan os.Signal, 1)
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
<-sigs
stopped <- true
}
// mainProcess will initialize the Statping application and run the HTTP server
func mainProcess() error {
if err := InitApp(); err != nil {
return err
}
services.LoadServicesYaml()
if err := handlers.RunHTTPServer(); err != nil {
log.Fatalln(err)
return errors.Wrap(err, "http server")
}
return nil
}
// InitApp will start the Statping instance with a valid database connection
// This function will gather all services in database, add/init Notifiers,
// and start the database cleanup routine
func InitApp() error {
// fetch Core row information about this instance.
if _, err := core.Select(); err != nil {
return err
}
// init prometheus metrics
metrics.InitMetrics()
// select all services in database and store services in a mapping of Service pointers
if _, err := services.SelectAllServices(true); err != nil {
return err
}
// start routines for each service checking process
services.CheckServices()
// connect each notifier, added them into database if needed
notifiers.InitNotifiers()
// start routine to delete old records (failures, hits)
go database.Maintenance()
// init Sentry error monitoring (its useful)
utils.SentryInit(&VERSION, core.App.AllowReports.Bool)
core.App.Setup = true
core.App.Started = utils.Now()
return nil
}