2021-06-09 11:20:38 +02:00
package datasource
import (
"encoding/json"
2022-05-13 15:19:32 +02:00
"flag"
2021-06-09 11:20:38 +02:00
"fmt"
"net/http"
"strconv"
"time"
2024-05-15 15:18:33 +02:00
"github.com/VictoriaMetrics/VictoriaMetrics/lib/bytesutil"
"github.com/valyala/fastjson"
2021-06-09 11:20:38 +02:00
)
2022-05-13 15:19:32 +02:00
var (
2024-10-29 16:30:39 +01:00
disablePathAppend = flag . Bool ( "remoteRead.disablePathAppend" , false , "Whether to disable automatic appending of '/api/v1/query' or '/select/logsql/stats_query' path " +
2022-05-13 15:19:32 +02:00
"to the configured -datasource.url and -remoteRead.url" )
2023-07-07 07:44:34 +02:00
disableStepParam = flag . Bool ( "datasource.disableStepParam" , false , "Whether to disable adding 'step' param to the issued instant queries. " +
"This might be useful when using vmalert with datasources that do not support 'step' param for instant queries, like Google Managed Prometheus. " +
"It is not recommended to enable this flag if you use vmalert with VictoriaMetrics." )
2022-05-13 15:19:32 +02:00
)
2021-06-09 11:20:38 +02:00
type promResponse struct {
Status string ` json:"status" `
ErrorType string ` json:"errorType" `
Error string ` json:"error" `
Data struct {
ResultType string ` json:"resultType" `
Result json . RawMessage ` json:"result" `
} ` json:"data" `
2023-05-08 09:36:39 +02:00
// Stats supported by VictoriaMetrics since v1.90
Stats struct {
SeriesFetched * string ` json:"seriesFetched,omitempty" `
} ` json:"stats,omitempty" `
2021-06-09 11:20:38 +02:00
}
2024-05-15 15:18:33 +02:00
// see https://prometheus.io/docs/prometheus/latest/querying/api/#instant-queries
2021-06-09 11:20:38 +02:00
type promInstant struct {
2024-05-15 15:18:33 +02:00
// ms is populated after Unmarshal call
ms [ ] Metric
2021-06-09 11:20:38 +02:00
}
2024-05-15 15:18:33 +02:00
// metrics returned parsed Metric slice
// Must be called only after Unmarshal
func ( pi * promInstant ) metrics ( ) ( [ ] Metric , error ) {
return pi . ms , nil
}
var jsonParserPool fastjson . ParserPool
// Unmarshal unmarshals the given byte slice into promInstant
// It is using fastjson to reduce number of allocations compared to
// standard json.Unmarshal function.
// Response example:
//
// [{"metric":{"__name__":"up","job":"prometheus"},value": [ 1435781451.781,"1"]},
// {"metric":{"__name__":"up","job":"node"},value": [ 1435781451.781,"0"]}]
func ( pi * promInstant ) Unmarshal ( b [ ] byte ) error {
2024-08-22 17:36:11 +02:00
var metrics [ ] json . RawMessage
// metrics slice could be large, so parsing it with fastjson could consume a lot of memory.
// We parse the slice with standard lib to keep mem usage low.
// And each metric object will be parsed with fastjson to reduce allocations.
if err := json . Unmarshal ( b , & metrics ) ; err != nil {
return fmt . Errorf ( "cannot unmarshal metrics: %w" , err )
}
2024-05-15 15:18:33 +02:00
p := jsonParserPool . Get ( )
defer jsonParserPool . Put ( p )
2024-08-22 17:36:11 +02:00
pi . ms = make ( [ ] Metric , len ( metrics ) )
for i , data := range metrics {
row , err := p . ParseBytes ( data )
if err != nil {
return fmt . Errorf ( "cannot parse metric object: %w" , err )
}
2024-05-15 15:18:33 +02:00
metric := row . Get ( "metric" )
if metric == nil {
return fmt . Errorf ( "can't find `metric` object in %q" , row )
}
labels := metric . GetObject ( )
r := & pi . ms [ i ]
r . Labels = make ( [ ] Label , 0 , labels . Len ( ) )
labels . Visit ( func ( key [ ] byte , v * fastjson . Value ) {
lv , errLocal := v . StringBytes ( )
if errLocal != nil {
err = fmt . Errorf ( "error when parsing label value %q: %s" , v , errLocal )
return
}
r . Labels = append ( r . Labels , Label {
Name : string ( key ) ,
Value : string ( lv ) ,
} )
} )
2021-06-09 11:20:38 +02:00
if err != nil {
2024-05-15 15:18:33 +02:00
return fmt . Errorf ( "error when parsing `metric` object in %q: %w" , row , err )
2021-06-09 11:20:38 +02:00
}
2024-05-15 15:18:33 +02:00
value := row . Get ( "value" )
if value == nil {
return fmt . Errorf ( "can't find `value` object in %q" , row )
}
sample := value . GetArray ( )
if len ( sample ) != 2 {
return fmt . Errorf ( "object `value` in %q should contain 2 values, but contains %d instead" , row , len ( sample ) )
}
r . Timestamps = [ ] int64 { sample [ 0 ] . GetInt64 ( ) }
val , err := sample [ 1 ] . StringBytes ( )
if err != nil {
return fmt . Errorf ( "error when parsing `value` object %q: %s" , sample [ 1 ] , err )
}
f , err := strconv . ParseFloat ( bytesutil . ToUnsafeString ( val ) , 64 )
if err != nil {
return fmt . Errorf ( "error when parsing float64 from %s in %q: %w" , sample [ 1 ] , row , err )
}
r . Values = [ ] float64 { f }
2021-06-09 11:20:38 +02:00
}
2024-05-15 15:18:33 +02:00
return nil
2021-06-09 11:20:38 +02:00
}
2022-05-18 09:50:46 +02:00
type promRange struct {
Result [ ] struct {
Labels map [ string ] string ` json:"metric" `
2024-07-10 00:14:15 +02:00
TVs [ ] [ 2 ] any ` json:"values" `
2022-05-18 09:50:46 +02:00
} ` json:"result" `
}
2021-06-09 11:20:38 +02:00
func ( r promRange ) metrics ( ) ( [ ] Metric , error ) {
var result [ ] Metric
for i , res := range r . Result {
var m Metric
for _ , tv := range res . TVs {
f , err := strconv . ParseFloat ( tv [ 1 ] . ( string ) , 64 )
if err != nil {
return nil , fmt . Errorf ( "metric %v, unable to parse float64 from %s: %w" , res , tv [ 1 ] , err )
}
m . Values = append ( m . Values , f )
m . Timestamps = append ( m . Timestamps , int64 ( tv [ 0 ] . ( float64 ) ) )
}
if len ( m . Values ) < 1 || len ( m . Timestamps ) < 1 {
return nil , fmt . Errorf ( "metric %v contains no values" , res )
}
m . Labels = nil
for k , v := range r . Result [ i ] . Labels {
m . AddLabel ( k , v )
}
result = append ( result , m )
}
return result , nil
}
2024-07-10 00:14:15 +02:00
type promScalar [ 2 ] any
2022-05-18 09:50:46 +02:00
func ( r promScalar ) metrics ( ) ( [ ] Metric , error ) {
var m Metric
f , err := strconv . ParseFloat ( r [ 1 ] . ( string ) , 64 )
if err != nil {
return nil , fmt . Errorf ( "metric %v, unable to parse float64 from %s: %w" , r , r [ 1 ] , err )
}
m . Values = append ( m . Values , f )
m . Timestamps = append ( m . Timestamps , int64 ( r [ 0 ] . ( float64 ) ) )
return [ ] Metric { m } , nil
}
2021-06-09 11:20:38 +02:00
const (
2022-05-18 09:50:46 +02:00
statusSuccess , statusError = "success" , "error"
rtVector , rtMatrix , rScalar = "vector" , "matrix" , "scalar"
2021-06-09 11:20:38 +02:00
)
2023-05-08 09:36:39 +02:00
func parsePrometheusResponse ( req * http . Request , resp * http . Response ) ( res Result , err error ) {
2021-06-09 11:20:38 +02:00
r := & promResponse { }
2023-05-08 09:36:39 +02:00
if err = json . NewDecoder ( resp . Body ) . Decode ( r ) ; err != nil {
2024-10-29 16:30:39 +01:00
return res , fmt . Errorf ( "error parsing response from %s: %w" , req . URL . Redacted ( ) , err )
2021-06-09 11:20:38 +02:00
}
if r . Status == statusError {
2023-05-08 09:36:39 +02:00
return res , fmt . Errorf ( "response error, query: %s, errorType: %s, error: %s" , req . URL . Redacted ( ) , r . ErrorType , r . Error )
2021-06-09 11:20:38 +02:00
}
if r . Status != statusSuccess {
2023-10-25 21:24:01 +02:00
return res , fmt . Errorf ( "unknown status: %s, Expected success or error" , r . Status )
2021-06-09 11:20:38 +02:00
}
2023-05-08 09:36:39 +02:00
var parseFn func ( ) ( [ ] Metric , error )
2021-06-09 11:20:38 +02:00
switch r . Data . ResultType {
case rtVector :
var pi promInstant
2024-05-15 15:18:33 +02:00
if err := pi . Unmarshal ( r . Data . Result ) ; err != nil {
2023-10-25 21:24:01 +02:00
return res , fmt . Errorf ( "unmarshal err %w; \n %#v" , err , string ( r . Data . Result ) )
2021-06-09 11:20:38 +02:00
}
2023-05-08 09:36:39 +02:00
parseFn = pi . metrics
2021-06-09 11:20:38 +02:00
case rtMatrix :
var pr promRange
if err := json . Unmarshal ( r . Data . Result , & pr . Result ) ; err != nil {
2023-05-08 09:36:39 +02:00
return res , err
2021-06-09 11:20:38 +02:00
}
2023-05-08 09:36:39 +02:00
parseFn = pr . metrics
2022-05-18 09:50:46 +02:00
case rScalar :
var ps promScalar
if err := json . Unmarshal ( r . Data . Result , & ps ) ; err != nil {
2023-05-08 09:36:39 +02:00
return res , err
2022-05-18 09:50:46 +02:00
}
2023-05-08 09:36:39 +02:00
parseFn = ps . metrics
2021-06-09 11:20:38 +02:00
default :
2023-05-08 09:36:39 +02:00
return res , fmt . Errorf ( "unknown result type %q" , r . Data . ResultType )
2021-06-09 11:20:38 +02:00
}
2023-05-08 09:36:39 +02:00
ms , err := parseFn ( )
if err != nil {
return res , err
}
res = Result { Data : ms }
if r . Stats . SeriesFetched != nil {
intV , err := strconv . Atoi ( * r . Stats . SeriesFetched )
if err != nil {
return res , fmt . Errorf ( "failed to convert stats.seriesFetched to int: %w" , err )
}
res . SeriesFetched = & intV
}
return res , nil
2021-06-09 11:20:38 +02:00
}
2024-10-29 16:30:39 +01:00
func ( s * Client ) setPrometheusInstantReqParams ( r * http . Request , query string , timestamp time . Time ) {
2021-06-09 11:20:38 +02:00
if s . appendTypePrefix {
2022-05-13 15:19:32 +02:00
r . URL . Path += "/prometheus"
2021-06-09 11:20:38 +02:00
}
2022-05-13 15:19:32 +02:00
if ! * disablePathAppend {
r . URL . Path += "/api/v1/query"
2021-10-18 09:24:52 +02:00
}
2021-06-09 11:20:38 +02:00
q := r . URL . Query ( )
2023-07-07 10:39:25 +02:00
q . Set ( "time" , timestamp . Format ( time . RFC3339 ) )
2023-07-07 07:44:34 +02:00
if ! * disableStepParam && s . evaluationInterval > 0 { // set step as evaluationInterval by default
2022-12-01 13:57:53 +01:00
// always convert to seconds to keep compatibility with older
// Prometheus versions. See https://github.com/VictoriaMetrics/VictoriaMetrics/issues/1943
q . Set ( "step" , fmt . Sprintf ( "%ds" , int ( s . evaluationInterval . Seconds ( ) ) ) )
}
2023-07-07 07:44:34 +02:00
if ! * disableStepParam && s . queryStep > 0 { // override step with user-specified value
2022-12-01 13:57:53 +01:00
// always convert to seconds to keep compatibility with older
// Prometheus versions. See https://github.com/VictoriaMetrics/VictoriaMetrics/issues/1943
q . Set ( "step" , fmt . Sprintf ( "%ds" , int ( s . queryStep . Seconds ( ) ) ) )
}
2021-06-09 11:20:38 +02:00
r . URL . RawQuery = q . Encode ( )
2024-10-29 16:30:39 +01:00
s . setReqParams ( r , query )
2021-06-09 11:20:38 +02:00
}
2024-10-29 16:30:39 +01:00
func ( s * Client ) setPrometheusRangeReqParams ( r * http . Request , query string , start , end time . Time ) {
2021-06-09 11:20:38 +02:00
if s . appendTypePrefix {
2022-05-13 15:19:32 +02:00
r . URL . Path += "/prometheus"
2021-06-09 11:20:38 +02:00
}
2022-05-13 15:19:32 +02:00
if ! * disablePathAppend {
r . URL . Path += "/api/v1/query_range"
2021-10-18 09:24:52 +02:00
}
2021-06-09 11:20:38 +02:00
q := r . URL . Query ( )
2023-07-07 10:39:25 +02:00
q . Add ( "start" , start . Format ( time . RFC3339 ) )
q . Add ( "end" , end . Format ( time . RFC3339 ) )
2022-12-01 13:57:53 +01:00
if s . evaluationInterval > 0 { // set step as evaluationInterval by default
// always convert to seconds to keep compatibility with older
// Prometheus versions. See https://github.com/VictoriaMetrics/VictoriaMetrics/issues/1943
q . Set ( "step" , fmt . Sprintf ( "%ds" , int ( s . evaluationInterval . Seconds ( ) ) ) )
}
2021-06-09 11:20:38 +02:00
r . URL . RawQuery = q . Encode ( )
2024-10-29 16:30:39 +01:00
s . setReqParams ( r , query )
2021-06-09 11:20:38 +02:00
}