mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2026-05-17 00:26:36 +03:00
This change allows specifying headers for provided API calls. This ability is required for proper testing of Tenant-via-Header feature in https://github.com/VictoriaMetrics/VictoriaMetrics/pull/10782 Signed-off-by: hagen1778 <roman@victoriametrics.com>
357 lines
12 KiB
Go
357 lines
12 KiB
Go
package apptest
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"regexp"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/golang/snappy"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prommetadata"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prompb"
|
|
)
|
|
|
|
// Vminsert holds the state of a vminsert app and provides vminsert-specific
|
|
// functions.
|
|
type Vminsert struct {
|
|
*app
|
|
*ServesMetrics
|
|
|
|
httpListenAddr string
|
|
clusternativeListenAddr string
|
|
graphiteListenAddr string
|
|
openTSDBListenAddr string
|
|
|
|
cli *Client
|
|
}
|
|
|
|
// storageNodes returns the storage node addresses passed to vminsert via
|
|
// -storageNode command line flag.
|
|
func storageNodes(flags []string) []string {
|
|
for _, flag := range flags {
|
|
if storageNodes, found := strings.CutPrefix(flag, "-storageNode="); found {
|
|
return strings.Split(storageNodes, ",")
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// StartVminsert starts an instance of vminsert with the given flags. It also
|
|
// sets the default flags and populates the app instance state with runtime
|
|
// values extracted from the application log (such as httpListenAddr)
|
|
func StartVminsert(instance string, flags []string, cli *Client, output io.Writer) (*Vminsert, error) {
|
|
extractREs := []*regexp.Regexp{
|
|
httpListenAddrRE,
|
|
vminsertClusterNativeAddrRE,
|
|
graphiteListenAddrRE,
|
|
openTSDBListenAddrRE,
|
|
}
|
|
// Add storageNode REs to block until vminsert establishes connections with
|
|
// all storage nodes. The extracted values are unused.
|
|
for _, sn := range storageNodes(flags) {
|
|
logRecord := fmt.Sprintf("successfully dialed -storageNode=\"%s\"", sn)
|
|
extractREs = append(extractREs, regexp.MustCompile(logRecord))
|
|
}
|
|
|
|
app, stderrExtracts, err := startApp(instance, "../../bin/vminsert-race", flags, &appOptions{
|
|
defaultFlags: map[string]string{
|
|
"-httpListenAddr": "127.0.0.1:0",
|
|
"-clusternativeListenAddr": "127.0.0.1:0",
|
|
"-graphiteListenAddr": ":0",
|
|
"-opentsdbListenAddr": "127.0.0.1:0",
|
|
"-clusternative.vminsertConnsShutdownDuration": "1ms",
|
|
},
|
|
extractREs: extractREs,
|
|
output: output,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &Vminsert{
|
|
app: app,
|
|
ServesMetrics: &ServesMetrics{
|
|
metricsURL: fmt.Sprintf("http://%s/metrics", stderrExtracts[0]),
|
|
cli: cli,
|
|
},
|
|
httpListenAddr: stderrExtracts[0],
|
|
clusternativeListenAddr: stderrExtracts[1],
|
|
graphiteListenAddr: stderrExtracts[2],
|
|
openTSDBListenAddr: stderrExtracts[3],
|
|
cli: cli,
|
|
}, nil
|
|
}
|
|
|
|
// ClusternativeListenAddr returns the address at which the vminsert process is
|
|
// listening for connections from other vminsert apps.
|
|
func (app *Vminsert) ClusternativeListenAddr() string {
|
|
return app.clusternativeListenAddr
|
|
}
|
|
|
|
// HTTPAddr returns the address at which the vminsert process is
|
|
// listening for incoming HTTP requests.
|
|
func (app *Vminsert) HTTPAddr() string {
|
|
return app.httpListenAddr
|
|
}
|
|
|
|
// InfluxWrite is a test helper function that inserts a
|
|
// collection of records in Influx line format by sending a HTTP
|
|
// POST request to /influx/write vmsingle endpoint.
|
|
//
|
|
// See https://docs.victoriametrics.com/victoriametrics/url-examples/#influxwrite
|
|
func (app *Vminsert) InfluxWrite(t *testing.T, records []string, opts QueryOpts) {
|
|
t.Helper()
|
|
|
|
url := fmt.Sprintf("http://%s/insert/%s/influx/write", app.httpListenAddr, opts.Tenant)
|
|
uv := opts.asURLValues()
|
|
uvs := uv.Encode()
|
|
if len(uvs) > 0 {
|
|
url += "?" + uvs
|
|
}
|
|
|
|
data := []byte(strings.Join(records, "\n"))
|
|
headers := opts.getHeaders()
|
|
headers.Set("Content-Type", "text/plain")
|
|
app.sendBlocking(t, len(records), func() {
|
|
_, statusCode := app.cli.Post(t, url, data, headers)
|
|
if statusCode != http.StatusNoContent {
|
|
t.Fatalf("unexpected status code: got %d, want %d", statusCode, http.StatusNoContent)
|
|
}
|
|
})
|
|
}
|
|
|
|
// GraphiteWrite is a test helper function that sends a
|
|
// collection of records to graphiteListenAddr port.
|
|
//
|
|
// See https://docs.victoriametrics.com/victoriametrics/integrations/graphite/#ingesting
|
|
func (app *Vminsert) GraphiteWrite(t *testing.T, records []string, _ QueryOpts) {
|
|
t.Helper()
|
|
app.cli.Write(t, app.graphiteListenAddr, records)
|
|
}
|
|
|
|
// PrometheusAPIV1ImportCSV is a test helper function that inserts a collection
|
|
// of records in CSV format for the given tenant by sending an HTTP POST
|
|
// request to prometheus/api/v1/import/csv vminsert endpoint.
|
|
//
|
|
// See https://docs.victoriametrics.com/cluster-victoriametrics/#url-format
|
|
func (app *Vminsert) PrometheusAPIV1ImportCSV(t *testing.T, records []string, opts QueryOpts) {
|
|
t.Helper()
|
|
|
|
url := fmt.Sprintf("http://%s/insert/%s/prometheus/api/v1/import/csv", app.httpListenAddr, opts.Tenant)
|
|
uv := opts.asURLValues()
|
|
uvs := uv.Encode()
|
|
if len(uvs) > 0 {
|
|
url += "?" + uvs
|
|
}
|
|
data := []byte(strings.Join(records, "\n"))
|
|
headers := opts.getHeaders()
|
|
headers.Set("Content-Type", "text/plain")
|
|
app.sendBlocking(t, len(records), func() {
|
|
_, statusCode := app.cli.Post(t, url, data, headers)
|
|
if statusCode != http.StatusNoContent {
|
|
t.Fatalf("unexpected status code: got %d, want %d", statusCode, http.StatusNoContent)
|
|
}
|
|
})
|
|
}
|
|
|
|
// PrometheusAPIV1ImportNative is a test helper function that inserts a collection
|
|
// of records in Native format for the given tenant by sending an HTTP POST
|
|
// request to prometheus/api/v1/import/native vminsert endpoint.
|
|
//
|
|
// See https://docs.victoriametrics.com/cluster-victoriametrics/#url-format
|
|
func (app *Vminsert) PrometheusAPIV1ImportNative(t *testing.T, data []byte, opts QueryOpts) {
|
|
t.Helper()
|
|
|
|
url := fmt.Sprintf("http://%s/insert/%s/prometheus/api/v1/import/native", app.httpListenAddr, opts.Tenant)
|
|
uv := opts.asURLValues()
|
|
uvs := uv.Encode()
|
|
if len(uvs) > 0 {
|
|
url += "?" + uvs
|
|
}
|
|
headers := opts.getHeaders()
|
|
headers.Set("Content-Type", "text/plain")
|
|
app.sendBlocking(t, 1, func() {
|
|
_, statusCode := app.cli.Post(t, url, data, headers)
|
|
if statusCode != http.StatusNoContent {
|
|
t.Fatalf("unexpected status code: got %d, want %d", statusCode, http.StatusNoContent)
|
|
}
|
|
})
|
|
}
|
|
|
|
// OpenTSDBAPIPut is a test helper function that inserts a collection of
|
|
// records in OpenTSDB format for the given tenant by sending an HTTP POST
|
|
// request to /opentsdb/api/put vminsert endpoint.
|
|
//
|
|
// See https://docs.victoriametrics.com/cluster-victoriametrics/#url-format
|
|
func (app *Vminsert) OpenTSDBAPIPut(t *testing.T, records []string, opts QueryOpts) {
|
|
t.Helper()
|
|
|
|
url := fmt.Sprintf("http://%s/insert/%s/opentsdb/api/put", app.openTSDBListenAddr, opts.Tenant)
|
|
uv := opts.asURLValues()
|
|
uvs := uv.Encode()
|
|
if len(uvs) > 0 {
|
|
url += "?" + uvs
|
|
}
|
|
data := []byte("[" + strings.Join(records, ",") + "]")
|
|
headers := opts.getHeaders()
|
|
headers.Set("Content-Type", "application/json")
|
|
app.sendBlocking(t, len(records), func() {
|
|
_, statusCode := app.cli.Post(t, url, data, headers)
|
|
if statusCode != http.StatusNoContent {
|
|
t.Fatalf("unexpected status code: got %d, want %d", statusCode, http.StatusNoContent)
|
|
}
|
|
})
|
|
}
|
|
|
|
// PrometheusAPIV1Write is a test helper function that inserts a
|
|
// collection of records in Prometheus remote-write format by sending a HTTP
|
|
// POST request to /prometheus/api/v1/write vminsert endpoint.
|
|
func (app *Vminsert) PrometheusAPIV1Write(t *testing.T, wr prompb.WriteRequest, opts QueryOpts) {
|
|
t.Helper()
|
|
|
|
url := fmt.Sprintf("http://%s/insert/%s/prometheus/api/v1/write", app.httpListenAddr, opts.Tenant)
|
|
data := snappy.Encode(nil, wr.MarshalProtobuf(nil))
|
|
recordsCount := len(wr.Timeseries)
|
|
if prommetadata.IsEnabled() {
|
|
recordsCount += len(wr.Metadata)
|
|
}
|
|
headers := opts.getHeaders()
|
|
headers.Set("Content-Type", "application/x-protobuf")
|
|
app.sendBlocking(t, recordsCount, func() {
|
|
_, statusCode := app.cli.Post(t, url, data, headers)
|
|
if statusCode != http.StatusNoContent {
|
|
t.Fatalf("unexpected status code: got %d, want %d", statusCode, http.StatusNoContent)
|
|
}
|
|
})
|
|
}
|
|
|
|
// PrometheusAPIV1ImportPrometheus is a test helper function that inserts a
|
|
// collection of records in Prometheus text exposition format for the given
|
|
// tenant by sending a HTTP POST request to
|
|
// /prometheus/api/v1/import/prometheus vminsert endpoint.
|
|
//
|
|
// See https://docs.victoriametrics.com/victoriametrics/url-examples/#apiv1importprometheus
|
|
func (app *Vminsert) PrometheusAPIV1ImportPrometheus(t *testing.T, records []string, opts QueryOpts) {
|
|
t.Helper()
|
|
|
|
url := fmt.Sprintf("http://%s/insert/%s/prometheus/api/v1/import/prometheus", app.httpListenAddr, opts.Tenant)
|
|
uv := opts.asURLValues()
|
|
uvs := uv.Encode()
|
|
if len(uvs) > 0 {
|
|
url += "?" + uvs
|
|
}
|
|
data := []byte(strings.Join(records, "\n"))
|
|
var recordsCount int
|
|
var metadataRecords int
|
|
uniqueMetadataMetricNames := make(map[string]struct{})
|
|
for _, record := range records {
|
|
// metric metadata has the following format:
|
|
//# HELP importprometheus_series
|
|
//# TYPE importprometheus_series
|
|
// it results into single metadata record
|
|
if strings.HasPrefix(record, "# ") {
|
|
metadataItems := strings.Split(record, " ")
|
|
if len(metadataItems) < 2 {
|
|
t.Fatalf("BUG: unexpected metadata format=%q", record)
|
|
}
|
|
metricName := metadataItems[2]
|
|
if _, ok := uniqueMetadataMetricNames[metricName]; ok {
|
|
continue
|
|
}
|
|
uniqueMetadataMetricNames[metricName] = struct{}{}
|
|
metadataRecords++
|
|
continue
|
|
}
|
|
recordsCount++
|
|
}
|
|
if prommetadata.IsEnabled() {
|
|
recordsCount += metadataRecords
|
|
}
|
|
headers := opts.getHeaders()
|
|
headers.Set("Content-Type", "text/plain")
|
|
app.sendBlocking(t, recordsCount, func() {
|
|
_, statusCode := app.cli.Post(t, url, data, headers)
|
|
if statusCode != http.StatusNoContent {
|
|
t.Fatalf("unexpected status code: got %d, want %d", statusCode, http.StatusNoContent)
|
|
}
|
|
})
|
|
}
|
|
|
|
// ZabbixConnectorHistory is a test helper function that inserts a
|
|
// collection of records in zabbixconnector format by sending a HTTP
|
|
// POST request to /zabbixconnector/api/v1/history vmsingle endpoint.
|
|
func (app *Vminsert) ZabbixConnectorHistory(t *testing.T, records []string, opts QueryOpts) {
|
|
t.Helper()
|
|
|
|
url := fmt.Sprintf("http://%s/insert/%s/zabbixconnector/api/v1/history", app.httpListenAddr, opts.Tenant)
|
|
uv := opts.asURLValues()
|
|
uvs := uv.Encode()
|
|
if len(uvs) > 0 {
|
|
url += "?" + uvs
|
|
}
|
|
data := []byte(strings.Join(records, "\n"))
|
|
headers := opts.getHeaders()
|
|
headers.Set("Content-Type", "application/json")
|
|
app.sendBlocking(t, len(records), func() {
|
|
_, statusCode := app.cli.Post(t, url, data, headers)
|
|
if statusCode != http.StatusOK {
|
|
t.Fatalf("unexpected status code: got %d, want %d", statusCode, http.StatusOK)
|
|
}
|
|
})
|
|
|
|
}
|
|
|
|
// String returns the string representation of the vminsert app state.
|
|
func (app *Vminsert) String() string {
|
|
return fmt.Sprintf("{app: %s httpListenAddr: %q}", app.app, app.httpListenAddr)
|
|
}
|
|
|
|
// sendBlocking sends the data to vmstorage by executing `send` function and
|
|
// waits until the data is actually sent.
|
|
//
|
|
// vminsert does not send the data immediately. It first puts the data into a
|
|
// buffer. Then a background goroutine takes the data from the buffer sends it
|
|
// to the vmstorage. This happens every 200ms.
|
|
//
|
|
// Waiting is implemented a retrieving the value of `vm_rpc_rows_sent_total`
|
|
// metric and checking whether it is equal or greater than the wanted value.
|
|
// If it is, then the data has been sent to vmstorage.
|
|
//
|
|
// Unreliable if the records are inserted concurrently.
|
|
// TODO(rtm0): Put sending and waiting into a critical section to make reliable?
|
|
func (app *Vminsert) sendBlocking(t *testing.T, numRecordsToSend int, send func()) {
|
|
t.Helper()
|
|
|
|
wantRowsSentCount := app.rpcRowsSentTotal(t) + numRecordsToSend
|
|
|
|
send()
|
|
|
|
const (
|
|
retries = 20
|
|
period = 100 * time.Millisecond
|
|
)
|
|
for range retries {
|
|
d := app.rpcRowsSentTotal(t)
|
|
if d >= wantRowsSentCount {
|
|
return
|
|
}
|
|
time.Sleep(period)
|
|
}
|
|
t.Fatalf("timed out while waiting for inserted rows to be sent to vmstorage")
|
|
}
|
|
|
|
// rpcRowsSentTotal retrieves the values of all vminsert
|
|
// `vm_rpc_rows_sent_total` metrics (there will be one for each vmstorage) and
|
|
// returns their integer sum.
|
|
func (app *Vminsert) rpcRowsSentTotal(t *testing.T) int {
|
|
total := 0.0
|
|
for _, v := range app.GetMetricsByPrefix(t, "vm_rpc_rows_sent_total") {
|
|
total += v
|
|
}
|
|
return int(total)
|
|
}
|