add redis3

This commit is contained in:
Chris Lu 2021-10-04 01:01:31 -07:00
parent ba7fbac07f
commit 366f522a2d
11 changed files with 296 additions and 48 deletions

View file

@ -29,6 +29,7 @@ import (
_ "github.com/chrislusf/seaweedfs/weed/filer/postgres2" _ "github.com/chrislusf/seaweedfs/weed/filer/postgres2"
_ "github.com/chrislusf/seaweedfs/weed/filer/redis" _ "github.com/chrislusf/seaweedfs/weed/filer/redis"
_ "github.com/chrislusf/seaweedfs/weed/filer/redis2" _ "github.com/chrislusf/seaweedfs/weed/filer/redis2"
_ "github.com/chrislusf/seaweedfs/weed/filer/redis3"
_ "github.com/chrislusf/seaweedfs/weed/filer/sqlite" _ "github.com/chrislusf/seaweedfs/weed/filer/sqlite"
_ "github.com/chrislusf/seaweedfs/weed/filer/tikv" _ "github.com/chrislusf/seaweedfs/weed/filer/tikv"
) )

5
weed/filer.toml Normal file
View file

@ -0,0 +1,5 @@
[redis3]
enabled = true
address = "localhost:6379"
password = ""
database = 0

View file

@ -3,11 +3,13 @@ package redis3
import ( import (
"context" "context"
"fmt" "fmt"
"github.com/chrislusf/seaweedfs/weed/util/bptree" "github.com/chrislusf/seaweedfs/weed/glog"
"github.com/chrislusf/seaweedfs/weed/util/skiplist"
"github.com/go-redis/redis/v8" "github.com/go-redis/redis/v8"
"github.com/golang/protobuf/proto"
) )
const maxNameBatchSizeLimit = 5
func insertChild(ctx context.Context, client redis.UniversalClient, key string, name string) error { func insertChild(ctx context.Context, client redis.UniversalClient, key string, name string) error {
data, err := client.Get(ctx, key).Result() data, err := client.Get(ctx, key).Result()
if err != nil { if err != nil {
@ -15,12 +17,22 @@ func insertChild(ctx context.Context, client redis.UniversalClient, key string,
return fmt.Errorf("read %s: %v", key, err) return fmt.Errorf("read %s: %v", key, err)
} }
} }
rootNode := &bptree.ProtoNode{} store := newSkipListElementStore(key, client)
if err := proto.UnmarshalMerge([]byte(data), rootNode); err != nil { nameList := skiplist.LoadNameList([]byte(data), store, maxNameBatchSizeLimit)
return fmt.Errorf("decoding root for %s: %v", key, err)
// println("add", key, name)
if err := nameList.WriteName(name); err != nil {
glog.Errorf("add %s %s: %v", key, name, err)
return err
} }
tree := rootNode.ToBpTree() if !nameList.HasChanges() {
tree.Add(bptree.String(name), nil) return nil
}
if err := client.Set(ctx, key, nameList.ToBytes(), 0).Err(); err != nil {
return err
}
return nil return nil
} }
@ -31,19 +43,69 @@ func removeChild(ctx context.Context, client redis.UniversalClient, key string,
return fmt.Errorf("read %s: %v", key, err) return fmt.Errorf("read %s: %v", key, err)
} }
} }
rootNode := &bptree.ProtoNode{} store := newSkipListElementStore(key, client)
if err := proto.UnmarshalMerge([]byte(data), rootNode); err != nil { nameList := skiplist.LoadNameList([]byte(data), store, maxNameBatchSizeLimit)
return fmt.Errorf("decoding root for %s: %v", key, err)
if err := nameList.DeleteName(name); err != nil {
return err
} }
tree := rootNode.ToBpTree() if !nameList.HasChanges() {
tree.Add(bptree.String(name), nil) return nil
}
if err := client.Set(ctx, key, nameList.ToBytes(), 0).Err(); err != nil {
return err
}
return nil return nil
} }
func removeChildren(ctx context.Context, client redis.UniversalClient, key string, onDeleteFn func(name string) error) error { func removeChildren(ctx context.Context, client redis.UniversalClient, key string, onDeleteFn func(name string) error) error {
data, err := client.Get(ctx, key).Result()
if err != nil {
if err != redis.Nil {
return fmt.Errorf("read %s: %v", key, err)
}
}
store := newSkipListElementStore(key, client)
nameList := skiplist.LoadNameList([]byte(data), store, maxNameBatchSizeLimit)
if err = nameList.ListNames("", func(name string) bool {
if err := onDeleteFn(name); err != nil {
glog.Errorf("delete %s child %s: %v", key, name, err)
return false
}
return true
}); err != nil {
return err
}
if err = nameList.RemoteAllListElement(); err != nil {
return err
}
return nil return nil
} }
func iterateChildren(ctx context.Context, client redis.UniversalClient, key string, eachFn func(name string) error) error { func listChildren(ctx context.Context, client redis.UniversalClient, key string, startFileName string, eachFn func(name string) bool) error {
data, err := client.Get(ctx, key).Result()
if err != nil {
if err != redis.Nil {
return fmt.Errorf("read %s: %v", key, err)
}
}
store := newSkipListElementStore(key, client)
nameList := skiplist.LoadNameList([]byte(data), store, maxNameBatchSizeLimit)
if err = nameList.ListNames(startFileName, func(name string) bool {
return eachFn(name)
}); err != nil {
return err
}
return nil return nil
} }

