2020-02-23 12:35:47 +01:00
package main
import (
"flag"
"fmt"
"net/http"
2020-05-16 10:59:30 +02:00
"os"
2020-02-23 12:35:47 +01:00
"strings"
2020-11-04 19:29:18 +01:00
"sync/atomic"
2020-02-23 12:35:47 +01:00
"time"
2020-03-10 18:35:58 +01:00
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/csvimport"
2020-02-23 12:35:47 +01:00
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/graphite"
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/influx"
2020-09-26 03:29:45 +02:00
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/native"
2020-02-23 12:35:47 +01:00
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/opentsdb"
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/opentsdbhttp"
2020-07-10 11:00:35 +02:00
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/prometheusimport"
2020-02-23 12:35:47 +01:00
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/promremotewrite"
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/remotewrite"
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmagent/vmimport"
"github.com/VictoriaMetrics/VictoriaMetrics/lib/buildinfo"
"github.com/VictoriaMetrics/VictoriaMetrics/lib/envflag"
2020-12-03 20:40:30 +01:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/flagutil"
2020-02-23 12:35:47 +01:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/httpserver"
graphiteserver "github.com/VictoriaMetrics/VictoriaMetrics/lib/ingestserver/graphite"
2020-02-25 18:09:46 +01:00
influxserver "github.com/VictoriaMetrics/VictoriaMetrics/lib/ingestserver/influx"
2020-02-23 12:35:47 +01:00
opentsdbserver "github.com/VictoriaMetrics/VictoriaMetrics/lib/ingestserver/opentsdb"
opentsdbhttpserver "github.com/VictoriaMetrics/VictoriaMetrics/lib/ingestserver/opentsdbhttp"
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logger"
"github.com/VictoriaMetrics/VictoriaMetrics/lib/procutil"
"github.com/VictoriaMetrics/VictoriaMetrics/lib/promscrape"
2020-09-28 03:11:55 +02:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/protoparser/common"
2020-02-23 12:35:47 +01:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/writeconcurrencylimiter"
"github.com/VictoriaMetrics/metrics"
)
var (
2020-02-26 19:57:51 +01:00
httpListenAddr = flag . String ( "httpListenAddr" , ":8429" , "TCP address to listen for http connections. " +
"Set this flag to empty value in order to disable listening on any port. This mode may be useful for running multiple vmagent instances on the same server. " +
"Note that /targets and /metrics pages aren't available if -httpListenAddr=''" )
2020-08-14 17:01:00 +02:00
influxListenAddr = flag . String ( "influxListenAddr" , "" , "TCP and UDP address to listen for Influx line protocol data. Usually :8189 must be set. Doesn't work if empty. " +
"This flag isn't needed when ingesting data over HTTP - just send it to `http://<vmagent>:8429/write`" )
2020-02-23 12:35:47 +01:00
graphiteListenAddr = flag . String ( "graphiteListenAddr" , "" , "TCP and UDP address to listen for Graphite plaintext data. Usually :2003 must be set. Doesn't work if empty" )
opentsdbListenAddr = flag . String ( "opentsdbListenAddr" , "" , "TCP and UDP address to listen for OpentTSDB metrics. " +
"Telnet put messages and HTTP /api/put messages are simultaneously served on TCP port. " +
"Usually :4242 must be set. Doesn't work if empty" )
opentsdbHTTPListenAddr = flag . String ( "opentsdbHTTPListenAddr" , "" , "TCP address to listen for OpentTSDB HTTP put requests. Usually :4242 must be set. Doesn't work if empty" )
2020-05-21 14:22:01 +02:00
dryRun = flag . Bool ( "dryRun" , false , "Whether to check only config files without running vmagent. The following files are checked: " +
2020-11-25 21:26:23 +01:00
"-promscrape.config, -remoteWrite.relabelConfig, -remoteWrite.urlRelabelConfig . " +
"Unknown config entries are allowed in -promscrape.config by default. This can be changed with -promscrape.config.strictParse" )
2021-03-15 20:10:47 +01:00
influxDatabasesNames = flag . String ( "influx.databasesNames" , "_internal" , "Comma separated names of databases, that will be returned for /query and /influx/query api." )
2020-02-23 12:35:47 +01:00
)
var (
2020-02-25 18:09:46 +01:00
influxServer * influxserver . Server
2020-02-23 12:35:47 +01:00
graphiteServer * graphiteserver . Server
opentsdbServer * opentsdbserver . Server
opentsdbhttpServer * opentsdbhttpserver . Server
)
func main ( ) {
2020-05-16 10:59:30 +02:00
// Write flags and help message to stdout, since it is easier to grep or pipe.
flag . CommandLine . SetOutput ( os . Stdout )
2020-06-05 09:37:36 +02:00
flag . Usage = usage
2020-02-23 12:35:47 +01:00
envflag . Parse ( )
2020-09-29 18:48:53 +02:00
remotewrite . InitSecretFlags ( )
2020-02-23 12:35:47 +01:00
buildinfo . Init ( )
logger . Init ( )
2020-05-21 14:22:01 +02:00
2020-11-25 21:59:13 +01:00
if promscrape . IsDryRun ( ) {
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
}
2020-05-21 14:22:01 +02:00
if * dryRun {
if err := remotewrite . CheckRelabelConfigs ( ) ; err != nil {
logger . Fatalf ( "error when checking relabel configs: %s" , err )
}
if err := promscrape . CheckConfig ( ) ; err != nil {
2020-11-25 21:59:13 +01:00
logger . Fatalf ( "error when checking -promscrape.config: %s" , err )
2020-05-21 14:22:01 +02:00
}
logger . Infof ( "all the configs are ok; exitting with 0 status code" )
return
}
2020-02-23 12:35:47 +01:00
logger . Infof ( "starting vmagent at %q..." , * httpListenAddr )
startTime := time . Now ( )
remotewrite . Init ( )
2020-09-28 03:11:55 +02:00
common . StartUnmarshalWorkers ( )
2020-02-23 12:35:47 +01:00
writeconcurrencylimiter . Init ( )
2020-02-25 18:09:46 +01:00
if len ( * influxListenAddr ) > 0 {
influxServer = influxserver . MustStart ( * influxListenAddr , influx . InsertHandlerForReader )
}
2020-02-23 12:35:47 +01:00
if len ( * graphiteListenAddr ) > 0 {
graphiteServer = graphiteserver . MustStart ( * graphiteListenAddr , graphite . InsertHandler )
}
if len ( * opentsdbListenAddr ) > 0 {
opentsdbServer = opentsdbserver . MustStart ( * opentsdbListenAddr , opentsdb . InsertHandler , opentsdbhttp . InsertHandler )
}
if len ( * opentsdbHTTPListenAddr ) > 0 {
opentsdbhttpServer = opentsdbhttpserver . MustStart ( * opentsdbHTTPListenAddr , opentsdbhttp . InsertHandler )
}
promscrape . Init ( remotewrite . Push )
2020-02-26 19:57:51 +01:00
if len ( * httpListenAddr ) > 0 {
go httpserver . Serve ( * httpListenAddr , requestHandler )
}
2020-02-23 12:35:47 +01:00
logger . Infof ( "started vmagent in %.3f seconds" , time . Since ( startTime ) . Seconds ( ) )
sig := procutil . WaitForSigterm ( )
logger . Infof ( "received signal %s" , sig )
startTime = time . Now ( )
2020-02-26 19:57:51 +01:00
if len ( * httpListenAddr ) > 0 {
logger . Infof ( "gracefully shutting down webservice at %q" , * httpListenAddr )
if err := httpserver . Stop ( * httpListenAddr ) ; err != nil {
logger . Fatalf ( "cannot stop the webservice: %s" , err )
}
logger . Infof ( "successfully shut down the webservice in %.3f seconds" , time . Since ( startTime ) . Seconds ( ) )
2020-02-23 12:35:47 +01:00
}
promscrape . Stop ( )
2020-02-25 18:09:46 +01:00
if len ( * influxListenAddr ) > 0 {
influxServer . MustStop ( )
}
2020-02-23 12:35:47 +01:00
if len ( * graphiteListenAddr ) > 0 {
graphiteServer . MustStop ( )
}
if len ( * opentsdbListenAddr ) > 0 {
opentsdbServer . MustStop ( )
}
if len ( * opentsdbHTTPListenAddr ) > 0 {
opentsdbhttpServer . MustStop ( )
}
2020-09-28 03:11:55 +02:00
common . StopUnmarshalWorkers ( )
2020-02-23 12:35:47 +01:00
remotewrite . Stop ( )
logger . Infof ( "successfully stopped vmagent in %.3f seconds" , time . Since ( startTime ) . Seconds ( ) )
}
func requestHandler ( w http . ResponseWriter , r * http . Request ) bool {
2020-12-14 13:02:57 +01:00
if r . URL . Path == "/" {
2020-10-06 14:09:07 +02:00
fmt . Fprintf ( w , "vmagent - see docs at https://victoriametrics.github.io/vmagent.html" )
return true
}
2020-02-23 12:35:47 +01:00
path := strings . Replace ( r . URL . Path , "//" , "/" , - 1 )
switch path {
case "/api/v1/write" :
prometheusWriteRequests . Inc ( )
if err := promremotewrite . InsertHandler ( r ) ; err != nil {
prometheusWriteErrors . Inc ( )
2020-07-20 13:00:33 +02:00
httpserver . Errorf ( w , r , "error in %q: %s" , r . URL . Path , err )
2020-02-23 12:35:47 +01:00
return true
}
w . WriteHeader ( http . StatusNoContent )
return true
case "/api/v1/import" :
vmimportRequests . Inc ( )
if err := vmimport . InsertHandler ( r ) ; err != nil {
vmimportErrors . Inc ( )
2020-07-20 13:00:33 +02:00
httpserver . Errorf ( w , r , "error in %q: %s" , r . URL . Path , err )
2020-02-23 12:35:47 +01:00
return true
}
w . WriteHeader ( http . StatusNoContent )
return true
2020-03-10 18:35:58 +01:00
case "/api/v1/import/csv" :
csvimportRequests . Inc ( )
if err := csvimport . InsertHandler ( r ) ; err != nil {
csvimportErrors . Inc ( )
2020-07-20 13:00:33 +02:00
httpserver . Errorf ( w , r , "error in %q: %s" , r . URL . Path , err )
2020-03-10 18:35:58 +01:00
return true
}
w . WriteHeader ( http . StatusNoContent )
return true
2020-07-10 11:00:35 +02:00
case "/api/v1/import/prometheus" :
prometheusimportRequests . Inc ( )
if err := prometheusimport . InsertHandler ( r ) ; err != nil {
prometheusimportErrors . Inc ( )
2020-07-20 13:00:33 +02:00
httpserver . Errorf ( w , r , "error in %q: %s" , r . URL . Path , err )
2020-07-10 11:00:35 +02:00
return true
}
w . WriteHeader ( http . StatusNoContent )
return true
2020-09-26 03:29:45 +02:00
case "/api/v1/import/native" :
nativeimportRequests . Inc ( )
if err := native . InsertHandler ( r ) ; err != nil {
nativeimportErrors . Inc ( )
httpserver . Errorf ( w , r , "error in %q: %s" , r . URL . Path , err )
return true
}
w . WriteHeader ( http . StatusNoContent )
return true
2020-02-23 12:35:47 +01:00
case "/write" , "/api/v2/write" :
influxWriteRequests . Inc ( )
2020-02-25 18:09:46 +01:00
if err := influx . InsertHandlerForHTTP ( r ) ; err != nil {
2020-02-23 12:35:47 +01:00
influxWriteErrors . Inc ( )
2020-07-20 13:00:33 +02:00
httpserver . Errorf ( w , r , "error in %q: %s" , r . URL . Path , err )
2020-02-23 12:35:47 +01:00
return true
}
w . WriteHeader ( http . StatusNoContent )
return true
case "/query" :
// Emulate fake response for influx query.
2021-03-15 20:10:47 +01:00
// This is required for TSBS benchmark and some telegraph plugins.
2020-02-23 12:35:47 +01:00
influxQueryRequests . Inc ( )
2021-03-15 20:10:47 +01:00
var dbs string
influxDbs := strings . Split ( * influxDatabasesNames , "," )
for i := range influxDbs {
dbs += fmt . Sprintf ( ` "[%s]" ` , influxDbs [ i ] )
if i != len ( influxDbs ) - 1 {
dbs += ","
}
}
fmt . Fprintf ( w , ` { "results":[ { "name":"databases","columns":["name"],"series":[ { "values":[%s]}]}]} ` , dbs )
2020-02-23 12:35:47 +01:00
return true
2020-02-25 17:13:11 +01:00
case "/targets" :
promscrapeTargetsRequests . Inc ( )
2020-12-14 13:02:57 +01:00
promscrape . WriteHumanReadableTargetsStatus ( w , r )
2020-02-25 17:13:11 +01:00
return true
2020-10-20 20:44:59 +02:00
case "/api/v1/targets" :
promscrapeAPIV1TargetsRequests . Inc ( )
2020-11-13 09:17:37 +01:00
w . Header ( ) . Set ( "Content-Type" , "application/json; charset=utf-8" )
2020-10-20 20:44:59 +02:00
state := r . FormValue ( "state" )
promscrape . WriteAPIV1Targets ( w , state )
return true
2020-04-30 01:00:32 +02:00
case "/-/reload" :
promscrapeConfigReloadRequests . Inc ( )
2020-04-30 01:15:39 +02:00
procutil . SelfSIGHUP ( )
2020-05-07 18:29:27 +02:00
w . WriteHeader ( http . StatusOK )
2020-04-30 01:00:32 +02:00
return true
2020-11-04 19:29:18 +01:00
case "/ready" :
if rdy := atomic . LoadInt32 ( & promscrape . PendingScrapeConfigs ) ; rdy > 0 {
errMsg := fmt . Sprintf ( "waiting for scrapes to init, left: %d" , rdy )
http . Error ( w , errMsg , http . StatusTooEarly )
} else {
2020-11-13 09:25:39 +01:00
w . Header ( ) . Set ( "Content-Type" , "text/plain; charset=utf-8" )
2020-11-04 19:29:18 +01:00
w . WriteHeader ( http . StatusOK )
w . Write ( [ ] byte ( "OK" ) )
}
return true
2020-02-23 12:35:47 +01:00
}
return false
}
var (
2020-03-10 18:35:58 +01:00
prometheusWriteRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/api/v1/write", protocol="promremotewrite"} ` )
prometheusWriteErrors = metrics . NewCounter ( ` vmagent_http_request_errors_total { path="/api/v1/write", protocol="promremotewrite"} ` )
vmimportRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/api/v1/import", protocol="vmimport"} ` )
vmimportErrors = metrics . NewCounter ( ` vmagent_http_request_errors_total { path="/api/v1/import", protocol="vmimport"} ` )
2020-02-23 12:35:47 +01:00
2020-03-10 18:35:58 +01:00
csvimportRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/api/v1/import/csv", protocol="csvimport"} ` )
csvimportErrors = metrics . NewCounter ( ` vmagent_http_request_errors_total { path="/api/v1/import/csv", protocol="csvimport"} ` )
2020-02-23 12:35:47 +01:00
2020-07-10 11:00:35 +02:00
prometheusimportRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/api/v1/import/prometheus", protocol="prometheusimport"} ` )
prometheusimportErrors = metrics . NewCounter ( ` vmagent_http_request_errors_total { path="/api/v1/import/prometheus", protocol="prometheusimport"} ` )
2020-09-26 03:29:45 +02:00
nativeimportRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/api/v1/import/native", protocol="nativeimport"} ` )
nativeimportErrors = metrics . NewCounter ( ` vmagent_http_request_errors_total { path="/api/v1/import/native", protocol="nativeimport"} ` )
2020-02-23 12:35:47 +01:00
influxWriteRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/write", protocol="influx"} ` )
influxWriteErrors = metrics . NewCounter ( ` vmagent_http_request_errors_total { path="/write", protocol="influx"} ` )
influxQueryRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/query", protocol="influx"} ` )
2020-02-25 17:13:11 +01:00
2020-10-20 20:44:59 +02:00
promscrapeTargetsRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/targets"} ` )
promscrapeAPIV1TargetsRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/api/v1/targets"} ` )
2020-04-30 01:00:32 +02:00
promscrapeConfigReloadRequests = metrics . NewCounter ( ` vmagent_http_requests_total { path="/-/reload"} ` )
2020-02-23 12:35:47 +01:00
)
2020-06-05 09:37:36 +02:00
func usage ( ) {
const s = `
vmagent collects metrics data via popular data ingestion protocols and routes it to VictoriaMetrics .
2020-12-11 20:08:13 +01:00
See the docs at https : //victoriametrics.github.io/vmagent.html .
2020-06-05 09:37:36 +02:00
`
2020-12-03 20:40:30 +01:00
flagutil . Usage ( s )
2020-06-05 09:37:36 +02:00
}