hunterlong/statup

View on GitHub
cmd/main.go

Summary

Maintainability
A
0 mins
Test Coverage
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, COMMIT)
    utils.InitEnvs()
    utils.Params.Set("VERSION", VERSION)
    utils.Params.Set("COMMIT", COMMIT)

    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 Sentry error monitoring (its useful)
    utils.SentryInit(core.App.AllowReports.Bool)
    // init prometheus metrics
    metrics.InitMetrics()
    // connect each notifier, added them into database if needed
    notifiers.InitNotifiers()
    // 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()
    // start routine to delete old records (failures, hits)
    go database.Maintenance()
    core.App.Setup = true
    core.App.Started = utils.Now()
    return nil
}