influxdb/kv/cursor.go

81 lines
1.7 KiB
Go
Raw Normal View History

feat(platform): add generic kv store Co-authored-by: Leonardo Di Donato <leodidonato@gmail.com> Co-authored-by: Michael Desa <mjdesa@gmail.com> feat(kv): add kv store interface for services feat(bolt): add boltdb implementation of kv.Store spike(platform): add kv backed user service feat(kv): add static cursor Note here that this operation cannot be transactionally done. This poses a bit of issues that will need to be worked out. fix(bolt): use error explicit error message squash: play with interface a bit fix(kv): remove commit and rollback from kv interface feat(inmem): add inmem kv store chore: add note for inmem transactions fix(bolt): remove call to tx in kv store tests feat(kv): add tests for static cursor doc(kv): add comments to store and associated interfaces doc(bolt): add comments to key value store feat(testing): add kv store tests test(testing): add conformance test for kv.Store test(inmem): add kv.Store conformance tests doc(inmem): add comments to key value store feat(inmem): remove CreateBucketIfNotExists from Tx interface feat(bolt): remove CreateBucketIfNotExists from Tx feat(inmem): remove CreateBucketIfNotExists from Tx doc(kv): add note to bucket interface about conditions methods can be called feat(kv): add context methods to kv.Tx feat(bolt): add context methods to bolt.Tx feat(inmem): add context methods to inmem.Tx test(kv): add contract tests for view/update transactions feat(kv): ensure that static cursor is always valid Co-authored-by: Leonardo Di Donato <leodidonato@gmail.com> Co-authored-by: Michael Desa <mjdesa@gmail.com> fix(kv): remove error from cursor methods test(kv): remove want errors from cursor test test(testing): add concurrent update test for kv.Store feat(kv): make kv user service an example service fix(testing): add concurrnent update test to the kv.Store contract tests test(platform): fix example kv service tests dep(platform): make platform tidy
2018-12-18 15:44:25 +00:00
package kv
import (
"bytes"
"sort"
)
// staticCursor implements the Cursor interface for a slice of
// static key value pairs.
type staticCursor struct {
idx int
pairs []Pair
}
// Pair is a struct for key value pairs.
type Pair struct {
Key []byte
Value []byte
}
// NewStaticCursor returns an instance of a StaticCursor. It
// destructively sorts the provided pairs to be in key ascending order.
func NewStaticCursor(pairs []Pair) Cursor {
sort.Slice(pairs, func(i, j int) bool {
return bytes.Compare(pairs[i].Key, pairs[j].Key) < 0
})
return &staticCursor{
pairs: pairs,
}
}
// Seek searches the slice for the first key with the provided prefix.
func (c *staticCursor) Seek(prefix []byte) ([]byte, []byte) {
// TODO: do binary search for prefix since pairs are ordered.
for i, pair := range c.pairs {
if bytes.HasPrefix(pair.Key, prefix) {
c.idx = i
return pair.Key, pair.Value
}
}
return nil, nil
}
func (c *staticCursor) getValueAtIndex(delta int) ([]byte, []byte) {
idx := c.idx + delta
if idx < 0 {
return nil, nil
}
if idx >= len(c.pairs) {
return nil, nil
}
c.idx = idx
pair := c.pairs[c.idx]
return pair.Key, pair.Value
}
// First retrieves the first element in the cursor.
func (c *staticCursor) First() ([]byte, []byte) {
return c.getValueAtIndex(-c.idx)
}
// Last retrieves the last element in the cursor.
func (c *staticCursor) Last() ([]byte, []byte) {
return c.getValueAtIndex(len(c.pairs) - 1 - c.idx)
}
// Next retrieves the next entry in the cursor.
func (c *staticCursor) Next() ([]byte, []byte) {
return c.getValueAtIndex(1)
}
// Prev retrieves the previous entry in the cursor.
func (c *staticCursor) Prev() ([]byte, []byte) {
return c.getValueAtIndex(-1)
}