package apptest import ( "fmt" "net/http" "net/url" "os" "regexp" "strings" "testing" "time" ) // Vmsingle holds the state of a vmsingle app and provides vmsingle-specific // functions. type Vmsingle struct { *app *ServesMetrics storageDataPath string httpListenAddr string forceFlushURL string prometheusAPIV1ImportPrometheusURL string prometheusAPIV1QueryURL string prometheusAPIV1QueryRangeURL string prometheusAPIV1SeriesURL string } // MustStartVmsingle is a test helper function that starts an instance of // vmsingle and fails the test if the app fails to start. func MustStartVmsingle(t *testing.T, instance string, flags []string, cli *Client) *Vmsingle { t.Helper() app, err := StartVmsingle(instance, flags, cli) if err != nil { t.Fatalf("Could not start %s: %v", instance, err) } return app } // StartVmsingle starts an instance of vmsingle 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 StartVmsingle(instance string, flags []string, cli *Client) (*Vmsingle, error) { app, stderrExtracts, err := startApp(instance, "../../bin/victoria-metrics", flags, &appOptions{ defaultFlags: map[string]string{ "-storageDataPath": fmt.Sprintf("%s/%s-%d", os.TempDir(), instance, time.Now().UnixNano()), "-httpListenAddr": "127.0.0.1:0", }, extractREs: []*regexp.Regexp{ storageDataPathRE, httpListenAddrRE, }, }) if err != nil { return nil, err } return &Vmsingle{ app: app, ServesMetrics: &ServesMetrics{ metricsURL: fmt.Sprintf("http://%s/metrics", stderrExtracts[1]), cli: cli, }, storageDataPath: stderrExtracts[0], httpListenAddr: stderrExtracts[1], forceFlushURL: fmt.Sprintf("http://%s/internal/force_flush", stderrExtracts[1]), prometheusAPIV1ImportPrometheusURL: fmt.Sprintf("http://%s/prometheus/api/v1/import/prometheus", stderrExtracts[1]), prometheusAPIV1QueryURL: fmt.Sprintf("http://%s/prometheus/api/v1/query", stderrExtracts[1]), prometheusAPIV1QueryRangeURL: fmt.Sprintf("http://%s/prometheus/api/v1/query_range", stderrExtracts[1]), prometheusAPIV1SeriesURL: fmt.Sprintf("http://%s/prometheus/api/v1/series", stderrExtracts[1]), }, nil } // ForceFlush is a test helper function that forces the flushing of insterted // data so it becomes available for searching immediately. func (app *Vmsingle) ForceFlush(t *testing.T) { t.Helper() app.cli.Get(t, app.forceFlushURL, http.StatusOK) } // PrometheusAPIV1ImportPrometheus is a test helper function that inserts a // collection of records in Prometheus text exposition format by sending a HTTP // POST request to /prometheus/api/v1/import/prometheus vmsingle endpoint. // // See https://docs.victoriametrics.com/url-examples/#apiv1importprometheus func (app *Vmsingle) PrometheusAPIV1ImportPrometheus(t *testing.T, records []string, _ QueryOpts) { t.Helper() app.cli.Post(t, app.prometheusAPIV1ImportPrometheusURL, "text/plain", strings.Join(records, "\n"), http.StatusNoContent) } // PrometheusAPIV1Query is a test helper function that performs PromQL/MetricsQL // instant query by sending a HTTP POST request to /prometheus/api/v1/query // vmsingle endpoint. // // See https://docs.victoriametrics.com/url-examples/#apiv1query func (app *Vmsingle) PrometheusAPIV1Query(t *testing.T, query, time, step string, opts QueryOpts) *PrometheusAPIV1QueryResponse { t.Helper() values := url.Values{} values.Add("query", query) values.Add("time", time) values.Add("step", step) values.Add("timeout", opts.Timeout) res := app.cli.PostForm(t, app.prometheusAPIV1QueryURL, values, http.StatusOK) return NewPrometheusAPIV1QueryResponse(t, res) } // PrometheusAPIV1QueryRange is a test helper function that performs // PromQL/MetricsQL range query by sending a HTTP POST request to // /prometheus/api/v1/query_range vmsingle endpoint. // // See https://docs.victoriametrics.com/url-examples/#apiv1query_range func (app *Vmsingle) PrometheusAPIV1QueryRange(t *testing.T, query, start, end, step string, opts QueryOpts) *PrometheusAPIV1QueryResponse { t.Helper() values := url.Values{} values.Add("query", query) values.Add("start", start) values.Add("end", end) values.Add("step", step) values.Add("timeout", opts.Timeout) res := app.cli.PostForm(t, app.prometheusAPIV1QueryRangeURL, values, http.StatusOK) return NewPrometheusAPIV1QueryResponse(t, res) } // PrometheusAPIV1Series sends a query to a /prometheus/api/v1/series endpoint // and returns the list of time series that match the query. // // See https://docs.victoriametrics.com/url-examples/#apiv1series func (app *Vmsingle) PrometheusAPIV1Series(t *testing.T, matchQuery string, _ QueryOpts) *PrometheusAPIV1SeriesResponse { t.Helper() values := url.Values{} values.Add("match[]", matchQuery) res := app.cli.PostForm(t, app.prometheusAPIV1SeriesURL, values, http.StatusOK) return NewPrometheusAPIV1SeriesResponse(t, res) } // String returns the string representation of the vmsingle app state. func (app *Vmsingle) String() string { return fmt.Sprintf("{app: %s storageDataPath: %q httpListenAddr: %q}", []any{ app.app, app.storageDataPath, app.httpListenAddr}...) }