prometheus/db.go

614 lines
13 KiB
Go
Raw Normal View History

2016-11-15 09:34:25 +00:00
// Package tsdb implements a time series storage for float64 sample data.
package tsdb
import (
"bytes"
2016-12-04 12:16:11 +00:00
"fmt"
"io/ioutil"
"math"
2016-12-04 12:16:11 +00:00
"os"
"path/filepath"
"reflect"
2016-12-15 07:31:26 +00:00
"strconv"
"sync"
"unsafe"
2016-11-15 09:34:25 +00:00
2016-12-15 07:31:26 +00:00
"golang.org/x/sync/errgroup"
2017-01-03 14:43:26 +00:00
"github.com/coreos/etcd/pkg/fileutil"
"github.com/fabxc/tsdb/chunks"
2016-12-21 08:39:01 +00:00
"github.com/fabxc/tsdb/labels"
"github.com/go-kit/kit/log"
2017-01-03 14:43:26 +00:00
"github.com/pkg/errors"
2016-12-31 08:48:49 +00:00
"github.com/prometheus/client_golang/prometheus"
2016-11-15 09:34:25 +00:00
)
2016-12-09 09:00:14 +00:00
// DefaultOptions used for the DB. They are sane for setups using
// millisecond precision timestamps.
2016-11-15 09:34:25 +00:00
var DefaultOptions = &Options{
2016-12-26 15:55:44 +00:00
Retention: 15 * 24 * 3600 * 1000, // 15 days
DisableWAL: false,
2016-11-15 09:34:25 +00:00
}
// Options of the DB storage.
type Options struct {
2016-12-26 15:55:44 +00:00
Retention int64
DisableWAL bool
2016-11-15 09:34:25 +00:00
}
// DB is a time series storage.
type DB struct {
logger log.Logger
opts *Options
path string
2016-11-15 09:34:25 +00:00
2017-01-06 07:08:02 +00:00
partitions []*Partition
2016-11-15 09:34:25 +00:00
}
2016-12-04 12:16:11 +00:00
// TODO(fabxc): make configurable
const (
2017-01-06 07:08:02 +00:00
partitionShift = 0
numPartitions = 1 << partitionShift
maxChunkSize = 1024
2016-12-04 12:16:11 +00:00
)
2016-11-15 09:34:25 +00:00
// Open or create a new DB.
func Open(path string, l log.Logger, opts *Options) (*DB, error) {
if opts == nil {
opts = DefaultOptions
}
2016-12-04 12:16:11 +00:00
if err := os.MkdirAll(path, 0777); err != nil {
return nil, err
}
if l == nil {
l = log.NewLogfmtLogger(os.Stdout)
l = log.NewContext(l).With("ts", log.DefaultTimestampUTC, "caller", log.DefaultCaller)
}
2016-11-15 09:34:25 +00:00
c := &DB{
2016-12-04 12:16:11 +00:00
logger: l,
opts: opts,
path: path,
2016-11-15 09:34:25 +00:00
}
2017-01-06 07:08:02 +00:00
// Initialize vertical partitions.
// TODO(fabxc): validate partition number to be power of 2, which is required
// for the bitshift-modulo when finding the right partition.
for i := 0; i < numPartitions; i++ {
l := log.NewContext(l).With("partition", i)
d := partitionDir(path, i)
2016-12-15 07:31:26 +00:00
2017-01-06 07:08:02 +00:00
s, err := OpenPartition(d, i, l)
2016-12-15 07:31:26 +00:00
if err != nil {
2017-01-06 07:08:02 +00:00
return nil, fmt.Errorf("initializing partition %q failed: %s", d, err)
2016-12-15 07:31:26 +00:00
}
2017-01-06 07:08:02 +00:00
c.partitions = append(c.partitions, s)
2016-12-04 12:16:11 +00:00
}
2016-11-15 09:34:25 +00:00
return c, nil
}
2017-01-06 07:08:02 +00:00
func partitionDir(base string, i int) string {
2016-12-15 07:31:26 +00:00
return filepath.Join(base, strconv.Itoa(i))
}
2016-12-04 12:16:11 +00:00
// Close the database.
func (db *DB) Close() error {
2016-12-15 07:31:26 +00:00
var g errgroup.Group
2016-12-09 09:00:14 +00:00
2017-01-06 07:08:02 +00:00
for _, partition := range db.partitions {
g.Go(partition.Close)
}
2016-12-09 09:00:14 +00:00
2016-12-15 07:31:26 +00:00
return g.Wait()
2016-12-04 12:16:11 +00:00
}
// Appender allows committing batches of samples to a database.
// The data held by the appender is reset after Commit returns.
type Appender interface {
// AddSeries registers a new known series label set with the appender
// and returns a reference number used to add samples to it over the
// life time of the Appender.
// AddSeries(Labels) uint64
// Add adds a sample pair for the referenced series.
2016-12-29 10:03:39 +00:00
Add(lset labels.Labels, t int64, v float64) error
// Commit submits the collected samples and purges the batch.
Commit() error
}
// Appender returns a new appender against the database.
func (db *DB) Appender() Appender {
return &bucketAppender{
db: db,
2017-01-06 07:08:02 +00:00
buckets: make([][]hashedSample, numPartitions),
}
}
type bucketAppender struct {
db *DB
buckets [][]hashedSample
}
2016-12-29 10:03:39 +00:00
func (ba *bucketAppender) Add(lset labels.Labels, t int64, v float64) error {
h := lset.Hash()
2017-01-06 07:08:02 +00:00
s := h >> (64 - partitionShift)
ba.buckets[s] = append(ba.buckets[s], hashedSample{
hash: h,
labels: lset,
t: t,
v: v,
})
2016-12-29 10:03:39 +00:00
return nil
}
func (ba *bucketAppender) reset() {
for i := range ba.buckets {
ba.buckets[i] = ba.buckets[i][:0]
2016-12-09 12:41:38 +00:00
}
}
func (ba *bucketAppender) Commit() error {
defer ba.reset()
2016-12-09 12:41:38 +00:00
var merr MultiError
2017-01-06 07:08:02 +00:00
// Spill buckets into partitions.
for s, b := range ba.buckets {
2017-01-06 07:08:02 +00:00
merr.Add(ba.db.partitions[s].appendBatch(b))
}
return merr.Err()
2016-12-09 12:41:38 +00:00
}
type hashedSample struct {
hash uint64
2016-12-21 08:39:01 +00:00
labels labels.Labels
2016-12-22 11:05:24 +00:00
ref uint32
t int64
v float64
2016-12-09 15:54:38 +00:00
}
2016-12-09 09:00:14 +00:00
const sep = '\xff'
2017-01-06 07:08:02 +00:00
// Partition handles reads and writes of time series falling into
// a hashed partition of a series.
type Partition struct {
dir string
2017-01-02 21:24:35 +00:00
logger log.Logger
2017-01-06 07:08:02 +00:00
metrics *partitionMetrics
2016-12-09 09:00:14 +00:00
2016-12-15 07:31:26 +00:00
mtx sync.RWMutex
2017-01-03 14:43:26 +00:00
persisted []*persistedBlock
heads []*HeadBlock
compactor *compactor
2016-12-09 09:00:14 +00:00
}
2017-01-06 07:08:02 +00:00
type partitionMetrics struct {
2016-12-31 08:48:49 +00:00
persistences prometheus.Counter
persistenceDuration prometheus.Histogram
samplesAppended prometheus.Counter
}
2017-01-06 07:08:02 +00:00
func newPartitionMetrics(r prometheus.Registerer, i int) *partitionMetrics {
partitionLabel := prometheus.Labels{
"partition": fmt.Sprintf("%d", i),
2016-12-31 08:48:49 +00:00
}
2017-01-06 07:08:02 +00:00
m := &partitionMetrics{}
2017-01-03 14:43:26 +00:00
m.persistences = prometheus.NewCounter(prometheus.CounterOpts{
2017-01-06 07:08:02 +00:00
Name: "tsdb_partition_persistences_total",
2017-01-03 14:43:26 +00:00
Help: "Total number of head persistances that ran so far.",
2017-01-06 07:08:02 +00:00
ConstLabels: partitionLabel,
2017-01-03 14:43:26 +00:00
})
m.persistenceDuration = prometheus.NewHistogram(prometheus.HistogramOpts{
2017-01-06 07:08:02 +00:00
Name: "tsdb_partition_persistence_duration_seconds",
2017-01-03 14:43:26 +00:00
Help: "Duration of persistences in seconds.",
2017-01-06 07:08:02 +00:00
ConstLabels: partitionLabel,
2017-01-03 14:43:26 +00:00
Buckets: prometheus.ExponentialBuckets(0.25, 2, 5),
})
m.samplesAppended = prometheus.NewCounter(prometheus.CounterOpts{
2017-01-06 07:08:02 +00:00
Name: "tsdb_partition_samples_appended_total",
Help: "Total number of appended samples for the partition.",
ConstLabels: partitionLabel,
2017-01-03 14:43:26 +00:00
})
2016-12-31 08:48:49 +00:00
if r != nil {
r.MustRegister(
m.persistences,
m.persistenceDuration,
m.samplesAppended,
)
}
return m
}
2017-01-06 07:08:02 +00:00
// OpenPartition returns a new Partition.
func OpenPartition(dir string, i int, logger log.Logger) (p *Partition, err error) {
2017-01-06 07:08:02 +00:00
// Create directory if partition is new.
if !fileutil.Exist(dir) {
if err := os.MkdirAll(dir, 0777); err != nil {
2016-12-15 07:31:26 +00:00
return nil, err
}
}
p = &Partition{
dir: dir,
logger: logger,
metrics: newPartitionMetrics(nil, i),
}
if err := p.initBlocks(); err != nil {
2016-12-15 07:31:26 +00:00
return nil, err
}
if p.compactor, err = newCompactor(i, p, logger); err != nil {
return nil, err
}
return p, nil
}
2016-12-15 07:31:26 +00:00
func isBlockDir(fi os.FileInfo) bool {
if !fi.IsDir() {
return false
2016-12-22 11:05:24 +00:00
}
if _, err := strconv.ParseUint(fi.Name(), 10, 32); err != nil {
return false
2016-12-22 11:05:24 +00:00
}
return true
}
func (p *Partition) initBlocks() error {
var (
pbs []*persistedBlock
heads []*HeadBlock
)
2016-12-22 11:05:24 +00:00
files, err := ioutil.ReadDir(p.dir)
if err != nil {
return err
2016-12-09 09:00:14 +00:00
}
for _, fi := range files {
if !isBlockDir(fi) {
continue
}
dir := filepath.Join(p.dir, fi.Name())
if fileutil.Exist(filepath.Join(dir, walFileName)) {
h, err := OpenHeadBlock(dir)
if err != nil {
return err
}
heads = append(heads, h)
continue
}
b, err := newPersistedBlock(dir)
if err != nil {
return err
}
pbs = append(pbs, b)
}
// Validate that blocks are sequential in time.
lastTime := int64(math.MinInt64)
2016-12-09 12:41:38 +00:00
for _, b := range pbs {
if b.stats().MinTime < lastTime {
return errors.Errorf("illegal order for block at %q", b.dir())
}
lastTime = b.stats().MaxTime
}
for _, b := range heads {
if b.stats().MinTime < lastTime {
return errors.Errorf("illegal order for block at %q", b.dir())
}
lastTime = b.stats().MaxTime
}
2017-01-03 14:43:26 +00:00
p.persisted = pbs
p.heads = heads
2017-01-03 14:43:26 +00:00
if len(heads) == 0 {
return p.cut()
2017-01-02 21:24:35 +00:00
}
return nil
2017-01-02 21:24:35 +00:00
}
2017-01-06 07:08:02 +00:00
// Close the partition.
func (p *Partition) Close() error {
2017-01-02 21:24:35 +00:00
var merr MultiError
merr.Add(p.compactor.Close())
2016-12-15 07:31:26 +00:00
p.mtx.Lock()
defer p.mtx.Unlock()
for _, pb := range p.persisted {
2017-01-02 21:24:35 +00:00
merr.Add(pb.Close())
2016-12-15 07:31:26 +00:00
}
for _, hb := range p.heads {
2017-01-03 14:43:26 +00:00
merr.Add(hb.Close())
}
2016-12-15 07:31:26 +00:00
2017-01-02 21:24:35 +00:00
return merr.Err()
2016-12-09 09:00:14 +00:00
}
2017-01-06 07:08:02 +00:00
func (s *Partition) appendBatch(samples []hashedSample) error {
if len(samples) == 0 {
return nil
}
2016-12-09 12:41:38 +00:00
s.mtx.Lock()
defer s.mtx.Unlock()
2017-01-03 14:43:26 +00:00
head := s.heads[len(s.heads)-1]
2016-12-22 00:12:28 +00:00
// TODO(fabxc): distinguish samples between concurrent heads for
// different time blocks. Those may occurr during transition to still
// allow late samples to arrive for a previous block.
2017-01-03 14:43:26 +00:00
err := head.appendBatch(samples)
if err == nil {
s.metrics.samplesAppended.Add(float64(len(samples)))
}
2016-12-09 12:41:38 +00:00
2016-12-22 00:12:28 +00:00
// TODO(fabxc): randomize over time and use better scoring function.
if head.bstats.SampleCount/(uint64(head.bstats.ChunkCount)+1) > 250 {
2017-01-03 14:43:26 +00:00
if err := s.cut(); err != nil {
s.logger.Log("msg", "cut failed", "err", err)
} else {
s.compactor.trigger()
2016-12-09 12:41:38 +00:00
}
}
2016-12-22 00:12:28 +00:00
return err
2016-12-09 12:41:38 +00:00
}
2017-01-06 07:08:02 +00:00
func (s *Partition) lock() sync.Locker {
2017-01-03 14:43:26 +00:00
return &s.mtx
}
2017-01-06 07:08:02 +00:00
func (s *Partition) headForDir(dir string) (int, bool) {
2017-01-03 14:43:26 +00:00
for i, b := range s.heads {
if b.dir() == dir {
return i, true
}
}
return -1, false
}
2017-01-06 07:08:02 +00:00
func (s *Partition) persistedForDir(dir string) (int, bool) {
2017-01-03 14:43:26 +00:00
for i, b := range s.persisted {
if b.dir() == dir {
return i, true
}
}
return -1, false
}
2017-01-06 07:08:02 +00:00
func (s *Partition) reinit(dir string) error {
2017-01-03 14:43:26 +00:00
if !fileutil.Exist(dir) {
if i, ok := s.headForDir(dir); ok {
if err := s.heads[i].Close(); err != nil {
return err
}
s.heads = append(s.heads[:i], s.heads[i+1:]...)
}
if i, ok := s.persistedForDir(dir); ok {
if err := s.persisted[i].Close(); err != nil {
return err
}
s.persisted = append(s.persisted[:i], s.persisted[i+1:]...)
}
return nil
}
// Remove a previous head block.
if i, ok := s.headForDir(dir); ok {
if err := s.heads[i].Close(); err != nil {
return err
}
s.heads = append(s.heads[:i], s.heads[i+1:]...)
}
// Close an old persisted block.
i, ok := s.persistedForDir(dir)
if ok {
if err := s.persisted[i].Close(); err != nil {
return err
}
}
pb, err := newPersistedBlock(dir)
if err != nil {
return errors.Wrap(err, "open persisted block")
}
if i >= 0 {
s.persisted[i] = pb
} else {
s.persisted = append(s.persisted, pb)
}
return nil
}
2017-01-06 07:08:02 +00:00
func (s *Partition) compactable() []block {
2017-01-03 14:43:26 +00:00
var blocks []block
for _, pb := range s.persisted {
blocks = append([]block{pb}, blocks...)
2017-01-03 14:43:26 +00:00
}
// threshold := s.heads[len(s.heads)-1].bstats.MaxTime - headGracePeriod
// for _, hb := range s.heads {
// if hb.bstats.MaxTime < threshold {
// blocks = append(blocks, hb)
// }
// }
for _, hb := range s.heads[:len(s.heads)-1] {
blocks = append([]block{hb}, blocks...)
2017-01-03 14:43:26 +00:00
}
return blocks
}
func intervalOverlap(amin, amax, bmin, bmax int64) bool {
if bmin >= amin && bmin <= amax {
return true
}
if amin >= bmin && amin <= bmax {
return true
}
return false
}
func intervalContains(min, max, t int64) bool {
return t >= min && t <= max
}
// blocksForInterval returns all blocks within the partition that may contain
2016-12-13 14:26:58 +00:00
// data for the given time range.
2017-01-06 07:08:02 +00:00
func (s *Partition) blocksForInterval(mint, maxt int64) []block {
var bs []block
for _, b := range s.persisted {
bmin, bmax := b.interval()
if intervalOverlap(mint, maxt, bmin, bmax) {
bs = append(bs, b)
}
}
2017-01-03 14:43:26 +00:00
for _, b := range s.heads {
bmin, bmax := b.interval()
2017-01-03 14:43:26 +00:00
if intervalOverlap(mint, maxt, bmin, bmax) {
bs = append(bs, b)
}
}
return bs
2016-12-13 14:26:58 +00:00
}
2016-12-09 12:41:38 +00:00
// TODO(fabxc): make configurable.
2017-01-03 14:43:26 +00:00
const headGracePeriod = 60 * 1000 // 60 seconds for millisecond scale
2016-12-09 12:41:38 +00:00
2017-01-03 14:43:26 +00:00
// cut starts a new head block to append to. The completed head block
// will still be appendable for the configured grace period.
func (p *Partition) cut() error {
dir, err := p.nextBlockDir()
2016-12-22 11:05:24 +00:00
if err != nil {
return err
}
newHead, err := OpenHeadBlock(dir)
if err != nil {
return err
}
p.heads = append(p.heads, newHead)
2016-12-09 12:41:38 +00:00
2017-01-03 14:43:26 +00:00
return nil
}
2016-12-09 12:41:38 +00:00
func (p *Partition) nextBlockDir() (string, error) {
names, err := fileutil.ReadDir(p.dir)
if err != nil {
return "", err
}
i := uint64(0)
if len(names) > 0 {
if i, err = strconv.ParseUint(names[len(names)-1], 10, 32); err != nil {
return "", err
}
}
return filepath.Join(p.dir, fmt.Sprintf("%0.6d", i+1)), nil
}
2016-12-09 12:41:38 +00:00
2016-12-09 09:00:14 +00:00
// chunkDesc wraps a plain data chunk and provides cached meta data about it.
type chunkDesc struct {
ref uint32
2016-12-21 08:39:01 +00:00
lset labels.Labels
2016-12-09 09:00:14 +00:00
chunk chunks.Chunk
// Caching fields.
firstTimestamp int64
lastTimestamp int64
lastValue float64
numSamples int
2016-12-09 09:00:14 +00:00
app chunks.Appender // Current appender for the chunks.
}
2016-12-31 09:10:27 +00:00
func (cd *chunkDesc) append(ts int64, v float64) {
if cd.numSamples == 0 {
cd.firstTimestamp = ts
2016-12-09 09:00:14 +00:00
}
2016-12-31 09:10:27 +00:00
cd.app.Append(ts, v)
2016-12-09 09:00:14 +00:00
cd.lastTimestamp = ts
cd.lastValue = v
2016-12-22 19:57:00 +00:00
cd.numSamples++
2016-12-09 09:00:14 +00:00
}
// The MultiError type implements the error interface, and contains the
// Errors used to construct it.
type MultiError []error
// Returns a concatenated string of the contained errors
func (es MultiError) Error() string {
var buf bytes.Buffer
if len(es) > 1 {
fmt.Fprintf(&buf, "%d errors: ", len(es))
2016-12-08 09:04:24 +00:00
}
for i, err := range es {
if i != 0 {
buf.WriteString("; ")
}
buf.WriteString(err.Error())
}
return buf.String()
2016-11-15 09:34:25 +00:00
}
2016-12-15 07:31:26 +00:00
// Add adds the error to the error list if it is not nil.
func (es *MultiError) Add(err error) {
if err == nil {
return
}
if merr, ok := err.(MultiError); ok {
*es = append(*es, merr...)
} else {
*es = append(*es, err)
2016-12-15 07:31:26 +00:00
}
}
// Err returns the error list as an error or nil if it is empty.
2016-12-15 07:31:26 +00:00
func (es MultiError) Err() error {
if len(es) == 0 {
return nil
}
return es
}
func yoloString(b []byte) string {
sh := (*reflect.SliceHeader)(unsafe.Pointer(&b))
h := reflect.StringHeader{
Data: sh.Data,
Len: sh.Len,
}
return *((*string)(unsafe.Pointer(&h)))
}
func yoloBytes(s string) []byte {
sh := (*reflect.StringHeader)(unsafe.Pointer(&s))
h := reflect.SliceHeader{
Cap: sh.Len,
Len: sh.Len,
Data: sh.Data,
}
return *((*[]byte)(unsafe.Pointer(&h)))
}