2020-09-01 07:21:19 +00:00
|
|
|
package filer
|
2019-12-13 08:23:05 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/glog"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
|
2020-02-25 06:28:45 +00:00
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/master_pb"
|
2020-03-23 07:01:34 +00:00
|
|
|
"github.com/chrislusf/seaweedfs/weed/util"
|
2019-12-13 08:23:05 +00:00
|
|
|
)
|
|
|
|
|
2020-09-24 18:11:42 +00:00
|
|
|
type HardLinkId []byte
|
2020-09-24 10:06:44 +00:00
|
|
|
|
2021-03-10 14:41:35 +00:00
|
|
|
const (
|
|
|
|
MsgFailDelNonEmptyFolder = "fail to delete non-empty folder"
|
|
|
|
)
|
|
|
|
|
2021-11-02 08:04:50 +00:00
|
|
|
type OnChunksFunc func([]*filer_pb.FileChunk) error
|
|
|
|
type OnHardLinkIdsFunc func([]HardLinkId) error
|
|
|
|
|
2020-08-29 06:48:48 +00:00
|
|
|
func (f *Filer) DeleteEntryMetaAndData(ctx context.Context, p util.FullPath, isRecursive, ignoreRecursiveError, shouldDeleteChunks, isFromOtherCluster bool, signatures []int32) (err error) {
|
2019-12-13 08:23:05 +00:00
|
|
|
if p == "/" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
entry, findErr := f.FindEntry(ctx, p)
|
|
|
|
if findErr != nil {
|
|
|
|
return findErr
|
|
|
|
}
|
|
|
|
|
2020-11-26 20:21:58 +00:00
|
|
|
isDeleteCollection := f.isBucket(entry)
|
2020-02-25 06:28:45 +00:00
|
|
|
|
2019-12-13 08:23:05 +00:00
|
|
|
if entry.IsDirectory() {
|
|
|
|
// delete the folder children, not including the folder itself
|
2021-11-02 08:04:50 +00:00
|
|
|
err = f.doBatchDeleteFolderMetaAndData(ctx, entry, isRecursive, ignoreRecursiveError, shouldDeleteChunks && !isDeleteCollection, isDeleteCollection, isFromOtherCluster, signatures, func(chunks []*filer_pb.FileChunk) error {
|
|
|
|
if shouldDeleteChunks && !isDeleteCollection {
|
|
|
|
f.DirectDeleteChunks(chunks)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}, func(hardLinkIds []HardLinkId) error {
|
|
|
|
// A case not handled:
|
|
|
|
// what if the chunk is in a different collection?
|
|
|
|
if shouldDeleteChunks {
|
|
|
|
f.maybeDeleteHardLinks(hardLinkIds)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2019-12-13 08:23:05 +00:00
|
|
|
if err != nil {
|
2020-02-25 06:28:45 +00:00
|
|
|
glog.V(0).Infof("delete directory %s: %v", p, err)
|
2019-12-13 08:23:05 +00:00
|
|
|
return fmt.Errorf("delete directory %s: %v", p, err)
|
|
|
|
}
|
2021-11-02 08:04:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if shouldDeleteChunks && !isDeleteCollection {
|
|
|
|
f.DirectDeleteChunks(entry.Chunks)
|
2019-12-13 08:23:05 +00:00
|
|
|
}
|
2020-02-25 06:28:45 +00:00
|
|
|
|
2019-12-13 08:23:05 +00:00
|
|
|
// delete the file or folder
|
2020-08-29 06:48:48 +00:00
|
|
|
err = f.doDeleteEntryMetaAndData(ctx, entry, shouldDeleteChunks, isFromOtherCluster, signatures)
|
2019-12-13 08:23:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("delete file %s: %v", p, err)
|
|
|
|
}
|
|
|
|
|
2020-11-26 20:21:58 +00:00
|
|
|
if isDeleteCollection {
|
2020-02-25 06:28:45 +00:00
|
|
|
collectionName := entry.Name()
|
2020-02-26 05:50:12 +00:00
|
|
|
f.doDeleteCollection(collectionName)
|
2020-02-25 06:28:45 +00:00
|
|
|
f.deleteBucket(collectionName)
|
|
|
|
}
|
2019-12-13 08:23:05 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-11-03 06:39:16 +00:00
|
|
|
func (f *Filer) doBatchDeleteFolderMetaAndData(ctx context.Context, entry *Entry, isRecursive, ignoreRecursiveError, shouldDeleteChunks, isDeletingBucket, isFromOtherCluster bool, signatures []int32, onChunksFn OnChunksFunc, onHardLinkIdsFn OnHardLinkIdsFunc) (err error) {
|
2019-12-13 08:23:05 +00:00
|
|
|
|
|
|
|
lastFileName := ""
|
|
|
|
includeLastFile := false
|
2021-06-11 06:37:54 +00:00
|
|
|
if !isDeletingBucket || !f.Store.CanDropWholeBucket() {
|
2021-01-13 21:49:04 +00:00
|
|
|
for {
|
2021-04-24 18:49:03 +00:00
|
|
|
entries, _, err := f.ListDirectoryEntries(ctx, entry.FullPath, lastFileName, includeLastFile, PaginationSize, "", "", "")
|
2021-01-13 21:49:04 +00:00
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("list folder %s: %v", entry.FullPath, err)
|
2021-11-02 08:04:50 +00:00
|
|
|
return fmt.Errorf("list folder %s: %v", entry.FullPath, err)
|
2021-01-13 21:49:04 +00:00
|
|
|
}
|
|
|
|
if lastFileName == "" && !isRecursive && len(entries) > 0 {
|
|
|
|
// only for first iteration in the loop
|
2021-10-29 21:14:29 +00:00
|
|
|
glog.V(0).Infof("deleting a folder %s has children: %+v ...", entry.FullPath, entries[0].Name())
|
2021-11-02 08:04:50 +00:00
|
|
|
return fmt.Errorf("%s: %s", MsgFailDelNonEmptyFolder, entry.FullPath)
|
2021-01-13 21:49:04 +00:00
|
|
|
}
|
2019-12-13 08:23:05 +00:00
|
|
|
|
2021-01-13 21:49:04 +00:00
|
|
|
for _, sub := range entries {
|
|
|
|
lastFileName = sub.Name()
|
|
|
|
if sub.IsDirectory() {
|
|
|
|
subIsDeletingBucket := f.isBucket(sub)
|
2021-11-02 08:04:50 +00:00
|
|
|
err = f.doBatchDeleteFolderMetaAndData(ctx, sub, isRecursive, ignoreRecursiveError, shouldDeleteChunks, subIsDeletingBucket, false, nil, onChunksFn, onHardLinkIdsFn)
|
2020-09-24 10:06:44 +00:00
|
|
|
} else {
|
2021-01-13 21:49:04 +00:00
|
|
|
f.NotifyUpdateEvent(ctx, sub, nil, shouldDeleteChunks, isFromOtherCluster, nil)
|
|
|
|
if len(sub.HardLinkId) != 0 {
|
|
|
|
// hard link chunk data are deleted separately
|
2021-11-02 08:04:50 +00:00
|
|
|
err = onHardLinkIdsFn([]HardLinkId{sub.HardLinkId})
|
2021-01-13 21:49:04 +00:00
|
|
|
} else {
|
2021-11-02 08:04:50 +00:00
|
|
|
err = onChunksFn(sub.Chunks)
|
2021-01-13 21:49:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if err != nil && !ignoreRecursiveError {
|
2021-11-02 08:04:50 +00:00
|
|
|
return err
|
2020-09-24 10:06:44 +00:00
|
|
|
}
|
2019-12-13 08:23:05 +00:00
|
|
|
}
|
|
|
|
|
2021-01-13 21:49:04 +00:00
|
|
|
if len(entries) < PaginationSize {
|
|
|
|
break
|
|
|
|
}
|
2019-12-13 08:23:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-02 08:04:50 +00:00
|
|
|
glog.V(3).Infof("deleting directory %v delete chunks: %v", entry.FullPath, shouldDeleteChunks)
|
2019-12-13 08:23:05 +00:00
|
|
|
|
2021-07-22 15:23:20 +00:00
|
|
|
if storeDeletionErr := f.Store.DeleteFolderChildren(ctx, entry.FullPath); storeDeletionErr != nil {
|
2021-11-02 08:04:50 +00:00
|
|
|
return fmt.Errorf("filer store delete: %v", storeDeletionErr)
|
2019-12-13 08:23:05 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 18:21:23 +00:00
|
|
|
f.NotifyUpdateEvent(ctx, entry, nil, shouldDeleteChunks, isFromOtherCluster, signatures)
|
2020-07-13 00:32:19 +00:00
|
|
|
|
2021-11-02 08:04:50 +00:00
|
|
|
return nil
|
2019-12-13 08:23:05 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 06:48:48 +00:00
|
|
|
func (f *Filer) doDeleteEntryMetaAndData(ctx context.Context, entry *Entry, shouldDeleteChunks bool, isFromOtherCluster bool, signatures []int32) (err error) {
|
2019-12-13 08:23:05 +00:00
|
|
|
|
2020-02-12 06:54:10 +00:00
|
|
|
glog.V(3).Infof("deleting entry %v, delete chunks: %v", entry.FullPath, shouldDeleteChunks)
|
2019-12-13 08:23:05 +00:00
|
|
|
|
2020-11-26 19:14:56 +00:00
|
|
|
if storeDeletionErr := f.Store.DeleteOneEntry(ctx, entry); storeDeletionErr != nil {
|
2019-12-13 08:23:05 +00:00
|
|
|
return fmt.Errorf("filer store delete: %v", storeDeletionErr)
|
|
|
|
}
|
2020-08-12 20:11:04 +00:00
|
|
|
if !entry.IsDirectory() {
|
2020-08-29 06:48:48 +00:00
|
|
|
f.NotifyUpdateEvent(ctx, entry, nil, shouldDeleteChunks, isFromOtherCluster, signatures)
|
2020-04-05 22:03:25 +00:00
|
|
|
}
|
2019-12-13 08:23:05 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2020-02-25 06:28:45 +00:00
|
|
|
|
2020-02-26 05:50:12 +00:00
|
|
|
func (f *Filer) doDeleteCollection(collectionName string) (err error) {
|
2020-02-25 06:28:45 +00:00
|
|
|
|
2020-02-26 05:50:12 +00:00
|
|
|
return f.MasterClient.WithClient(func(client master_pb.SeaweedClient) error {
|
|
|
|
_, err := client.CollectionDelete(context.Background(), &master_pb.CollectionDeleteRequest{
|
2020-02-25 06:28:45 +00:00
|
|
|
Name: collectionName,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
glog.Infof("delete collection %s: %v", collectionName, err)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
2020-11-26 19:25:56 +00:00
|
|
|
|
|
|
|
func (f *Filer) maybeDeleteHardLinks(hardLinkIds []HardLinkId) {
|
|
|
|
for _, hardLinkId := range hardLinkIds {
|
|
|
|
if err := f.Store.DeleteHardLink(context.Background(), hardLinkId); err != nil {
|
|
|
|
glog.Errorf("delete hard link id %d : %v", hardLinkId, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|