2019-05-22 21:16:55 +00:00
package influx
import (
2019-06-14 06:57:13 +00:00
"flag"
2020-02-25 17:09:46 +00:00
"io"
2019-05-22 21:16:55 +00:00
"net/http"
"runtime"
"sync"
2019-05-22 21:23:23 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/app/vminsert/netstorage"
2020-07-02 16:42:12 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/app/vminsert/relabel"
2019-05-22 21:23:23 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/auth"
2019-05-22 21:16:55 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/bytesutil"
2020-07-23 09:50:41 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prompb"
2020-02-23 11:35:47 +00:00
parser "github.com/VictoriaMetrics/VictoriaMetrics/lib/protoparser/influx"
2019-05-22 21:16:55 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/storage"
2019-06-07 18:16:05 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/tenantmetrics"
2020-02-23 11:35:47 +00:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/writeconcurrencylimiter"
2019-05-22 21:16:55 +00:00
"github.com/VictoriaMetrics/metrics"
2020-05-12 10:13:00 +00:00
"github.com/valyala/fastjson/fastfloat"
2019-05-22 21:16:55 +00:00
)
2019-06-14 06:57:13 +00:00
var (
2020-02-04 13:46:13 +00:00
measurementFieldSeparator = flag . String ( "influxMeasurementFieldSeparator" , "_" , "Separator for '{measurement}{separator}{field_name}' metric name when inserted via Influx line protocol" )
skipSingleField = flag . Bool ( "influxSkipSingleField" , false , "Uses '{measurement}' instead of '{measurement}{separator}{field_name}' for metic name if Influx line contains only a single field" )
2020-07-14 11:17:22 +00:00
skipMeasurement = flag . Bool ( "influxSkipMeasurement" , false , "Uses '{field_name}' as a metric name while ignoring '{measurement}' and '-influxMeasurementFieldSeparator'" )
2019-06-14 06:57:13 +00:00
)
2019-07-27 10:20:47 +00:00
var (
rowsInserted = tenantmetrics . NewCounterMap ( ` vm_rows_inserted_total { type="influx"} ` )
2020-02-23 11:35:47 +00:00
rowsPerInsert = metrics . NewHistogram ( ` vm_rows_per_insert { type="influx"} ` )
2019-07-27 10:20:47 +00:00
)
2019-05-22 21:16:55 +00:00
2020-02-25 17:09:46 +00:00
// InsertHandlerForReader processes remote write for influx line protocol.
//
// See https://github.com/influxdata/telegraf/tree/master/plugins/inputs/socket_listener/
func InsertHandlerForReader ( at * auth . Token , r io . Reader ) error {
return writeconcurrencylimiter . Do ( func ( ) error {
return parser . ParseStream ( r , false , "" , "" , func ( db string , rows [ ] parser . Row ) error {
2020-05-12 10:13:00 +00:00
return insertRows ( at , db , rows , true )
2020-02-25 17:09:46 +00:00
} )
} )
}
// InsertHandlerForHTTP processes remote write for influx line protocol.
2019-05-22 21:16:55 +00:00
//
// See https://github.com/influxdata/influxdb/blob/4cbdc197b8117fee648d62e2e5be75c6575352f0/tsdb/README.md
2020-02-25 17:09:46 +00:00
func InsertHandlerForHTTP ( at * auth . Token , req * http . Request ) error {
2020-02-23 11:35:47 +00:00
return writeconcurrencylimiter . Do ( func ( ) error {
2020-02-25 17:09:46 +00:00
isGzipped := req . Header . Get ( "Content-Encoding" ) == "gzip"
q := req . URL . Query ( )
precision := q . Get ( "precision" )
// Read db tag from https://docs.influxdata.com/influxdb/v1.7/tools/api/#write-http-endpoint
db := q . Get ( "db" )
return parser . ParseStream ( req . Body , isGzipped , precision , db , func ( db string , rows [ ] parser . Row ) error {
2020-05-12 10:13:00 +00:00
return insertRows ( at , db , rows , false )
2020-02-23 11:35:47 +00:00
} )
2019-05-22 21:16:55 +00:00
} )
}
2020-05-12 10:13:00 +00:00
func insertRows ( at * auth . Token , db string , rows [ ] parser . Row , mayOverrideAccountProjectID bool ) error {
2019-05-22 21:16:55 +00:00
ctx := getPushCtx ( )
defer putPushCtx ( ctx )
ic := & ctx . Common
2020-02-26 19:17:35 +00:00
ic . Reset ( ) // This line is required for initializing ic internals.
2019-07-27 10:20:47 +00:00
rowsTotal := 0
2020-05-12 10:13:00 +00:00
atCopy := * at
2020-07-02 16:42:12 +00:00
hasRelabeling := relabel . HasRelabeling ( )
2019-05-22 21:16:55 +00:00
for i := range rows {
r := & rows [ i ]
ic . Labels = ic . Labels [ : 0 ]
for j := range r . Tags {
tag := & r . Tags [ j ]
2020-05-12 10:13:00 +00:00
if mayOverrideAccountProjectID {
// Multi-tenancy support via custom tags.
if tag . Key == "VictoriaMetrics_AccountID" {
atCopy . AccountID = uint32 ( fastfloat . ParseUint64BestEffort ( tag . Value ) )
}
if tag . Key == "VictoriaMetrics_ProjectID" {
atCopy . ProjectID = uint32 ( fastfloat . ParseUint64BestEffort ( tag . Value ) )
}
}
2019-08-24 10:51:51 +00:00
if tag . Key == "db" {
2020-07-23 09:50:41 +00:00
db = ""
2019-08-24 10:51:51 +00:00
}
2019-05-22 21:16:55 +00:00
ic . AddLabel ( tag . Key , tag . Value )
}
2020-07-23 09:50:41 +00:00
ic . AddLabel ( "db" , db )
2020-07-14 11:17:22 +00:00
ctx . metricGroupBuf = ctx . metricGroupBuf [ : 0 ]
if ! * skipMeasurement {
ctx . metricGroupBuf = append ( ctx . metricGroupBuf , r . Measurement ... )
}
2019-06-14 07:51:57 +00:00
skipFieldKey := len ( r . Fields ) == 1 && * skipSingleField
2019-11-30 19:54:34 +00:00
if len ( ctx . metricGroupBuf ) > 0 && ! skipFieldKey {
2019-06-14 07:51:57 +00:00
ctx . metricGroupBuf = append ( ctx . metricGroupBuf , * measurementFieldSeparator ... )
}
2019-05-22 21:16:55 +00:00
metricGroupPrefixLen := len ( ctx . metricGroupBuf )
2020-07-02 20:13:13 +00:00
if hasRelabeling {
2020-07-23 09:50:41 +00:00
ctx . originLabels = append ( ctx . originLabels [ : 0 ] , ic . Labels ... )
ic . MetricNameBuf = storage . MarshalMetricNameRaw ( ic . MetricNameBuf [ : 0 ] , atCopy . AccountID , atCopy . ProjectID , nil )
metricNameBufLen := len ( ic . MetricNameBuf )
for j := range r . Fields {
f := & r . Fields [ j ]
if ! skipFieldKey {
ctx . metricGroupBuf = append ( ctx . metricGroupBuf [ : metricGroupPrefixLen ] , f . Key ... )
}
metricGroup := bytesutil . ToUnsafeString ( ctx . metricGroupBuf )
ic . Labels = append ( ic . Labels [ : 0 ] , ctx . originLabels ... )
ic . AddLabel ( "" , metricGroup )
ic . ApplyRelabeling ( )
if len ( ic . Labels ) == 0 {
// Skip metric without labels.
continue
}
ic . MetricNameBuf = ic . MetricNameBuf [ : metricNameBufLen ]
for i := range ic . Labels {
ic . MetricNameBuf = storage . MarshalMetricLabelRaw ( ic . MetricNameBuf , & ic . Labels [ i ] )
}
storageNodeIdx := ic . GetStorageNodeIdx ( & atCopy , ic . Labels )
if err := ic . WriteDataPointExt ( & atCopy , storageNodeIdx , ic . MetricNameBuf , r . Timestamp , f . Value ) ; err != nil {
return err
}
2020-07-02 16:42:12 +00:00
}
2020-07-23 09:50:41 +00:00
} else {
ic . MetricNameBuf = storage . MarshalMetricNameRaw ( ic . MetricNameBuf [ : 0 ] , atCopy . AccountID , atCopy . ProjectID , ic . Labels )
metricNameBufLen := len ( ic . MetricNameBuf )
labelsLen := len ( ic . Labels )
for j := range r . Fields {
f := & r . Fields [ j ]
if ! skipFieldKey {
ctx . metricGroupBuf = append ( ctx . metricGroupBuf [ : metricGroupPrefixLen ] , f . Key ... )
}
metricGroup := bytesutil . ToUnsafeString ( ctx . metricGroupBuf )
ic . Labels = ic . Labels [ : labelsLen ]
ic . AddLabel ( "" , metricGroup )
if len ( ic . Labels ) == 0 {
// Skip metric without labels.
continue
}
ic . MetricNameBuf = ic . MetricNameBuf [ : metricNameBufLen ]
ic . MetricNameBuf = storage . MarshalMetricLabelRaw ( ic . MetricNameBuf , & ic . Labels [ len ( ic . Labels ) - 1 ] )
storageNodeIdx := ic . GetStorageNodeIdx ( & atCopy , ic . Labels )
if err := ic . WriteDataPointExt ( & atCopy , storageNodeIdx , ic . MetricNameBuf , r . Timestamp , f . Value ) ; err != nil {
return err
}
2019-05-22 21:23:23 +00:00
}
2019-05-22 21:16:55 +00:00
}
2019-07-27 10:20:47 +00:00
rowsTotal += len ( r . Fields )
2019-05-22 21:16:55 +00:00
}
2020-05-12 10:13:00 +00:00
rowsInserted . Get ( & atCopy ) . Add ( rowsTotal )
2019-07-27 10:20:47 +00:00
rowsPerInsert . Update ( float64 ( rowsTotal ) )
2019-05-22 21:16:55 +00:00
return ic . FlushBufs ( )
}
type pushCtx struct {
2020-02-23 11:35:47 +00:00
Common netstorage . InsertCtx
2019-05-22 21:16:55 +00:00
metricGroupBuf [ ] byte
2020-07-23 09:50:41 +00:00
originLabels [ ] prompb . Label
2019-05-22 21:16:55 +00:00
}
func ( ctx * pushCtx ) reset ( ) {
2019-05-22 21:23:23 +00:00
ctx . Common . Reset ( )
2019-05-22 21:16:55 +00:00
ctx . metricGroupBuf = ctx . metricGroupBuf [ : 0 ]
2020-07-23 09:50:41 +00:00
originLabels := ctx . originLabels
for i := range originLabels {
label := & originLabels [ i ]
label . Name = nil
label . Value = nil
}
ctx . originLabels = ctx . originLabels [ : 0 ]
2019-05-22 21:16:55 +00:00
}
func getPushCtx ( ) * pushCtx {
select {
case ctx := <- pushCtxPoolCh :
return ctx
default :
if v := pushCtxPool . Get ( ) ; v != nil {
return v . ( * pushCtx )
}
return & pushCtx { }
}
}
func putPushCtx ( ctx * pushCtx ) {
ctx . reset ( )
select {
case pushCtxPoolCh <- ctx :
default :
pushCtxPool . Put ( ctx )
}
}
var pushCtxPool sync . Pool
var pushCtxPoolCh = make ( chan * pushCtx , runtime . GOMAXPROCS ( - 1 ) )