package main import ( "flag" "fmt" "net/http" "os" "time" "github.com/VictoriaMetrics/VictoriaMetrics/app/vminsert" "github.com/VictoriaMetrics/VictoriaMetrics/app/vmselect" "github.com/VictoriaMetrics/VictoriaMetrics/app/vmselect/promql" "github.com/VictoriaMetrics/VictoriaMetrics/app/vmstorage" "github.com/VictoriaMetrics/VictoriaMetrics/lib/buildinfo" "github.com/VictoriaMetrics/VictoriaMetrics/lib/envflag" "github.com/VictoriaMetrics/VictoriaMetrics/lib/flagutil" "github.com/VictoriaMetrics/VictoriaMetrics/lib/fs" "github.com/VictoriaMetrics/VictoriaMetrics/lib/httpserver" "github.com/VictoriaMetrics/VictoriaMetrics/lib/logger" "github.com/VictoriaMetrics/VictoriaMetrics/lib/procutil" "github.com/VictoriaMetrics/VictoriaMetrics/lib/promscrape" "github.com/VictoriaMetrics/VictoriaMetrics/lib/storage" ) var ( httpListenAddr = flag.String("httpListenAddr", ":8428", "TCP address to listen for http connections") minScrapeInterval = flag.Duration("dedup.minScrapeInterval", 0, "Leave only the first sample in every time series per each discrete interval "+ "equal to -dedup.minScrapeInterval > 0. See https://docs.victoriametrics.com/#deduplication and https://docs.victoriametrics.com/#downsampling") dryRun = flag.Bool("dryRun", false, "Whether to check only -promscrape.config and then exit. "+ "Unknown config entries aren't allowed in -promscrape.config by default. This can be changed with -promscrape.config.strictParse=false command-line flag") downsamplingPeriods = flagutil.NewArray("downsampling.period", "Comma-separated downsampling periods in the format 'offset:period'. For example, '30d:10m' instructs "+ "to leave a single sample per 10 minutes for samples older than 30 days. See https://docs.victoriametrics.com/#downsampling for details") ) // custom api help links [["/api","doc"]] without http.pathPrefix. var customAPIPathList = [][]string{ {"/graph/explore", "explore metrics grafana page"}, {"/graph/d/prometheus-advanced/advanced-data-exploration", "PMM grafana dashboard"}, } func main() { // Write flags and help message to stdout, since it is easier to grep or pipe. flag.CommandLine.SetOutput(os.Stdout) flag.Usage = usage envflag.Parse() buildinfo.Init() logger.Init() if promscrape.IsDryRun() { *dryRun = true } if *dryRun { if err := promscrape.CheckConfig(); err != nil { logger.Fatalf("error when checking -promscrape.config: %s", err) } logger.Infof("-promscrape.config is ok; exitting with 0 status code") return } logger.Infof("starting VictoriaMetrics at %q...", *httpListenAddr) startTime := time.Now() err := storage.SetDownsamplingPeriods(*downsamplingPeriods, *minScrapeInterval) if err != nil { logger.Fatalf("cannot parse -downsampling.period: %s", err) } vmstorage.Init(promql.ResetRollupResultCacheIfNeeded) vmselect.Init() vminsert.Init() startSelfScraper() go httpserver.Serve(*httpListenAddr, requestHandler) logger.Infof("started VictoriaMetrics in %.3f seconds", time.Since(startTime).Seconds()) sig := procutil.WaitForSigterm() logger.Infof("received signal %s", sig) stopSelfScraper() logger.Infof("gracefully shutting down webservice at %q", *httpListenAddr) startTime = time.Now() if err := httpserver.Stop(*httpListenAddr); err != nil { logger.Fatalf("cannot stop the webservice: %s", err) } vminsert.Stop() logger.Infof("successfully shut down the webservice in %.3f seconds", time.Since(startTime).Seconds()) vmstorage.Stop() vmselect.Stop() fs.MustStopDirRemover() logger.Infof("the VictoriaMetrics has been stopped in %.3f seconds", time.Since(startTime).Seconds()) } func requestHandler(w http.ResponseWriter, r *http.Request) bool { if r.URL.Path == "/" { if r.Method != "GET" { return false } fmt.Fprintf(w, "

Single-node VictoriaMetrics


") fmt.Fprintf(w, "See docs at https://docs.victoriametrics.com/
") fmt.Fprintf(w, "Useful endpoints:
") httpserver.WriteAPIHelp(w, [][2]string{ {"vmui", "Web UI"}, {"targets", "discovered targets list"}, {"api/v1/targets", "advanced information about discovered targets in JSON format"}, {"config", "-promscrape.config contents"}, {"metrics", "available service metrics"}, {"flags", "command-line flags"}, {"api/v1/status/tsdb", "tsdb status page"}, {"api/v1/status/top_queries", "top queries"}, {"api/v1/status/active_queries", "active queries"}, }) for _, p := range customAPIPathList { p, doc := p[0], p[1] fmt.Fprintf(w, "%s - %s
", p, p, doc) } return true } if vminsert.RequestHandler(w, r) { return true } if vmselect.RequestHandler(w, r) { return true } if vmstorage.RequestHandler(w, r) { return true } return false } func usage() { const s = ` victoria-metrics is a time series database and monitoring solution. See the docs at https://docs.victoriametrics.com/ ` flagutil.Usage(s) }