2021-01-19 03:02:26 +00:00
|
|
|
package tsm1
|
|
|
|
|
|
|
|
import (
|
2021-06-09 15:03:53 +00:00
|
|
|
"context"
|
2021-01-19 03:02:26 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/influxdata/influxdb/v2/models"
|
|
|
|
"github.com/influxdata/influxdb/v2/tsdb"
|
|
|
|
"github.com/stretchr/testify/require"
|
2021-02-11 15:12:39 +00:00
|
|
|
"go.uber.org/zap/zaptest"
|
2021-01-19 03:02:26 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestEngine_ConcurrentShardSnapshots(t *testing.T) {
|
|
|
|
tmpDir, err := ioutil.TempDir("", "shard_test")
|
|
|
|
require.NoError(t, err, "error creating temporary directory")
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
|
|
|
|
tmpShard := filepath.Join(tmpDir, "shard")
|
|
|
|
tmpWal := filepath.Join(tmpDir, "wal")
|
|
|
|
|
2021-02-11 15:12:39 +00:00
|
|
|
sfile := NewSeriesFile(t, tmpDir)
|
2021-01-19 03:02:26 +00:00
|
|
|
defer sfile.Close()
|
|
|
|
|
|
|
|
opts := tsdb.NewEngineOptions()
|
|
|
|
opts.Config.WALDir = filepath.Join(tmpDir, "wal")
|
|
|
|
opts.SeriesIDSets = seriesIDSets([]*tsdb.SeriesIDSet{})
|
|
|
|
|
|
|
|
sh := tsdb.NewShard(1, tmpShard, tmpWal, sfile, opts)
|
2021-06-09 15:03:53 +00:00
|
|
|
require.NoError(t, sh.Open(context.Background()), "error opening shard")
|
2021-01-19 03:02:26 +00:00
|
|
|
defer sh.Close()
|
|
|
|
|
|
|
|
points := make([]models.Point, 0, 10000)
|
|
|
|
for i := 0; i < cap(points); i++ {
|
|
|
|
points = append(points, models.MustNewPoint(
|
|
|
|
"cpu",
|
|
|
|
models.NewTags(map[string]string{"host": "server"}),
|
|
|
|
map[string]interface{}{"value": 1.0},
|
|
|
|
time.Unix(int64(i), 0),
|
|
|
|
))
|
|
|
|
}
|
2021-06-09 15:03:53 +00:00
|
|
|
err = sh.WritePoints(context.Background(), points)
|
2021-01-19 03:02:26 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
engineInterface, err := sh.Engine()
|
|
|
|
require.NoError(t, err, "error retrieving shard engine")
|
|
|
|
|
|
|
|
// Get the struct underlying the interface. Not a recommended practice.
|
|
|
|
realEngineStruct, ok := (engineInterface).(*Engine)
|
|
|
|
if !ok {
|
|
|
|
t.Log("Engine type does not permit simulating Cache race conditions")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// fake a race condition in snapshotting the cache.
|
|
|
|
realEngineStruct.Cache.snapshotting = true
|
|
|
|
defer func() {
|
|
|
|
realEngineStruct.Cache.snapshotting = false
|
|
|
|
}()
|
|
|
|
|
|
|
|
snapshotFunc := func(skipCacheOk bool) {
|
|
|
|
if f, err := sh.CreateSnapshot(skipCacheOk); err == nil {
|
|
|
|
require.NoError(t, os.RemoveAll(f), "error cleaning up TestEngine_ConcurrentShardSnapshots")
|
|
|
|
} else if err == ErrSnapshotInProgress {
|
|
|
|
if skipCacheOk {
|
|
|
|
t.Fatalf("failing to ignore this error,: %s", err.Error())
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Fatalf("error creating shard snapshot: %s", err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Permit skipping cache in the snapshot
|
|
|
|
snapshotFunc(true)
|
|
|
|
// do not permit skipping the cache in the snapshot
|
|
|
|
snapshotFunc(false)
|
|
|
|
realEngineStruct.Cache.snapshotting = false
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewSeriesFile returns a new instance of SeriesFile with a temporary file path.
|
2021-02-11 15:12:39 +00:00
|
|
|
func NewSeriesFile(tb testing.TB, tmpDir string) *tsdb.SeriesFile {
|
|
|
|
tb.Helper()
|
|
|
|
|
2021-01-19 03:02:26 +00:00
|
|
|
dir, err := ioutil.TempDir(tmpDir, "tsdb-series-file-")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
f := tsdb.NewSeriesFile(dir)
|
2021-02-11 15:12:39 +00:00
|
|
|
f.Logger = zaptest.NewLogger(tb)
|
2021-01-19 03:02:26 +00:00
|
|
|
if err := f.Open(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
|
|
|
type seriesIDSets []*tsdb.SeriesIDSet
|
|
|
|
|
|
|
|
func (a seriesIDSets) ForEach(f func(ids *tsdb.SeriesIDSet)) error {
|
|
|
|
for _, v := range a {
|
|
|
|
f(v)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|