Memory and file store list fixes (#1959)
* Refactor file and memory stores
This commit is contained in:
parent
329bc2f265
commit
78a79ca9e1
@ -2,11 +2,10 @@
|
|||||||
package file
|
package file
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sort"
|
|
||||||
"strings"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/micro/go-micro/v3/store"
|
"github.com/micro/go-micro/v3/store"
|
||||||
@ -111,8 +110,9 @@ func (f *fileStore) getDB(database, table string) (*bolt.DB, error) {
|
|||||||
return bolt.Open(dbPath, 0700, &bolt.Options{Timeout: 5 * time.Second})
|
return bolt.Open(dbPath, 0700, &bolt.Options{Timeout: 5 * time.Second})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *fileStore) list(db *bolt.DB, limit, offset uint) []string {
|
func (m *fileStore) list(db *bolt.DB, limit, offset uint, prefix, suffix string) []string {
|
||||||
var allItems []string
|
|
||||||
|
var keys []string
|
||||||
|
|
||||||
db.View(func(tx *bolt.Tx) error {
|
db.View(func(tx *bolt.Tx) error {
|
||||||
b := tx.Bucket([]byte(dataBucket))
|
b := tx.Bucket([]byte(dataBucket))
|
||||||
@ -120,54 +120,53 @@ func (m *fileStore) list(db *bolt.DB, limit, offset uint) []string {
|
|||||||
if b == nil {
|
if b == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
c := b.Cursor()
|
||||||
|
var k, v []byte
|
||||||
|
var cont func(k []byte) bool
|
||||||
|
|
||||||
// @todo very inefficient
|
if prefix != "" {
|
||||||
if err := b.ForEach(func(k, v []byte) error {
|
// for prefix we can speed up the search, not for suffix though :(
|
||||||
|
k, v = c.Seek([]byte(prefix))
|
||||||
|
cont = func(k []byte) bool {
|
||||||
|
return bytes.HasPrefix(k, []byte(prefix))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
k, v = c.First()
|
||||||
|
cont = func(k []byte) bool {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for ; k != nil && cont(k); k, v = c.Next() {
|
||||||
storedRecord := &record{}
|
storedRecord := &record{}
|
||||||
|
|
||||||
if err := json.Unmarshal(v, storedRecord); err != nil {
|
if err := json.Unmarshal(v, storedRecord); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if !storedRecord.ExpiresAt.IsZero() {
|
if !storedRecord.ExpiresAt.IsZero() {
|
||||||
if storedRecord.ExpiresAt.Before(time.Now()) {
|
if storedRecord.ExpiresAt.Before(time.Now()) {
|
||||||
return nil
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if suffix != "" && !bytes.HasSuffix(k, []byte(suffix)) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if offset > 0 {
|
||||||
|
offset--
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
keys = append(keys, string(k))
|
||||||
|
// this check still works if no limit was passed to begin with, you'll just end up with large -ve value
|
||||||
|
if limit == 1 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
limit--
|
||||||
|
|
||||||
allItems = append(allItems, string(k))
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
|
||||||
allKeys := make([]string, len(allItems))
|
return keys
|
||||||
|
|
||||||
for i, k := range allItems {
|
|
||||||
allKeys[i] = k
|
|
||||||
}
|
|
||||||
|
|
||||||
if limit != 0 || offset != 0 {
|
|
||||||
sort.Slice(allKeys, func(i, j int) bool { return allKeys[i] < allKeys[j] })
|
|
||||||
end := len(allKeys)
|
|
||||||
if limit > 0 {
|
|
||||||
calcLimit := int(offset + limit)
|
|
||||||
if calcLimit < end {
|
|
||||||
end = calcLimit
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if int(offset) >= end {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return allKeys[offset:end]
|
|
||||||
}
|
|
||||||
|
|
||||||
return allKeys
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *fileStore) get(db *bolt.DB, k string) (*store.Record, error) {
|
func (m *fileStore) get(db *bolt.DB, k string) (*store.Record, error) {
|
||||||
@ -283,21 +282,17 @@ func (m *fileStore) Read(key string, opts ...store.ReadOption) ([]*store.Record,
|
|||||||
var keys []string
|
var keys []string
|
||||||
|
|
||||||
// Handle Prefix / suffix
|
// Handle Prefix / suffix
|
||||||
// TODO: do range scan here rather than listing all keys
|
|
||||||
if readOpts.Prefix || readOpts.Suffix {
|
if readOpts.Prefix || readOpts.Suffix {
|
||||||
// list the keys
|
prefix := ""
|
||||||
k := m.list(db, readOpts.Limit, readOpts.Offset)
|
if readOpts.Prefix {
|
||||||
|
prefix = key
|
||||||
// check for prefix and suffix
|
|
||||||
for _, v := range k {
|
|
||||||
if readOpts.Prefix && !strings.HasPrefix(v, key) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if readOpts.Suffix && !strings.HasSuffix(v, key) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
keys = append(keys, v)
|
|
||||||
}
|
}
|
||||||
|
suffix := ""
|
||||||
|
if readOpts.Suffix {
|
||||||
|
suffix = key
|
||||||
|
}
|
||||||
|
// list the keys
|
||||||
|
keys = m.list(db, readOpts.Limit, readOpts.Offset, prefix, suffix)
|
||||||
} else {
|
} else {
|
||||||
keys = []string{key}
|
keys = []string{key}
|
||||||
}
|
}
|
||||||
@ -369,28 +364,7 @@ func (m *fileStore) List(opts ...store.ListOption) ([]string, error) {
|
|||||||
}
|
}
|
||||||
defer db.Close()
|
defer db.Close()
|
||||||
|
|
||||||
// TODO apply prefix/suffix in range query
|
allKeys := m.list(db, listOptions.Limit, listOptions.Offset, listOptions.Prefix, listOptions.Suffix)
|
||||||
allKeys := m.list(db, listOptions.Limit, listOptions.Offset)
|
|
||||||
|
|
||||||
if len(listOptions.Prefix) > 0 {
|
|
||||||
var prefixKeys []string
|
|
||||||
for _, k := range allKeys {
|
|
||||||
if strings.HasPrefix(k, listOptions.Prefix) {
|
|
||||||
prefixKeys = append(prefixKeys, k)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
allKeys = prefixKeys
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(listOptions.Suffix) > 0 {
|
|
||||||
var suffixKeys []string
|
|
||||||
for _, k := range allKeys {
|
|
||||||
if strings.HasSuffix(k, listOptions.Suffix) {
|
|
||||||
suffixKeys = append(suffixKeys, k)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
allKeys = suffixKeys
|
|
||||||
}
|
|
||||||
|
|
||||||
return allKeys, nil
|
return allKeys, nil
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ import (
|
|||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/micro/go-micro/v3/store"
|
"github.com/micro/go-micro/v3/store"
|
||||||
@ -19,7 +20,7 @@ func NewStore(opts ...store.Option) store.Store {
|
|||||||
Database: "micro",
|
Database: "micro",
|
||||||
Table: "micro",
|
Table: "micro",
|
||||||
},
|
},
|
||||||
store: cache.New(cache.NoExpiration, 5*time.Minute),
|
stores: map[string]*cache.Cache{}, // cache.New(cache.NoExpiration, 5*time.Minute),
|
||||||
}
|
}
|
||||||
for _, o := range opts {
|
for _, o := range opts {
|
||||||
o(&s.options)
|
o(&s.options)
|
||||||
@ -28,9 +29,10 @@ func NewStore(opts ...store.Option) store.Store {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type memoryStore struct {
|
type memoryStore struct {
|
||||||
|
sync.RWMutex
|
||||||
options store.Options
|
options store.Options
|
||||||
|
|
||||||
store *cache.Cache
|
stores map[string]*cache.Cache
|
||||||
}
|
}
|
||||||
|
|
||||||
type storeRecord struct {
|
type storeRecord struct {
|
||||||
@ -40,10 +42,6 @@ type storeRecord struct {
|
|||||||
expiresAt time.Time
|
expiresAt time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *memoryStore) key(prefix, key string) string {
|
|
||||||
return filepath.Join(prefix, key)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *memoryStore) prefix(database, table string) string {
|
func (m *memoryStore) prefix(database, table string) string {
|
||||||
if len(database) == 0 {
|
if len(database) == 0 {
|
||||||
database = m.options.Database
|
database = m.options.Database
|
||||||
@ -54,11 +52,24 @@ func (m *memoryStore) prefix(database, table string) string {
|
|||||||
return filepath.Join(database, table)
|
return filepath.Join(database, table)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *memoryStore) get(prefix, key string) (*store.Record, error) {
|
func (m *memoryStore) getStore(prefix string) *cache.Cache {
|
||||||
key = m.key(prefix, key)
|
m.RLock()
|
||||||
|
store := m.stores[prefix]
|
||||||
|
m.RUnlock()
|
||||||
|
if store == nil {
|
||||||
|
m.Lock()
|
||||||
|
if m.stores[prefix] == nil {
|
||||||
|
m.stores[prefix] = cache.New(cache.NoExpiration, 5*time.Minute)
|
||||||
|
}
|
||||||
|
store = m.stores[prefix]
|
||||||
|
m.Unlock()
|
||||||
|
}
|
||||||
|
return store
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *memoryStore) get(prefix, key string) (*store.Record, error) {
|
||||||
var storedRecord *storeRecord
|
var storedRecord *storeRecord
|
||||||
r, found := m.store.Get(key)
|
r, found := m.getStore(prefix).Get(key)
|
||||||
if !found {
|
if !found {
|
||||||
return nil, store.ErrNotFound
|
return nil, store.ErrNotFound
|
||||||
}
|
}
|
||||||
@ -91,8 +102,6 @@ func (m *memoryStore) get(prefix, key string) (*store.Record, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (m *memoryStore) set(prefix string, r *store.Record) {
|
func (m *memoryStore) set(prefix string, r *store.Record) {
|
||||||
key := m.key(prefix, r.Key)
|
|
||||||
|
|
||||||
// copy the incoming record and then
|
// copy the incoming record and then
|
||||||
// convert the expiry in to a hard timestamp
|
// convert the expiry in to a hard timestamp
|
||||||
i := &storeRecord{}
|
i := &storeRecord{}
|
||||||
@ -113,33 +122,54 @@ func (m *memoryStore) set(prefix string, r *store.Record) {
|
|||||||
i.metadata[k] = v
|
i.metadata[k] = v
|
||||||
}
|
}
|
||||||
|
|
||||||
m.store.Set(key, i, r.Expiry)
|
m.getStore(prefix).Set(r.Key, i, r.Expiry)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *memoryStore) delete(prefix, key string) {
|
func (m *memoryStore) delete(prefix, key string) {
|
||||||
key = m.key(prefix, key)
|
m.getStore(prefix).Delete(key)
|
||||||
m.store.Delete(key)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *memoryStore) list(prefix string, limit, offset uint) []string {
|
func (m *memoryStore) list(prefix string, limit, offset uint, prefixFilter, suffixFilter string) []string {
|
||||||
allItems := m.store.Items()
|
|
||||||
keys := make([]string, len(allItems))
|
|
||||||
i := 0
|
|
||||||
|
|
||||||
|
allItems := m.getStore(prefix).Items()
|
||||||
|
|
||||||
|
allKeys := make([]string, len(allItems))
|
||||||
|
|
||||||
|
// construct list of keys for this prefix
|
||||||
|
i := 0
|
||||||
for k := range allItems {
|
for k := range allItems {
|
||||||
if !strings.HasPrefix(k, prefix+"/") {
|
allKeys[i] = k
|
||||||
continue
|
|
||||||
}
|
|
||||||
keys[i] = strings.TrimPrefix(k, prefix+"/")
|
|
||||||
i++
|
i++
|
||||||
}
|
}
|
||||||
|
keys := make([]string, 0, len(allKeys))
|
||||||
sort.Slice(keys, func(i, j int) bool { return keys[i] < keys[j] })
|
sort.Slice(allKeys, func(i, j int) bool { return allKeys[i] < allKeys[j] })
|
||||||
return applyLimitAndOffset(keys, limit, offset)
|
for _, k := range allKeys {
|
||||||
|
if prefixFilter != "" && !strings.HasPrefix(k, prefixFilter) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if suffixFilter != "" && !strings.HasSuffix(k, suffixFilter) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if offset > 0 {
|
||||||
|
offset--
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
keys = append(keys, k)
|
||||||
|
// this check still works if no limit was passed to begin with, you'll just end up with large -ve value
|
||||||
|
if limit == 1 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
limit--
|
||||||
|
}
|
||||||
|
return keys
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *memoryStore) Close() error {
|
func (m *memoryStore) Close() error {
|
||||||
m.store.Flush()
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
|
for _, s := range m.stores {
|
||||||
|
s.Flush()
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -165,20 +195,15 @@ func (m *memoryStore) Read(key string, opts ...store.ReadOption) ([]*store.Recor
|
|||||||
var keys []string
|
var keys []string
|
||||||
// Handle Prefix / suffix
|
// Handle Prefix / suffix
|
||||||
if readOpts.Prefix || readOpts.Suffix {
|
if readOpts.Prefix || readOpts.Suffix {
|
||||||
// apply limit / offset once filtering is complete
|
prefixFilter := ""
|
||||||
for _, kk := range m.list(prefix, 0, 0) {
|
if readOpts.Prefix {
|
||||||
if readOpts.Prefix && !strings.HasPrefix(kk, key) {
|
prefixFilter = key
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if readOpts.Suffix && !strings.HasSuffix(kk, key) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
keys = append(keys, kk)
|
|
||||||
}
|
}
|
||||||
|
suffixFilter := ""
|
||||||
keys = applyLimitAndOffset(keys, readOpts.Limit, readOpts.Offset)
|
if readOpts.Suffix {
|
||||||
|
suffixFilter = key
|
||||||
|
}
|
||||||
|
keys = m.list(prefix, readOpts.Limit, readOpts.Offset, prefixFilter, suffixFilter)
|
||||||
} else {
|
} else {
|
||||||
keys = []string{key}
|
keys = []string{key}
|
||||||
}
|
}
|
||||||
@ -257,47 +282,6 @@ func (m *memoryStore) List(opts ...store.ListOption) ([]string, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
prefix := m.prefix(listOptions.Database, listOptions.Table)
|
prefix := m.prefix(listOptions.Database, listOptions.Table)
|
||||||
keys := m.list(prefix, listOptions.Limit, listOptions.Offset)
|
keys := m.list(prefix, listOptions.Limit, listOptions.Offset, listOptions.Prefix, listOptions.Suffix)
|
||||||
|
|
||||||
if len(listOptions.Prefix) > 0 {
|
|
||||||
var prefixKeys []string
|
|
||||||
for _, k := range keys {
|
|
||||||
if strings.HasPrefix(k, listOptions.Prefix) {
|
|
||||||
prefixKeys = append(prefixKeys, k)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
keys = prefixKeys
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(listOptions.Suffix) > 0 {
|
|
||||||
var suffixKeys []string
|
|
||||||
for _, k := range keys {
|
|
||||||
if strings.HasSuffix(k, listOptions.Suffix) {
|
|
||||||
suffixKeys = append(suffixKeys, k)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
keys = suffixKeys
|
|
||||||
}
|
|
||||||
|
|
||||||
return keys, nil
|
return keys, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func applyLimitAndOffset(keys []string, limit, offset uint) []string {
|
|
||||||
if limit == 0 && offset == 0 {
|
|
||||||
return keys
|
|
||||||
}
|
|
||||||
|
|
||||||
end := len(keys)
|
|
||||||
if limit > 0 {
|
|
||||||
calcLimit := int(offset + limit)
|
|
||||||
if calcLimit < end {
|
|
||||||
end = calcLimit
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if int(offset) >= end {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return keys[offset:end]
|
|
||||||
}
|
|
||||||
|
@ -264,6 +264,15 @@ func listTests(s store.Store, t *testing.T) {
|
|||||||
t.Fatalf("Expected 2 records, received %d %+v", len(recs), recs)
|
t.Fatalf("Expected 2 records, received %d %+v", len(recs), recs)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for i := 0; i < 10; i++ {
|
||||||
|
s.Write(&store.Record{Key: fmt.Sprintf("ListOffset%d", i), Value: []byte("bar")})
|
||||||
|
}
|
||||||
|
|
||||||
|
recs, err = s.List(store.ListPrefix("ListOffset"), store.ListOffset(6))
|
||||||
|
if len(recs) != 4 {
|
||||||
|
t.Fatalf("Expected 4 records, received %d %+v", len(recs), recs)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func expiryTests(s store.Store, t *testing.T) {
|
func expiryTests(s store.Store, t *testing.T) {
|
||||||
@ -306,7 +315,7 @@ func expiryTests(s store.Store, t *testing.T) {
|
|||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
if len(results) != 3 {
|
if len(results) != 3 {
|
||||||
t.Fatal("Results should have returned 3 records")
|
t.Fatalf("Results should have returned 3 records, returned %d", len(results))
|
||||||
}
|
}
|
||||||
time.Sleep(1 * time.Second)
|
time.Sleep(1 * time.Second)
|
||||||
results, err = s.Read("a", store.ReadPrefix())
|
results, err = s.Read("a", store.ReadPrefix())
|
||||||
|
Loading…
Reference in New Issue
Block a user