bootstrap filer from one peer
This commit is contained in:
84
weed/server/filer_grpc_server_traverse_meta.go
Normal file
84
weed/server/filer_grpc_server_traverse_meta.go
Normal file
@@ -0,0 +1,84 @@
|
||||
package weed_server
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"github.com/seaweedfs/seaweedfs/weed/filer"
|
||||
"github.com/seaweedfs/seaweedfs/weed/glog"
|
||||
"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
|
||||
"github.com/seaweedfs/seaweedfs/weed/util"
|
||||
"github.com/viant/ptrie"
|
||||
)
|
||||
|
||||
func (fs *FilerServer) TraverseBfsMetadata(req *filer_pb.TraverseBfsMetadataRequest, stream filer_pb.SeaweedFiler_TraverseBfsMetadataServer) error {
|
||||
|
||||
glog.V(0).Infof("TraverseBfsMetadata %v", req)
|
||||
|
||||
excludedTrie := ptrie.New[bool]()
|
||||
for _, excluded := range req.ExcludedPrefixes {
|
||||
excludedTrie.Put([]byte(excluded), true)
|
||||
}
|
||||
|
||||
ctx := stream.Context()
|
||||
|
||||
queue := util.NewQueue[*filer.Entry]()
|
||||
dirEntry, err := fs.filer.FindEntry(ctx, util.FullPath(req.Directory))
|
||||
if err != nil {
|
||||
return fmt.Errorf("find dir %s: %v", req.Directory, err)
|
||||
}
|
||||
queue.Enqueue(dirEntry)
|
||||
|
||||
for item := queue.Dequeue(); item != nil; item = queue.Dequeue() {
|
||||
if excludedTrie.MatchPrefix([]byte(item.FullPath), func(key []byte, value bool) bool {
|
||||
return true
|
||||
}) {
|
||||
// println("excluded", item.FullPath)
|
||||
continue
|
||||
}
|
||||
parent, _ := item.FullPath.DirAndName()
|
||||
if err := stream.Send(&filer_pb.TraverseBfsMetadataResponse{
|
||||
Directory: parent,
|
||||
Entry: item.ToProtoEntry(),
|
||||
}); err != nil {
|
||||
return fmt.Errorf("send traverse bfs metadata response: %v", err)
|
||||
}
|
||||
|
||||
if !item.IsDirectory() {
|
||||
continue
|
||||
}
|
||||
|
||||
if err := fs.iterateDirectory(ctx, item.FullPath, func(entry *filer.Entry) error {
|
||||
queue.Enqueue(entry)
|
||||
return nil
|
||||
}); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (fs *FilerServer) iterateDirectory(ctx context.Context, dirPath util.FullPath, fn func(entry *filer.Entry) error) (err error) {
|
||||
var lastFileName string
|
||||
var listErr error
|
||||
for {
|
||||
var hasEntries bool
|
||||
lastFileName, listErr = fs.filer.StreamListDirectoryEntries(ctx, dirPath, lastFileName, false, 1024, "", "", "", func(entry *filer.Entry) bool {
|
||||
hasEntries = true
|
||||
if fnErr := fn(entry); fnErr != nil {
|
||||
err = fnErr
|
||||
return false
|
||||
}
|
||||
return true
|
||||
})
|
||||
if listErr != nil {
|
||||
return listErr
|
||||
}
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if !hasEntries {
|
||||
return nil
|
||||
}
|
||||
}
|
||||
}
|
||||
31
weed/server/filer_grpc_server_traverse_meta_test.go
Normal file
31
weed/server/filer_grpc_server_traverse_meta_test.go
Normal file
@@ -0,0 +1,31 @@
|
||||
package weed_server
|
||||
|
||||
import (
|
||||
"github.com/stretchr/testify/assert"
|
||||
"github.com/viant/ptrie"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestPtrie(t *testing.T) {
|
||||
b := []byte("/topics/abc/dev")
|
||||
excludedTrie := ptrie.New[bool]()
|
||||
excludedTrie.Put([]byte("/topics/abc/d"), true)
|
||||
excludedTrie.Put([]byte("/topics/abc"), true)
|
||||
|
||||
assert.True(t, excludedTrie.MatchPrefix(b, func(key []byte, value bool) bool {
|
||||
println("matched1", string(key))
|
||||
return true
|
||||
}))
|
||||
|
||||
assert.True(t, excludedTrie.MatchAll(b, func(key []byte, value bool) bool {
|
||||
println("matched2", string(key))
|
||||
return true
|
||||
}))
|
||||
|
||||
assert.False(t, excludedTrie.MatchAll([]byte("/topics/ab"), func(key []byte, value bool) bool {
|
||||
println("matched3", string(key))
|
||||
return true
|
||||
}))
|
||||
|
||||
assert.False(t, excludedTrie.Has(b))
|
||||
}
|
||||
@@ -203,7 +203,7 @@ func NewFilerServer(defaultMux, readonlyMux *http.ServeMux, option *FilerOption)
|
||||
}
|
||||
if isFresh {
|
||||
glog.V(0).Infof("%s bootstrap from peers %+v", option.Host, existingNodes)
|
||||
if err := fs.filer.MaybeBootstrapFromPeers(option.Host, existingNodes, startFromTime); err != nil {
|
||||
if err := fs.filer.MaybeBootstrapFromOnePeer(option.Host, existingNodes, startFromTime); err != nil {
|
||||
glog.Fatalf("%s bootstrap from %+v: %v", option.Host, existingNodes, err)
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user