mockster/prometheus can output data in the Prometheus format, which consists of values that are repeatably generatable for the provided query and timerange inputs. The data output by mockster/prometheus does not represent reality in any way, and is only useful for unit testing and integration testing, by providing a synthesized Prometheus environment that outputs meaningless data. None of mockster/prometheus's result sets are stored on or retrieved from disk, and are calculated just-in-time on every request, using simple mathematical computations. In Trickster, we use mockster/prometheus to conduct end-to-end testing of our DeltaProxyCache during unit testing, without requiring a real Prometheus server.
/prometheus/api/v1/query
(Instantaneous)/prometheus/api/v1/query_range
(Time Series)
mockster/byterange only uses builtin golang packages and should thus work out-of-the-box without any other dependency concerns.
package mypackage
import (
"io/ioutil"
"net/http"
"testing"
"github.com/trickstercache/mockster/pkg/testutil"
)
func TestMocksterPromethues(t *testing.T) {
ts := testutil.NewTestServer()
client := &http.Client{}
const expected = `{"status":"success","data":{"resultType":"matrix","result":[{"metric":{"random_label":"57","series_count":"1","series_id":"0"},"values":[[2,"93"]]}]}}`
resp, err := client.Get(ts.URL + `/prometheus/api/v1/query_range?query=my_test_query{random_label="57",series_count="1"}&start=2&end=2&step=15`)
if err != nil {
t.Error(err)
}
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
t.Error(err)
}
if string(body) != expected {
t.Errorf("expected [%s] got [%s]", expected, string(body))
}
}
mockster/prometheus's behavior can be modified in several ways, on a per-query basis, to produce a desired behavior. This is done by providing specific query label values as part of your test queries. All modifier labels are optional, and can be used together in any possible combination without conflict. Providing the same modifier label more than once in a query will result in the last instance of the modifier to be used when constructing the response values.
By default, mockster/prometheus will only return a single series in the result set. You can provide a label of series_count
to indicate the exact number of series that should be returned.
Example query that returns 3 series: query=my_test_query{series_count="3"}&start=2&end=2&step=15
By default, mockster/prometheus uses a "repeatable number generator" to output data. Under the hood, it works by re-seeding math.Rand
with a hashed value for the provided query string and the timestamp for which a value is needed, and returning the first value from the generator after seeding.
You can provide a line_pattern
label to utilize other supported number generators. The options are repeatable_random
(default, described above) and usage_curve
.
usage_curve
will return numbers that follow a simulated usage curve pattern (rising in the afternoon, peaking in the evening, troughing overnight).
Example using the usage_curve line pattern: query=my_test_query{series_count="3",max_value="250",min_value="10",line_pattern="usage_curve"}&start=2&end=2&step=15
mockster/prometheus is capable of simulating latency by accepting 2 optional query labels: latency_ms
and range_latency_ms
. Both labels can be used in conjunction to produce a desired effect.
The latency_ms
label introduces an upfront static processing latency of the provided duration on each http response. This is useful in simulating roundtrip wire latency.
Example adding 300ms of upfront latency: query=my_test_query{latency_ms="300"}&start=2&end=2&step=15
The range_latency_ms
label produces a per-unique-value latency effect. The result is that the response from mockster/prometheus will be delayed by a certain amount, depending upon on the number of series, size of desired timerange and step value. This is useful in simulating very broad label scopes that slow down query response times in the real world.
Example adding 5ms of range latency: query=my_test_query{range_latency_ms="5",series_count="2"}&start=0&end=1800&step=15
. In this example, 1.2s of total latency is introduced (120 datapoints x 2 series x 5ms) into the HTTP response.
The min_value
and max_value
labels allow you to define the extent of possible values returned by mockster/prometheus in the result set, and are fairly straightforward. The default min and max values, when not customized, are 0 and 100, respectively.
Example of min and max: query=my_test_query{series_count="2",min_value="32",max_value="212"}&start=0&end=90&step=15
. In this case, the returned values will be between 32 and 212, rather than 0 and 100.
The status_code
label will cause mockster/prometheus to return the provided status code instead of 200 OK
. This is useful for testing simulated failcases such as invalid query parameters.
Example query that returns 400 Bad Request: query=my_test_query{status_code="400"}&start=2&end=2&step=15
The invalid_response_body
label, when provided and set to a value other than 0, will cause mockster/prometheus to return a response that cannot be deserialized into a Prometheus Matrix or Vector object, which is again useful for testing failure handling within your app.
Example query that returns invalid response: query=my_test_query{invalid_response_body="1"}&start=2&end=2&step=15