View file

@ -0,0 +1,52 @@
package redis3
import (
"context"
"fmt"
"github.com/chrislusf/seaweedfs/weed/glog"
"github.com/chrislusf/seaweedfs/weed/util/skiplist"
"github.com/go-redis/redis/v8"
"github.com/golang/protobuf/proto"
)
type SkipListElementStore struct {
prefix string
client redis.UniversalClient
}
var _ = skiplist.ListStore(&SkipListElementStore{})
func newSkipListElementStore(prefix string, client redis.UniversalClient) *SkipListElementStore {
return &SkipListElementStore{
prefix: prefix,
client: client,
}
}
func (m *SkipListElementStore) SaveElement(id int64, element *skiplist.SkipListElement) error {
key := fmt.Sprintf("%s%d", m.prefix, id)
data, err := proto.Marshal(element)
if err != nil {
glog.Errorf("marshal %s: %v", key, err)
}
return m.client.Set(context.Background(), key, data, 0).Err()
}
func (m *SkipListElementStore) DeleteElement(id int64) error {
key := fmt.Sprintf("%s%d", m.prefix, id)
return m.client.Del(context.Background(), key).Err()
}
func (m *SkipListElementStore) LoadElement(id int64) (*skiplist.SkipListElement, error) {
key := fmt.Sprintf("%s%d", m.prefix, id)
data, err := m.client.Get(context.Background(), key).Result()
if err != nil {
if err == redis.Nil {
return nil, nil
}
return nil, err
}
t := &skiplist.SkipListElement{}
err = proto.Unmarshal([]byte(data), t)
return t, err
}

View file

