2024-10-30 14:22:06 +00:00
|
|
|
package apptest
|
|
|
|
|
|
|
|
import (
|
2024-11-20 15:30:55 +00:00
|
|
|
"fmt"
|
2024-10-30 14:22:06 +00:00
|
|
|
"testing"
|
2024-11-20 15:30:55 +00:00
|
|
|
"time"
|
2024-10-30 14:22:06 +00:00
|
|
|
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/fs"
|
2024-11-20 15:30:55 +00:00
|
|
|
"github.com/google/go-cmp/cmp"
|
2024-10-30 14:22:06 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// TestCase holds the state and defines clean-up procedure common for all test
|
|
|
|
// cases.
|
|
|
|
type TestCase struct {
|
|
|
|
t *testing.T
|
|
|
|
cli *Client
|
2024-11-08 13:49:00 +00:00
|
|
|
|
|
|
|
startedApps []Stopper
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stopper is an interface of objects that needs to be stopped via Stop() call
|
|
|
|
type Stopper interface {
|
|
|
|
Stop()
|
2024-10-30 14:22:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewTestCase creates a new test case.
|
|
|
|
func NewTestCase(t *testing.T) *TestCase {
|
2024-11-08 13:49:00 +00:00
|
|
|
return &TestCase{t, NewClient(), nil}
|
2024-10-30 14:22:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Dir returns the directory name that should be used by as the -storageDataDir.
|
|
|
|
func (tc *TestCase) Dir() string {
|
|
|
|
return tc.t.Name()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Client returns an instance of the client that can be used for interacting with
|
|
|
|
// the app(s) under test.
|
|
|
|
func (tc *TestCase) Client() *Client {
|
|
|
|
return tc.cli
|
|
|
|
}
|
|
|
|
|
2024-11-08 13:49:00 +00:00
|
|
|
// Stop performs the test case clean up, such as closing all client connections
|
2024-10-30 14:22:06 +00:00
|
|
|
// and removing the -storageDataDir directory.
|
|
|
|
//
|
|
|
|
// Note that the -storageDataDir is not removed in case of test case failure to
|
2024-11-08 13:49:00 +00:00
|
|
|
// allow for further manual debugging.
|
|
|
|
func (tc *TestCase) Stop() {
|
2024-10-30 14:22:06 +00:00
|
|
|
tc.cli.CloseConnections()
|
2024-11-08 13:49:00 +00:00
|
|
|
for _, app := range tc.startedApps {
|
|
|
|
app.Stop()
|
|
|
|
}
|
2024-10-30 14:22:06 +00:00
|
|
|
if !tc.t.Failed() {
|
|
|
|
fs.MustRemoveAll(tc.Dir())
|
|
|
|
}
|
|
|
|
}
|
2024-11-08 13:49:00 +00:00
|
|
|
|
|
|
|
// MustStartVmsingle is a test helper function that starts an instance of
|
|
|
|
// vmsingle and fails the test if the app fails to start.
|
|
|
|
func (tc *TestCase) MustStartVmsingle(instance string, flags []string) *Vmsingle {
|
|
|
|
tc.t.Helper()
|
|
|
|
|
|
|
|
app, err := StartVmsingle(instance, flags, tc.cli)
|
|
|
|
if err != nil {
|
|
|
|
tc.t.Fatalf("Could not start %s: %v", instance, err)
|
|
|
|
}
|
|
|
|
tc.addApp(app)
|
|
|
|
return app
|
|
|
|
}
|
|
|
|
|
|
|
|
// MustStartVmstorage is a test helper function that starts an instance of
|
|
|
|
// vmstorage and fails the test if the app fails to start.
|
|
|
|
func (tc *TestCase) MustStartVmstorage(instance string, flags []string) *Vmstorage {
|
|
|
|
tc.t.Helper()
|
|
|
|
|
|
|
|
app, err := StartVmstorage(instance, flags, tc.cli)
|
|
|
|
if err != nil {
|
|
|
|
tc.t.Fatalf("Could not start %s: %v", instance, err)
|
|
|
|
}
|
|
|
|
tc.addApp(app)
|
|
|
|
return app
|
|
|
|
}
|
|
|
|
|
|
|
|
// MustStartVmselect is a test helper function that starts an instance of
|
|
|
|
// vmselect and fails the test if the app fails to start.
|
|
|
|
func (tc *TestCase) MustStartVmselect(instance string, flags []string) *Vmselect {
|
|
|
|
tc.t.Helper()
|
|
|
|
|
|
|
|
app, err := StartVmselect(instance, flags, tc.cli)
|
|
|
|
if err != nil {
|
|
|
|
tc.t.Fatalf("Could not start %s: %v", instance, err)
|
|
|
|
}
|
|
|
|
tc.addApp(app)
|
|
|
|
return app
|
|
|
|
}
|
|
|
|
|
|
|
|
// MustStartVminsert is a test helper function that starts an instance of
|
|
|
|
// vminsert and fails the test if the app fails to start.
|
|
|
|
func (tc *TestCase) MustStartVminsert(instance string, flags []string) *Vminsert {
|
|
|
|
tc.t.Helper()
|
|
|
|
|
|
|
|
app, err := StartVminsert(instance, flags, tc.cli)
|
|
|
|
if err != nil {
|
|
|
|
tc.t.Fatalf("Could not start %s: %v", instance, err)
|
|
|
|
}
|
|
|
|
tc.addApp(app)
|
|
|
|
return app
|
|
|
|
}
|
|
|
|
|
2024-11-20 15:30:55 +00:00
|
|
|
type vmcluster struct {
|
|
|
|
*Vminsert
|
|
|
|
*Vmselect
|
|
|
|
vmstorages []*Vmstorage
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *vmcluster) ForceFlush(t *testing.T) {
|
|
|
|
for _, s := range c.vmstorages {
|
|
|
|
s.ForceFlush(t)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MustStartCluster is a typical cluster configuration.
|
|
|
|
//
|
|
|
|
// The cluster consists of two vmstorages, one vminsert and one vmselect, no
|
|
|
|
// data replication.
|
|
|
|
//
|
|
|
|
// Such configuration is suitable for tests that don't verify the
|
|
|
|
// cluster-specific behavior (such as sharding, replication, or multilevel
|
|
|
|
// vmselect) but instead just need a typical cluster configuration to verify
|
|
|
|
// some business logic (such as API surface, or MetricsQL). Such cluster
|
|
|
|
// tests usually come paired with corresponding vmsingle tests.
|
|
|
|
func (tc *TestCase) MustStartCluster() PrometheusWriteQuerier {
|
|
|
|
tc.t.Helper()
|
|
|
|
|
|
|
|
vmstorage1 := tc.MustStartVmstorage("vmstorage-1", []string{
|
|
|
|
"-storageDataPath=" + tc.Dir() + "/vmstorage-1",
|
|
|
|
"-retentionPeriod=100y",
|
|
|
|
})
|
|
|
|
vmstorage2 := tc.MustStartVmstorage("vmstorage-2", []string{
|
|
|
|
"-storageDataPath=" + tc.Dir() + "/vmstorage-2",
|
|
|
|
"-retentionPeriod=100y",
|
|
|
|
})
|
|
|
|
vminsert := tc.MustStartVminsert("vminsert", []string{
|
|
|
|
"-storageNode=" + vmstorage1.VminsertAddr() + "," + vmstorage2.VminsertAddr(),
|
|
|
|
})
|
|
|
|
vmselect := tc.MustStartVmselect("vmselect", []string{
|
|
|
|
"-storageNode=" + vmstorage1.VmselectAddr() + "," + vmstorage2.VmselectAddr(),
|
|
|
|
})
|
|
|
|
|
|
|
|
return &vmcluster{vminsert, vmselect, []*Vmstorage{vmstorage1, vmstorage2}}
|
|
|
|
}
|
|
|
|
|
2024-11-08 13:49:00 +00:00
|
|
|
func (tc *TestCase) addApp(app Stopper) {
|
|
|
|
tc.startedApps = append(tc.startedApps, app)
|
|
|
|
}
|
2024-11-20 15:30:55 +00:00
|
|
|
|
|
|
|
// AssertOptions hold the assertion params, such as got and wanted values as
|
|
|
|
// well as the message that should be included into the assertion error message
|
|
|
|
// in case of failure.
|
|
|
|
//
|
|
|
|
// In VictoriaMetrics (especially the cluster version) the inserted data does
|
|
|
|
// not become visible for querying right away. Therefore, the first comparisons
|
|
|
|
// may fail. AssertOptions allow to configure how many times the actual result
|
|
|
|
// must be retrieved and compared with the expected one and for long to wait
|
|
|
|
// between the retries. If these two params (`Retries` and `Period`) are not
|
|
|
|
// set, the default values will be used.
|
|
|
|
//
|
|
|
|
// If it is known that the data is available, then the retry functionality can
|
|
|
|
// be disabled by setting the `DoNotRetry` field.
|
|
|
|
//
|
|
|
|
// AssertOptions are used by the TestCase.Assert() method, and this method uses
|
|
|
|
// cmp.Diff() from go-cmp package for comparing got and wanted values.
|
|
|
|
// AssertOptions, therefore, allows to pass cmp.Options to cmp.Diff() via
|
|
|
|
// `CmpOpts` field.
|
|
|
|
//
|
|
|
|
// Finally the `FailNow` field controls whether the assertion should fail using
|
|
|
|
// `testing.T.Errorf()` or `testing.T.Fatalf()`.
|
|
|
|
type AssertOptions struct {
|
|
|
|
Msg string
|
|
|
|
Got func() any
|
|
|
|
Want any
|
|
|
|
CmpOpts []cmp.Option
|
|
|
|
DoNotRetry bool
|
|
|
|
Retries int
|
|
|
|
Period time.Duration
|
|
|
|
FailNow bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assert compares the actual result with the expected one possibly multiple
|
|
|
|
// times in order to account for the fact that the inserted data does not become
|
|
|
|
// available for querying right away (especially in cluster version of
|
|
|
|
// VictoriaMetrics).
|
|
|
|
func (tc *TestCase) Assert(opts *AssertOptions) {
|
|
|
|
tc.t.Helper()
|
|
|
|
|
|
|
|
const (
|
|
|
|
defaultRetries = 20
|
|
|
|
defaultPeriod = 100 * time.Millisecond
|
|
|
|
)
|
|
|
|
|
|
|
|
if opts.DoNotRetry {
|
|
|
|
opts.Retries = 1
|
|
|
|
opts.Period = 0
|
|
|
|
} else {
|
|
|
|
if opts.Retries <= 0 {
|
|
|
|
opts.Retries = defaultRetries
|
|
|
|
}
|
|
|
|
if opts.Period <= 0 {
|
|
|
|
opts.Period = defaultPeriod
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var diff string
|
|
|
|
|
|
|
|
for range opts.Retries {
|
|
|
|
diff = cmp.Diff(opts.Want, opts.Got(), opts.CmpOpts...)
|
|
|
|
if diff == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
time.Sleep(opts.Period)
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := fmt.Sprintf("%s (-want, +got):\n%s", opts.Msg, diff)
|
|
|
|
|
|
|
|
if opts.FailNow {
|
|
|
|
tc.t.Fatal(msg)
|
|
|
|
} else {
|
|
|
|
tc.t.Error(msg)
|
|
|
|
}
|
|
|
|
}
|