2015-05-22 20:08:43 +00:00
|
|
|
package tsdb
|
|
|
|
|
|
|
|
import (
|
2015-06-18 15:07:51 +00:00
|
|
|
"fmt"
|
2015-05-22 20:08:43 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2015-06-02 20:57:39 +00:00
|
|
|
"path"
|
2015-06-18 15:07:51 +00:00
|
|
|
"path/filepath"
|
2015-05-22 20:08:43 +00:00
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestShardWriteAndIndex(t *testing.T) {
|
2015-06-03 14:09:50 +00:00
|
|
|
tmpDir, _ := ioutil.TempDir("", "shard_test")
|
2015-06-02 20:57:39 +00:00
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
tmpShard := path.Join(tmpDir, "shard")
|
2015-05-22 20:08:43 +00:00
|
|
|
|
2015-05-24 11:39:45 +00:00
|
|
|
index := NewDatabaseIndex()
|
2015-06-02 20:57:39 +00:00
|
|
|
sh := NewShard(index, tmpShard)
|
2015-05-26 15:41:15 +00:00
|
|
|
if err := sh.Open(); err != nil {
|
2015-05-22 20:08:43 +00:00
|
|
|
t.Fatalf("error openeing shard: %s", err.Error())
|
|
|
|
}
|
|
|
|
|
2015-05-22 21:00:51 +00:00
|
|
|
pt := NewPoint(
|
|
|
|
"cpu",
|
|
|
|
map[string]string{"host": "server"},
|
|
|
|
map[string]interface{}{"value": 1.0},
|
|
|
|
time.Unix(1, 2),
|
|
|
|
)
|
2015-05-22 20:08:43 +00:00
|
|
|
|
2015-05-22 21:39:55 +00:00
|
|
|
err := sh.WritePoints([]Point{pt})
|
2015-05-22 20:08:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf(err.Error())
|
|
|
|
}
|
|
|
|
|
2015-05-22 21:00:51 +00:00
|
|
|
pt.SetTime(time.Unix(2, 3))
|
2015-05-22 21:39:55 +00:00
|
|
|
err = sh.WritePoints([]Point{pt})
|
2015-05-22 20:08:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
validateIndex := func() {
|
2015-05-23 22:06:07 +00:00
|
|
|
if !reflect.DeepEqual(index.names, []string{"cpu"}) {
|
|
|
|
t.Fatalf("measurement names in shard didn't match")
|
2015-05-22 20:08:43 +00:00
|
|
|
}
|
2015-05-23 22:06:07 +00:00
|
|
|
if len(index.series) != 1 {
|
2015-05-22 20:08:43 +00:00
|
|
|
t.Fatalf("series wasn't in index")
|
|
|
|
}
|
2015-05-28 21:47:52 +00:00
|
|
|
seriesTags := index.series[string(pt.Key())].Tags
|
2015-05-22 21:12:34 +00:00
|
|
|
if len(seriesTags) != len(pt.Tags()) || pt.Tags()["host"] != seriesTags["host"] {
|
2015-05-28 21:47:52 +00:00
|
|
|
t.Fatalf("tags weren't properly saved to series index: %v, %v", pt.Tags(), index.series[string(pt.Key())].Tags)
|
2015-05-22 20:08:43 +00:00
|
|
|
}
|
2015-06-04 18:50:32 +00:00
|
|
|
if !reflect.DeepEqual(index.measurements["cpu"].TagKeys(), []string{"host"}) {
|
2015-05-22 20:08:43 +00:00
|
|
|
t.Fatalf("tag key wasn't saved to measurement index")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
validateIndex()
|
|
|
|
|
|
|
|
// ensure the index gets loaded after closing and opening the shard
|
|
|
|
sh.Close()
|
|
|
|
|
2015-05-24 11:39:45 +00:00
|
|
|
index = NewDatabaseIndex()
|
2015-06-02 20:57:39 +00:00
|
|
|
sh = NewShard(index, tmpShard)
|
2015-05-26 15:41:15 +00:00
|
|
|
if err := sh.Open(); err != nil {
|
2015-05-22 20:08:43 +00:00
|
|
|
t.Fatalf("error openeing shard: %s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
validateIndex()
|
|
|
|
|
|
|
|
// and ensure that we can still write data
|
2015-05-22 21:00:51 +00:00
|
|
|
pt.SetTime(time.Unix(2, 6))
|
2015-05-22 21:39:55 +00:00
|
|
|
err = sh.WritePoints([]Point{pt})
|
2015-05-22 20:08:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf(err.Error())
|
|
|
|
}
|
|
|
|
}
|
2015-06-02 20:57:39 +00:00
|
|
|
|
2015-06-10 16:27:11 +00:00
|
|
|
func TestShardWriteAddNewField(t *testing.T) {
|
|
|
|
tmpDir, _ := ioutil.TempDir("", "shard_test")
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
tmpShard := path.Join(tmpDir, "shard")
|
|
|
|
|
|
|
|
index := NewDatabaseIndex()
|
|
|
|
sh := NewShard(index, tmpShard)
|
|
|
|
if err := sh.Open(); err != nil {
|
|
|
|
t.Fatalf("error openeing shard: %s", err.Error())
|
|
|
|
}
|
|
|
|
defer sh.Close()
|
|
|
|
|
|
|
|
pt := NewPoint(
|
|
|
|
"cpu",
|
|
|
|
map[string]string{"host": "server"},
|
|
|
|
map[string]interface{}{"value": 1.0},
|
|
|
|
time.Unix(1, 2),
|
|
|
|
)
|
|
|
|
|
|
|
|
err := sh.WritePoints([]Point{pt})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
pt = NewPoint(
|
|
|
|
"cpu",
|
|
|
|
map[string]string{"host": "server"},
|
|
|
|
map[string]interface{}{"value": 1.0, "value2": 2.0},
|
|
|
|
time.Unix(1, 2),
|
|
|
|
)
|
|
|
|
|
|
|
|
err = sh.WritePoints([]Point{pt})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(index.names, []string{"cpu"}) {
|
|
|
|
t.Fatalf("measurement names in shard didn't match")
|
|
|
|
}
|
|
|
|
if len(index.series) != 1 {
|
|
|
|
t.Fatalf("series wasn't in index")
|
|
|
|
}
|
|
|
|
seriesTags := index.series[string(pt.Key())].Tags
|
|
|
|
if len(seriesTags) != len(pt.Tags()) || pt.Tags()["host"] != seriesTags["host"] {
|
|
|
|
t.Fatalf("tags weren't properly saved to series index: %v, %v", pt.Tags(), index.series[string(pt.Key())].Tags)
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(index.measurements["cpu"].TagKeys(), []string{"host"}) {
|
|
|
|
t.Fatalf("tag key wasn't saved to measurement index")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(index.measurements["cpu"].FieldNames()) != 2 {
|
|
|
|
t.Fatalf("field names wasn't saved to measurement index")
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-06-18 15:07:51 +00:00
|
|
|
// Ensure the shard will automatically flush the WAL after a threshold has been reached.
|
|
|
|
func TestShard_Autoflush(t *testing.T) {
|
|
|
|
path, _ := ioutil.TempDir("", "shard_test")
|
|
|
|
defer os.RemoveAll(path)
|
|
|
|
|
|
|
|
// Open shard with a really low size threshold, high flush interval.
|
|
|
|
sh := NewShard(NewDatabaseIndex(), filepath.Join(path, "shard"))
|
|
|
|
sh.MaxWALSize = 1024 // 1KB
|
|
|
|
sh.WALFlushInterval = 1 * time.Hour
|
|
|
|
if err := sh.Open(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer sh.Close()
|
|
|
|
|
|
|
|
// Write a bunch of points.
|
|
|
|
for i := 0; i < 100; i++ {
|
|
|
|
if err := sh.WritePoints([]Point{NewPoint(
|
|
|
|
fmt.Sprintf("cpu%d", i),
|
|
|
|
map[string]string{"host": "server"},
|
|
|
|
map[string]interface{}{"value": 1.0},
|
|
|
|
time.Unix(1, 2),
|
|
|
|
)}); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for autoflush.
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
|
|
|
|
// Make sure we have series buckets created outside the WAL.
|
|
|
|
if n, err := sh.SeriesCount(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
} else if n < 10 {
|
|
|
|
t.Fatalf("not enough series, expected at least 10, got %d", n)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the shard will automatically flush the WAL after a threshold has been reached.
|
|
|
|
func TestShard_Autoflush_FlushInterval(t *testing.T) {
|
|
|
|
path, _ := ioutil.TempDir("", "shard_test")
|
|
|
|
defer os.RemoveAll(path)
|
|
|
|
|
|
|
|
// Open shard with a high size threshold, small time threshold.
|
|
|
|
sh := NewShard(NewDatabaseIndex(), filepath.Join(path, "shard"))
|
|
|
|
sh.MaxWALSize = 10 * 1024 * 1024 // 10MB
|
|
|
|
sh.WALFlushInterval = 100 * time.Millisecond
|
|
|
|
if err := sh.Open(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer sh.Close()
|
|
|
|
|
|
|
|
// Write some points.
|
|
|
|
for i := 0; i < 100; i++ {
|
|
|
|
if err := sh.WritePoints([]Point{NewPoint(
|
|
|
|
fmt.Sprintf("cpu%d", i),
|
|
|
|
map[string]string{"host": "server"},
|
|
|
|
map[string]interface{}{"value": 1.0},
|
|
|
|
time.Unix(1, 2),
|
|
|
|
)}); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for time-based flush.
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
|
|
|
|
// Make sure we have series buckets created outside the WAL.
|
|
|
|
if n, err := sh.SeriesCount(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
} else if n < 10 {
|
|
|
|
t.Fatalf("not enough series, expected at least 10, got %d", n)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-02 20:57:39 +00:00
|
|
|
func BenchmarkWritePoints_NewSeries_1K(b *testing.B) { benchmarkWritePoints(b, 38, 3, 3, 1) }
|
|
|
|
func BenchmarkWritePoints_NewSeries_100K(b *testing.B) { benchmarkWritePoints(b, 32, 5, 5, 1) }
|
|
|
|
func BenchmarkWritePoints_NewSeries_250K(b *testing.B) { benchmarkWritePoints(b, 80, 5, 5, 1) }
|
|
|
|
func BenchmarkWritePoints_NewSeries_500K(b *testing.B) { benchmarkWritePoints(b, 160, 5, 5, 1) }
|
|
|
|
func BenchmarkWritePoints_NewSeries_1M(b *testing.B) { benchmarkWritePoints(b, 320, 5, 5, 1) }
|
|
|
|
|
|
|
|
func BenchmarkWritePoints_ExistingSeries_1K(b *testing.B) {
|
|
|
|
benchmarkWritePointsExistingSeries(b, 38, 3, 3, 1)
|
|
|
|
}
|
|
|
|
func BenchmarkWritePoints_ExistingSeries_100K(b *testing.B) {
|
|
|
|
benchmarkWritePointsExistingSeries(b, 32, 5, 5, 1)
|
|
|
|
}
|
|
|
|
func BenchmarkWritePoints_ExistingSeries_250K(b *testing.B) {
|
|
|
|
benchmarkWritePointsExistingSeries(b, 80, 5, 5, 1)
|
|
|
|
}
|
|
|
|
func BenchmarkWritePoints_ExistingSeries_500K(b *testing.B) {
|
|
|
|
benchmarkWritePointsExistingSeries(b, 160, 5, 5, 1)
|
|
|
|
}
|
|
|
|
func BenchmarkWritePoints_ExistingSeries_1M(b *testing.B) {
|
|
|
|
benchmarkWritePointsExistingSeries(b, 320, 5, 5, 1)
|
|
|
|
}
|
|
|
|
|
2015-06-03 14:09:50 +00:00
|
|
|
// benchmarkWritePoints benchmarks writing new series to a shard.
|
|
|
|
// mCnt - measurmeent count
|
|
|
|
// tkCnt - tag key count
|
|
|
|
// tvCnt - tag value count (values per tag)
|
|
|
|
// pntCnt - points per series. # of series = mCnt * (tvCnt ^ tkCnt)
|
2015-06-02 20:57:39 +00:00
|
|
|
func benchmarkWritePoints(b *testing.B, mCnt, tkCnt, tvCnt, pntCnt int) {
|
|
|
|
// Generate test series (measurements + unique tag sets).
|
|
|
|
series := genTestSeries(mCnt, tkCnt, tvCnt)
|
2015-06-02 21:17:31 +00:00
|
|
|
// Create index for the shard to use.
|
2015-06-02 20:57:39 +00:00
|
|
|
index := NewDatabaseIndex()
|
|
|
|
// Generate point data to write to the shard.
|
|
|
|
points := []Point{}
|
|
|
|
for _, s := range series {
|
|
|
|
for val := 0.0; val < float64(pntCnt); val++ {
|
|
|
|
p := NewPoint(s.Measurement, s.Series.Tags, map[string]interface{}{"value": val}, time.Now())
|
|
|
|
points = append(points, p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stop & reset timers and mem-stats before the main benchmark loop.
|
|
|
|
b.StopTimer()
|
|
|
|
b.ResetTimer()
|
|
|
|
|
|
|
|
// Run the benchmark loop.
|
|
|
|
for n := 0; n < b.N; n++ {
|
2015-06-03 14:09:50 +00:00
|
|
|
tmpDir, _ := ioutil.TempDir("", "shard_test")
|
2015-06-02 20:57:39 +00:00
|
|
|
tmpShard := path.Join(tmpDir, "shard")
|
|
|
|
shard := NewShard(index, tmpShard)
|
|
|
|
shard.Open()
|
|
|
|
|
|
|
|
b.StartTimer()
|
|
|
|
// Call the function being benchmarked.
|
|
|
|
chunkedWrite(shard, points)
|
|
|
|
|
|
|
|
b.StopTimer()
|
|
|
|
shard.Close()
|
|
|
|
os.RemoveAll(tmpDir)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-03 14:09:50 +00:00
|
|
|
// benchmarkWritePointsExistingSeries benchmarks writing to existing series in a shard.
|
|
|
|
// mCnt - measurmeent count
|
|
|
|
// tkCnt - tag key count
|
|
|
|
// tvCnt - tag value count (values per tag)
|
|
|
|
// pntCnt - points per series. # of series = mCnt * (tvCnt ^ tkCnt)
|
2015-06-02 20:57:39 +00:00
|
|
|
func benchmarkWritePointsExistingSeries(b *testing.B, mCnt, tkCnt, tvCnt, pntCnt int) {
|
|
|
|
// Generate test series (measurements + unique tag sets).
|
|
|
|
series := genTestSeries(mCnt, tkCnt, tvCnt)
|
2015-06-02 21:17:31 +00:00
|
|
|
// Create index for the shard to use.
|
2015-06-02 20:57:39 +00:00
|
|
|
index := NewDatabaseIndex()
|
|
|
|
// Generate point data to write to the shard.
|
|
|
|
points := []Point{}
|
|
|
|
for _, s := range series {
|
|
|
|
for val := 0.0; val < float64(pntCnt); val++ {
|
|
|
|
p := NewPoint(s.Measurement, s.Series.Tags, map[string]interface{}{"value": val}, time.Now())
|
|
|
|
points = append(points, p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tmpDir, _ := ioutil.TempDir("", "")
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
tmpShard := path.Join(tmpDir, "shard")
|
|
|
|
shard := NewShard(index, tmpShard)
|
|
|
|
shard.Open()
|
|
|
|
defer shard.Close()
|
|
|
|
chunkedWrite(shard, points)
|
|
|
|
|
|
|
|
// Reset timers and mem-stats before the main benchmark loop.
|
|
|
|
b.ResetTimer()
|
|
|
|
|
|
|
|
// Run the benchmark loop.
|
|
|
|
for n := 0; n < b.N; n++ {
|
|
|
|
b.StopTimer()
|
|
|
|
for _, p := range points {
|
|
|
|
p.SetTime(p.Time().Add(time.Second))
|
|
|
|
}
|
|
|
|
|
|
|
|
b.StartTimer()
|
|
|
|
// Call the function being benchmarked.
|
|
|
|
chunkedWrite(shard, points)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func chunkedWrite(shard *Shard, points []Point) {
|
|
|
|
nPts := len(points)
|
|
|
|
chunkSz := 10000
|
|
|
|
start := 0
|
|
|
|
end := chunkSz
|
|
|
|
|
|
|
|
for {
|
|
|
|
if end > nPts {
|
|
|
|
end = nPts
|
|
|
|
}
|
|
|
|
if end-start == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
shard.WritePoints(points[start:end])
|
|
|
|
start = end
|
|
|
|
end += chunkSz
|
|
|
|
}
|
|
|
|
}
|