@ -115,6 +115,8 @@ func (store *UniversalRedis3Store) DeleteFolderChildren(ctx context.Context, ful
if err != nil { if err != nil {
return fmt.Errorf("DeleteFolderChildren %s in parent dir: %v", fullpath, err) return fmt.Errorf("DeleteFolderChildren %s in parent dir: %v", fullpath, err)
} }
// not efficient, but need to remove if it is a directory
store.Client.Del(ctx, genDirectoryListKey(string(path)))
return nil return nil
}) })
@ -127,41 +129,41 @@ func (store *UniversalRedis3Store) ListDirectoryPrefixedEntries(ctx context.Cont
func (store *UniversalRedis3Store) ListDirectoryEntries(ctx context.Context, dirPath util.FullPath, startFileName string, includeStartFile bool, limit int64, eachEntryFunc filer.ListEachEntryFunc) (lastFileName string, err error) { func (store *UniversalRedis3Store) ListDirectoryEntries(ctx context.Context, dirPath util.FullPath, startFileName string, includeStartFile bool, limit int64, eachEntryFunc filer.ListEachEntryFunc) (lastFileName string, err error) {
dirListKey := genDirectoryListKey(string(dirPath)) dirListKey := genDirectoryListKey(string(dirPath))
start := int64(0) counter := int64(0)
err = listChildren(ctx, store.Client, dirListKey, startFileName, func(fileName string) bool {
if startFileName != "" { if startFileName != "" {
start, _ = store.Client.ZRank(ctx, dirListKey, startFileName).Result() if !includeStartFile && startFileName == fileName {
if !includeStartFile { return true
start++
} }
} }
members, err := store.Client.ZRange(ctx, dirListKey, start, start+int64(limit)-1).Result()
if err != nil {
return lastFileName, fmt.Errorf("list %s : %v", dirPath, err)
}
// fetch entry meta
for _, fileName := range members {
path := util.NewFullPath(string(dirPath), fileName) path := util.NewFullPath(string(dirPath), fileName)
entry, err := store.FindEntry(ctx, path) entry, err := store.FindEntry(ctx, path)
lastFileName = fileName lastFileName = fileName
if err != nil { if err != nil {
glog.V(0).Infof("list %s : %v", path, err) glog.V(0).Infof("list %s : %v", path, err)
if err == filer_pb.ErrNotFound { if err == filer_pb.ErrNotFound {
continue return true
} }
} else { } else {
if entry.TtlSec > 0 { if entry.TtlSec > 0 {
if entry.Attr.Crtime.Add(time.Duration(entry.TtlSec) * time.Second).Before(time.Now()) { if entry.Attr.Crtime.Add(time.Duration(entry.TtlSec) * time.Second).Before(time.Now()) {
store.Client.Del(ctx, string(path)).Result() store.Client.Del(ctx, string(path)).Result()
store.Client.ZRem(ctx, dirListKey, fileName).Result() store.Client.ZRem(ctx, dirListKey, fileName).Result()
continue return true
} }
} }
counter++
if !eachEntryFunc(entry) { if !eachEntryFunc(entry) {
break return false
} }
if counter >= limit {
return false
} }
} }
return true
})
return lastFileName, err return lastFileName, err
} }

View file

@ -34,6 +34,7 @@ import (
_ "github.com/chrislusf/seaweedfs/weed/filer/postgres2" _ "github.com/chrislusf/seaweedfs/weed/filer/postgres2"
_ "github.com/chrislusf/seaweedfs/weed/filer/redis" _ "github.com/chrislusf/seaweedfs/weed/filer/redis"
_ "github.com/chrislusf/seaweedfs/weed/filer/redis2" _ "github.com/chrislusf/seaweedfs/weed/filer/redis2"
_ "github.com/chrislusf/seaweedfs/weed/filer/redis3"
_ "github.com/chrislusf/seaweedfs/weed/filer/sqlite" _ "github.com/chrislusf/seaweedfs/weed/filer/sqlite"
"github.com/chrislusf/seaweedfs/weed/glog" "github.com/chrislusf/seaweedfs/weed/glog"
"github.com/chrislusf/seaweedfs/weed/notification" "github.com/chrislusf/seaweedfs/weed/notification"

View file

@ -35,7 +35,7 @@ func (nb *NameBatch) DeleteName(name string) {
} }
func (nb *NameBatch) ListNames(startFrom string, visitNamesFn func(name string) bool) bool { func (nb *NameBatch) ListNames(startFrom string, visitNamesFn func(name string) bool) bool {
var names []string var names []string
needFilter := startFrom == "" needFilter := startFrom != ""
for n := range nb.names { for n := range nb.names {
if !needFilter || strings.Compare(n, startFrom) >= 0 { if !needFilter || strings.Compare(n, startFrom) >= 0 {
names = append(names, n) names = append(names, n)

View file

@ -9,7 +9,7 @@ type NameList struct {
batchSize int batchSize int
} }
func NewNameList(store ListStore, batchSize int) *NameList { func newNameList(store ListStore, batchSize int) *NameList {
return &NameList{ return &NameList{
skipList: New(store), skipList: New(store),
batchSize: batchSize, batchSize: batchSize,
@ -59,6 +59,7 @@ There are multiple cases after finding the name for greater or equal node
*/ */
func (nl *NameList) WriteName(name string) error { func (nl *NameList) WriteName(name string) error {
lookupKey := []byte(name) lookupKey := []byte(name)
prevNode, nextNode, found, err := nl.skipList.FindGreaterOrEqual(lookupKey) prevNode, nextNode, found, err := nl.skipList.FindGreaterOrEqual(lookupKey)
if err != nil { if err != nil {
@ -301,3 +302,25 @@ func (nl *NameList) ListNames(startFrom string, visitNamesFn func(name string) b
return nil return nil
} }
func (nl *NameList) RemoteAllListElement() error {
t := nl.skipList
nodeRef := t.startLevels[0]
for nodeRef != nil {
node, err := t.loadElement(nodeRef)
if err != nil {
return err
}
if node == nil {
return nil
}
if err := t.deleteElement(node); err != nil {
return err
}
nodeRef = node.Next[0]
}
return nil
}

View file

@ -0,0 +1,71 @@
package skiplist
import (
"github.com/chrislusf/seaweedfs/weed/glog"
"github.com/golang/protobuf/proto"
)
func LoadNameList(data []byte, store ListStore, batchSize int) *NameList {
nl := &NameList{
skipList: New(store),
batchSize: batchSize,
}
if len(data) == 0 {
return nl
}
message := &SkipListProto{}
if err := proto.Unmarshal(data, message); err != nil {
glog.Errorf("loading skiplist: %v", err)
}
nl.skipList.maxNewLevel = int(message.MaxNewLevel)
nl.skipList.maxLevel = int(message.MaxLevel)
for i, ref := range message.StartLevels {
nl.skipList.startLevels[i] = &SkipListElementReference{
ElementPointer: ref.ElementPointer,
Key: ref.Key,
}
}
for i, ref := range message.EndLevels {
nl.skipList.endLevels[i] = &SkipListElementReference{
ElementPointer: ref.ElementPointer,
Key: ref.Key,
}
}
return nl
}
func (nl *NameList) HasChanges() bool {
return nl.skipList.hasChanges
}
func (nl *NameList) ToBytes() []byte {
message := &SkipListProto{}
message.MaxNewLevel = int32(nl.skipList.maxNewLevel)
message.MaxLevel = int32(nl.skipList.maxLevel)
for _, ref := range nl.skipList.startLevels {
if ref == nil {
break
}
message.StartLevels = append(message.StartLevels, &SkipListElementReference{
ElementPointer: ref.ElementPointer,
Key: ref.Key,
})
}
for _, ref := range nl.skipList.endLevels {
if ref == nil {
break
}
message.EndLevels = append(message.EndLevels, &SkipListElementReference{
ElementPointer: ref.ElementPointer,
Key: ref.Key,
})
}
data, err := proto.Marshal(message)
if err != nil {
glog.Errorf("marshal skiplist: %v", err)
}
return data
}

View file

@ -15,7 +15,7 @@ func String(x int) string {
} }
func TestNameList(t *testing.T) { func TestNameList(t *testing.T) {
list := NewNameList(memStore, 7) list := newNameList(memStore, 7)
for i := 0; i < maxNameCount; i++ { for i := 0; i < maxNameCount; i++ {
list.WriteName(String(i)) list.WriteName(String(i))
@ -51,7 +51,7 @@ func TestNameList(t *testing.T) {
} }
// randomized deletion // randomized deletion
list = NewNameList(memStore, 7) list = newNameList(memStore, 7)
// Delete elements at random positions in the list. // Delete elements at random positions in the list.
rList := rand.Perm(maxN) rList := rand.Perm(maxN)
for _, i := range rList { for _, i := range rList {

View file

@ -22,6 +22,7 @@ type SkipList struct {
maxNewLevel int maxNewLevel int
maxLevel int maxLevel int
listStore ListStore listStore ListStore
hasChanges bool
// elementCount int // elementCount int
} }
@ -93,6 +94,9 @@ func (t *SkipList) findExtended(key []byte, findGreaterOrEqual bool) (prevElemen
if err != nil { if err != nil {
return return
} }
if currentNode == nil {
return
}
// In case, that our first element is already greater-or-equal! // In case, that our first element is already greater-or-equal!
if findGreaterOrEqual && compareElement(currentNode, key) > 0 { if findGreaterOrEqual && compareElement(currentNode, key) > 0 {
@ -115,6 +119,9 @@ func (t *SkipList) findExtended(key []byte, findGreaterOrEqual bool) (prevElemen
if err != nil { if err != nil {
return return
} }
if currentNode == nil {
return
}
} else { } else {
if index > 0 { if index > 0 {
@ -126,6 +133,9 @@ func (t *SkipList) findExtended(key []byte, findGreaterOrEqual bool) (prevElemen
if err != nil { if err != nil {
return return
} }
if currentNodeNext == nil {
return
}
foundElem = currentNodeNext foundElem = currentNodeNext
ok = true ok = true
return return
@ -216,11 +226,13 @@ func (t *SkipList) Delete(key []byte) (err error) {
if err != nil { if err != nil {
return err return err
} }
if nextNextNode != nil {
nextNextNode.Prev = currentNode.Reference() nextNextNode.Prev = currentNode.Reference()
if err = t.saveElement(nextNextNode); err != nil { if err = t.saveElement(nextNextNode); err != nil {
return err return err
} }
} }
}
// t.elementCount-- // t.elementCount--
if err = t.deleteElement(nextNode); err != nil { if err = t.deleteElement(nextNode); err != nil {
return err return err
@ -230,6 +242,7 @@ func (t *SkipList) Delete(key []byte) (err error) {
// Link from start needs readjustments. // Link from start needs readjustments.
startNextKey := t.startLevels[index].Key startNextKey := t.startLevels[index].Key
if compareElement(nextNode, startNextKey) == 0 { if compareElement(nextNode, startNextKey) == 0 {
t.hasChanges = true
t.startLevels[index] = nextNode.Next[index] t.startLevels[index] = nextNode.Next[index]
// This was our currently highest node! // This was our currently highest node!
if t.startLevels[index] == nil { if t.startLevels[index] == nil {
@ -240,6 +253,7 @@ func (t *SkipList) Delete(key []byte) (err error) {
// Link from end needs readjustments. // Link from end needs readjustments.
if nextNode.Next[index] == nil { if nextNode.Next[index] == nil {
t.endLevels[index] = currentNode.Reference() t.endLevels[index] = currentNode.Reference()
t.hasChanges = true
} }
nextNode.Next[index] = nil nextNode.Next[index] = nil
} }
@ -260,7 +274,7 @@ func (t *SkipList) Delete(key []byte) (err error) {
// Insert inserts the given ListElement into the skiplist. // Insert inserts the given ListElement into the skiplist.
// Insert runs in approx. O(log(n)) // Insert runs in approx. O(log(n))
func (t *SkipList) Insert(key, value []byte) (err error){ func (t *SkipList) Insert(key, value []byte) (err error) {
if t == nil || key == nil { if t == nil || key == nil {
return return
@ -272,6 +286,7 @@ func (t *SkipList) Insert(key, value []byte) (err error){
if level > t.maxLevel { if level > t.maxLevel {
level = t.maxLevel + 1 level = t.maxLevel + 1
t.maxLevel = level t.maxLevel = level
t.hasChanges = true
} }
elem := &SkipListElement{ elem := &SkipListElement{
@ -326,6 +341,7 @@ func (t *SkipList) Insert(key, value []byte) (err error){
if nextNode, err = t.loadElement(nextNodeRef); err != nil { if nextNode, err = t.loadElement(nextNodeRef); err != nil {
return return
} }
if nextNode != nil {
nextNode.Prev = elem.Reference() nextNode.Prev = elem.Reference()
if err = t.saveElement(nextNode); err != nil { if err = t.saveElement(nextNode); err != nil {
return return
@ -333,6 +349,7 @@ func (t *SkipList) Insert(key, value []byte) (err error){
} }
} }
} }
}
if nextNodeRef != nil && bytes.Compare(nextNodeRef.Key, key) <= 0 { if nextNodeRef != nil && bytes.Compare(nextNodeRef.Key, key) <= 0 {
// Go right // Go right
@ -343,6 +360,9 @@ func (t *SkipList) Insert(key, value []byte) (err error){
} }
} }
currentNode = nextNode currentNode = nextNode
if currentNode == nil {
return
}
} else { } else {
// Go down // Go down
index-- index--
@ -366,18 +386,22 @@ func (t *SkipList) Insert(key, value []byte) (err error){
if err != nil { if err != nil {
return err return err
} }
if startLevelElement != nil {
startLevelElement.Prev = elem.Reference() startLevelElement.Prev = elem.Reference()
if err = t.saveElement(startLevelElement); err != nil { if err = t.saveElement(startLevelElement); err != nil {
return err return err
} }
} }
}
elem.Next[i] = t.startLevels[i] elem.Next[i] = t.startLevels[i]
t.startLevels[i] = elem.Reference() t.startLevels[i] = elem.Reference()
t.hasChanges = true
} }
// link the endLevels to this element! // link the endLevels to this element!
if elem.Next[i] == nil { if elem.Next[i] == nil {
t.endLevels[i] = elem.Reference() t.endLevels[i] = elem.Reference()
t.hasChanges = true
} }
didSomething = true didSomething = true
@ -392,20 +416,24 @@ func (t *SkipList) Insert(key, value []byte) (err error){
if err != nil { if err != nil {
return err return err
} }
if endLevelElement != nil {
endLevelElement.Next[i] = elem.Reference() endLevelElement.Next[i] = elem.Reference()
if err = t.saveElement(endLevelElement); err != nil { if err = t.saveElement(endLevelElement); err != nil {
return err return err
} }
} }
}
if i == 0 { if i == 0 {
elem.Prev = t.endLevels[i] elem.Prev = t.endLevels[i]
} }
t.endLevels[i] = elem.Reference() t.endLevels[i] = elem.Reference()
t.hasChanges = true
} }
// Link the startLevels to this element! // Link the startLevels to this element!
if t.startLevels[i] == nil || bytes.Compare(t.startLevels[i].Key, key) > 0 { if t.startLevels[i] == nil || bytes.Compare(t.startLevels[i].Key, key) > 0 {
t.startLevels[i] = elem.Reference() t.startLevels[i] = elem.Reference()
t.hasChanges = true
} }
didSomething = true didSomething = true
@ -486,6 +514,9 @@ func (t *SkipList) println() {
for nodeRef != nil { for nodeRef != nil {
print(fmt.Sprintf("%v: ", string(nodeRef.Key))) print(fmt.Sprintf("%v: ", string(nodeRef.Key)))
node, _ := t.loadElement(nodeRef) node, _ := t.loadElement(nodeRef)
if node == nil {
break
}
for i := 0; i <= int(node.Level); i++ { for i := 0; i <= int(node.Level); i++ {
l := node.Next[i] l := node.Next[i]
@ -510,8 +541,8 @@ func (t *SkipList) println() {
} }
} }
println()
nodeRef = node.Next[0] nodeRef = node.Next[0]
println()
} }
print("end --> ") print("end --> ")