2016-08-05 07:47:04 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2016-08-05 09:35:17 +02:00
|
|
|
"io/ioutil"
|
2016-08-05 07:47:04 +02:00
|
|
|
"log"
|
2016-08-05 09:35:17 +02:00
|
|
|
"net/http"
|
2016-08-05 07:47:04 +02:00
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
"runtime/pprof"
|
|
|
|
"syscall"
|
|
|
|
|
2016-08-05 09:35:17 +02:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
2016-08-05 07:47:04 +02:00
|
|
|
"github.com/spf13/cobra"
|
2016-08-05 09:35:17 +02:00
|
|
|
"gopkg.in/yaml.v2"
|
2016-08-05 07:47:04 +02:00
|
|
|
|
|
|
|
"github.com/jzelinskie/trakr"
|
|
|
|
)
|
|
|
|
|
2016-08-05 09:35:17 +02:00
|
|
|
type ConfigFile struct {
|
|
|
|
Config struct {
|
|
|
|
PrometheusAddr string `yaml:"prometheus_addr"`
|
|
|
|
trakr.MultiTracker
|
|
|
|
} `yaml:"trakr"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// ParseConfigFile returns a new ConfigFile given the path to a YAML
|
|
|
|
// configuration file.
|
|
|
|
//
|
|
|
|
// It supports relative and absolute paths and environment variables.
|
|
|
|
func ParseConfigFile(path string) (*ConfigFile, error) {
|
|
|
|
if path == "" {
|
|
|
|
return nil, errors.New("no config path specified")
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := os.Open(os.ExpandEnv(path))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
contents, err := ioutil.ReadAll(f)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var cfgFile ConfigFile
|
|
|
|
err = yaml.Unmarshal(contents, &cfgFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &cfgFile, nil
|
|
|
|
}
|
|
|
|
|
2016-08-05 07:47:04 +02:00
|
|
|
func main() {
|
|
|
|
var configFilePath string
|
|
|
|
var cpuProfilePath string
|
|
|
|
|
|
|
|
var rootCmd = &cobra.Command{
|
|
|
|
Use: "trakr",
|
|
|
|
Short: "BitTorrent Tracker",
|
|
|
|
Long: "A customizible, multi-protocol BitTorrent Tracker",
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
if err := func() error {
|
|
|
|
if cpuProfilePath != "" {
|
|
|
|
log.Println("enabled CPU profiling to " + cpuProfilePath)
|
|
|
|
f, err := os.Create(cpuProfilePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
pprof.StartCPUProfile(f)
|
|
|
|
defer pprof.StopCPUProfile()
|
|
|
|
}
|
|
|
|
|
2016-08-05 09:35:17 +02:00
|
|
|
configFile, err := ParseConfigFile(configFilePath)
|
2016-08-05 07:47:04 +02:00
|
|
|
if err != nil {
|
|
|
|
return errors.New("failed to read config: " + err.Error())
|
|
|
|
}
|
|
|
|
|
2016-08-05 09:35:17 +02:00
|
|
|
go func() {
|
|
|
|
promServer := http.Server{
|
|
|
|
Addr: configFile.Config.PrometheusAddr,
|
|
|
|
Handler: prometheus.Handler(),
|
|
|
|
}
|
|
|
|
log.Println("started serving prometheus stats on", configFile.Config.PrometheusAddr)
|
|
|
|
if err := promServer.ListenAndServe(); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2016-08-05 07:47:04 +02:00
|
|
|
go func() {
|
|
|
|
shutdown := make(chan os.Signal)
|
|
|
|
signal.Notify(shutdown, syscall.SIGINT, syscall.SIGTERM)
|
|
|
|
<-shutdown
|
2016-08-05 09:35:17 +02:00
|
|
|
configFile.Config.MultiTracker.Stop()
|
2016-08-05 07:47:04 +02:00
|
|
|
}()
|
|
|
|
|
2016-08-05 09:35:17 +02:00
|
|
|
if err := configFile.Config.MultiTracker.ListenAndServe(); err != nil {
|
2016-08-05 07:47:04 +02:00
|
|
|
return errors.New("failed to cleanly shutdown: " + err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}(); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
rootCmd.Flags().StringVar(&configFilePath, "config", "/etc/trakr.yaml", "location of configuration file (defaults to /etc/trakr.yaml)")
|
|
|
|
rootCmd.Flags().StringVarP(&cpuProfilePath, "cpuprofile", "", "", "location to save a CPU profile")
|
|
|
|
|
|
|
|
if err := rootCmd.Execute(); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|