123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384 |
- 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
- }
- }
- }
|