-
Notifications
You must be signed in to change notification settings - Fork 810
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add protobuf codec for query range and instant query responses (#5527)
Co-authored-by: Ahmed Hassan <[email protected]>
- Loading branch information
Showing
23 changed files
with
2,124 additions
and
258 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,161 @@ | ||
package codec | ||
|
||
import ( | ||
"github.com/gogo/protobuf/proto" | ||
jsoniter "github.com/json-iterator/go" | ||
"github.com/prometheus/common/model" | ||
"github.com/prometheus/prometheus/promql" | ||
"github.com/prometheus/prometheus/util/stats" | ||
v1 "github.com/prometheus/prometheus/web/api/v1" | ||
|
||
"github.com/cortexproject/cortex/pkg/cortexpb" | ||
"github.com/cortexproject/cortex/pkg/querier/tripperware" | ||
) | ||
|
||
type ProtobufCodec struct{} | ||
|
||
func (p ProtobufCodec) ContentType() v1.MIMEType { | ||
return v1.MIMEType{Type: "application", SubType: "x-protobuf"} | ||
} | ||
|
||
func (p ProtobufCodec) CanEncode(resp *v1.Response) bool { | ||
// Errors are parsed by default json codec | ||
if resp.Error != "" || resp.Data == nil { | ||
return false | ||
} | ||
return true | ||
} | ||
|
||
func (p ProtobufCodec) Encode(resp *v1.Response) ([]byte, error) { | ||
prometheusQueryResponse, err := createPrometheusQueryResponse(resp) | ||
if err != nil { | ||
return []byte{}, err | ||
} | ||
b, err := proto.Marshal(prometheusQueryResponse) | ||
return b, err | ||
} | ||
|
||
func createPrometheusQueryResponse(resp *v1.Response) (*tripperware.PrometheusResponse, error) { | ||
var data = resp.Data.(*v1.QueryData) | ||
|
||
var queryResult tripperware.PrometheusQueryResult | ||
switch string(data.ResultType) { | ||
case model.ValMatrix.String(): | ||
queryResult.Result = &tripperware.PrometheusQueryResult_Matrix{ | ||
Matrix: &tripperware.Matrix{ | ||
SampleStreams: *getMatrixSampleStreams(data), | ||
}, | ||
} | ||
case model.ValVector.String(): | ||
queryResult.Result = &tripperware.PrometheusQueryResult_Vector{ | ||
Vector: &tripperware.Vector{ | ||
Samples: *getVectorSamples(data), | ||
}, | ||
} | ||
default: | ||
json := jsoniter.ConfigCompatibleWithStandardLibrary | ||
rawBytes, err := json.Marshal(data) | ||
if err != nil { | ||
return nil, err | ||
} | ||
queryResult.Result = &tripperware.PrometheusQueryResult_RawBytes{RawBytes: rawBytes} | ||
} | ||
|
||
var stats *tripperware.PrometheusResponseStats | ||
if data.Stats != nil { | ||
builtin := data.Stats.Builtin() | ||
stats = &tripperware.PrometheusResponseStats{Samples: getStats(&builtin)} | ||
} | ||
|
||
return &tripperware.PrometheusResponse{ | ||
Status: string(resp.Status), | ||
Data: tripperware.PrometheusData{ | ||
ResultType: string(data.ResultType), | ||
Result: queryResult, | ||
Stats: stats, | ||
}, | ||
ErrorType: string(resp.ErrorType), | ||
Error: resp.Error, | ||
Warnings: resp.Warnings, | ||
}, nil | ||
} | ||
|
||
func getMatrixSampleStreams(data *v1.QueryData) *[]tripperware.SampleStream { | ||
sampleStreamsLen := len(data.Result.(promql.Matrix)) | ||
sampleStreams := make([]tripperware.SampleStream, sampleStreamsLen) | ||
|
||
for i := 0; i < sampleStreamsLen; i++ { | ||
labelsLen := len(data.Result.(promql.Matrix)[i].Metric) | ||
var labels []cortexpb.LabelAdapter | ||
if labelsLen > 0 { | ||
labels = make([]cortexpb.LabelAdapter, labelsLen) | ||
for j := 0; j < labelsLen; j++ { | ||
labels[j] = cortexpb.LabelAdapter{ | ||
Name: data.Result.(promql.Matrix)[i].Metric[j].Name, | ||
Value: data.Result.(promql.Matrix)[i].Metric[j].Value, | ||
} | ||
} | ||
} | ||
|
||
samplesLen := len(data.Result.(promql.Matrix)[i].Floats) | ||
var samples []cortexpb.Sample | ||
if samplesLen > 0 { | ||
samples = make([]cortexpb.Sample, samplesLen) | ||
for j := 0; j < samplesLen; j++ { | ||
samples[j] = cortexpb.Sample{ | ||
Value: data.Result.(promql.Matrix)[i].Floats[j].F, | ||
TimestampMs: data.Result.(promql.Matrix)[i].Floats[j].T, | ||
} | ||
} | ||
} | ||
sampleStreams[i] = tripperware.SampleStream{Labels: labels, Samples: samples} | ||
} | ||
return &sampleStreams | ||
} | ||
|
||
func getVectorSamples(data *v1.QueryData) *[]tripperware.Sample { | ||
vectorSamplesLen := len(data.Result.(promql.Vector)) | ||
vectorSamples := make([]tripperware.Sample, vectorSamplesLen) | ||
|
||
for i := 0; i < vectorSamplesLen; i++ { | ||
labelsLen := len(data.Result.(promql.Vector)[i].Metric) | ||
var labels []cortexpb.LabelAdapter | ||
if labelsLen > 0 { | ||
labels = make([]cortexpb.LabelAdapter, labelsLen) | ||
for j := 0; j < labelsLen; j++ { | ||
labels[j] = cortexpb.LabelAdapter{ | ||
Name: data.Result.(promql.Vector)[i].Metric[j].Name, | ||
Value: data.Result.(promql.Vector)[i].Metric[j].Value, | ||
} | ||
} | ||
} | ||
|
||
vectorSamples[i] = tripperware.Sample{ | ||
Labels: labels, | ||
Sample: &cortexpb.Sample{ | ||
TimestampMs: data.Result.(promql.Vector)[i].T, | ||
Value: data.Result.(promql.Vector)[i].F, | ||
}, | ||
} | ||
} | ||
return &vectorSamples | ||
} | ||
|
||
func getStats(builtin *stats.BuiltinStats) *tripperware.PrometheusResponseSamplesStats { | ||
queryableSamplesStatsPerStepLen := len(builtin.Samples.TotalQueryableSamplesPerStep) | ||
queryableSamplesStatsPerStep := make([]*tripperware.PrometheusResponseQueryableSamplesStatsPerStep, queryableSamplesStatsPerStepLen) | ||
for i := 0; i < queryableSamplesStatsPerStepLen; i++ { | ||
queryableSamplesStatsPerStep[i] = &tripperware.PrometheusResponseQueryableSamplesStatsPerStep{ | ||
Value: builtin.Samples.TotalQueryableSamplesPerStep[i].V, | ||
TimestampMs: builtin.Samples.TotalQueryableSamplesPerStep[i].T, | ||
} | ||
} | ||
|
||
statSamples := tripperware.PrometheusResponseSamplesStats{ | ||
TotalQueryableSamples: builtin.Samples.TotalQueryableSamples, | ||
TotalQueryableSamplesPerStep: queryableSamplesStatsPerStep, | ||
PeakSamples: int64(builtin.Samples.PeakSamples), | ||
} | ||
|
||
return &statSamples | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.