influxdb/tsdb/tsm1/reader_time_range.go

90 lines
2.1 KiB
Go
Raw Normal View History

package tsm1
// TimeRange holds a min and max timestamp.
type TimeRange struct {
Min, Max int64
}
func (t TimeRange) Overlaps(min, max int64) bool {
return t.Min <= max && t.Max >= min
}
func (t TimeRange) Less(o TimeRange) bool {
return t.Min < o.Min || (t.Min == o.Min && t.Max < o.Max)
}
// timeRangesCoverEntries returns true if the time ranges fully cover the entries.
func timeRangesCoverEntries(merger timeRangeMerger, entries []IndexEntry) (covers bool) {
if len(entries) == 0 {
return true
}
mustCover := entries[0].MinTime
ts, ok := merger.Pop()
for len(entries) > 0 && ok {
switch {
// If the tombstone does not include mustCover, we
// know we do not fully cover every entry.
case ts.Min > mustCover:
return false
// Otherwise, if the tombstone covers the rest of
// the entry, consume it and bump mustCover to the
// start of the next entry.
case ts.Max >= entries[0].MaxTime:
entries = entries[1:]
if len(entries) > 0 {
mustCover = entries[0].MinTime
}
// Otherwise, we're still inside of an entry, and
// so the tombstone must adjoin the current tombstone.
default:
if ts.Max >= mustCover {
mustCover = ts.Max + 1
}
ts, ok = merger.Pop()
}
}
return len(entries) == 0
}
// timeRangeMerger is a special purpose data structure to merge three sources of
// TimeRanges so that we can check if they cover a slice of index entries.
type timeRangeMerger struct {
sorted []TimeRange
unsorted []TimeRange
single TimeRange
used bool // if single has been used
}
// Pop returns the next TimeRange in sorted order and a boolean indicating that
// there was a TimeRange to read.
func (t *timeRangeMerger) Pop() (out TimeRange, ok bool) {
var where *[]TimeRange
var what []TimeRange
if len(t.sorted) > 0 {
where, what = &t.sorted, t.sorted[1:]
out, ok = t.sorted[0], true
}
if len(t.unsorted) > 0 && (!ok || t.unsorted[0].Less(out)) {
where, what = &t.unsorted, t.unsorted[1:]
out, ok = t.unsorted[0], true
}
if !t.used && (!ok || t.single.Less(out)) {
t.used = true
return t.single, true
}
if ok {
*where = what
}
return out, ok
}