2020-09-24 16:43:00 +00:00
|
|
|
package filer
|
|
|
|
|
|
|
|
import (
|
2020-09-24 18:11:42 +00:00
|
|
|
"bytes"
|
2020-09-24 16:43:00 +00:00
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/glog"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (fsw *FilerStoreWrapper) handleUpdateToHardLinks(ctx context.Context, entry *Entry) error {
|
2020-09-24 18:11:42 +00:00
|
|
|
if len(entry.HardLinkId) == 0 {
|
2020-09-24 16:43:00 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// handle hard links
|
|
|
|
if err := fsw.setHardLink(ctx, entry); err != nil {
|
|
|
|
return fmt.Errorf("setHardLink %d: %v", entry.HardLinkId, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// check what is existing entry
|
2020-12-11 03:55:28 +00:00
|
|
|
glog.V(4).Infof("handleUpdateToHardLinks FindEntry %s", entry.FullPath)
|
2020-12-22 05:46:32 +00:00
|
|
|
actualStore := fsw.getActualStore(entry.FullPath)
|
|
|
|
existingEntry, err := actualStore.FindEntry(ctx, entry.FullPath)
|
2020-09-24 16:43:00 +00:00
|
|
|
if err != nil && err != filer_pb.ErrNotFound {
|
|
|
|
return fmt.Errorf("update existing entry %s: %v", entry.FullPath, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove old hard link
|
2020-09-24 18:16:43 +00:00
|
|
|
if err == nil && len(existingEntry.HardLinkId) != 0 && bytes.Compare(existingEntry.HardLinkId, entry.HardLinkId) != 0 {
|
2020-12-11 03:55:28 +00:00
|
|
|
glog.V(4).Infof("handleUpdateToHardLinks DeleteHardLink %s", entry.FullPath)
|
2020-09-24 16:43:00 +00:00
|
|
|
if err = fsw.DeleteHardLink(ctx, existingEntry.HardLinkId); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fsw *FilerStoreWrapper) setHardLink(ctx context.Context, entry *Entry) error {
|
2020-09-24 18:11:42 +00:00
|
|
|
if len(entry.HardLinkId) == 0 {
|
2020-09-24 16:43:00 +00:00
|
|
|
return nil
|
|
|
|
}
|
2020-09-24 18:11:42 +00:00
|
|
|
key := entry.HardLinkId
|
2020-09-24 16:43:00 +00:00
|
|
|
|
|
|
|
newBlob, encodeErr := entry.EncodeAttributesAndChunks()
|
|
|
|
if encodeErr != nil {
|
|
|
|
return encodeErr
|
|
|
|
}
|
|
|
|
|
|
|
|
return fsw.KvPut(ctx, key, newBlob)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fsw *FilerStoreWrapper) maybeReadHardLink(ctx context.Context, entry *Entry) error {
|
2020-09-24 18:11:42 +00:00
|
|
|
if len(entry.HardLinkId) == 0 {
|
2020-09-24 16:43:00 +00:00
|
|
|
return nil
|
|
|
|
}
|
2020-09-24 18:11:42 +00:00
|
|
|
key := entry.HardLinkId
|
2020-09-24 16:43:00 +00:00
|
|
|
|
2020-12-11 03:55:28 +00:00
|
|
|
glog.V(4).Infof("maybeReadHardLink KvGet %v", key)
|
2020-09-24 16:43:00 +00:00
|
|
|
value, err := fsw.KvGet(ctx, key)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("read %s hardlink %d: %v", entry.FullPath, entry.HardLinkId, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = entry.DecodeAttributesAndChunks(value); err != nil {
|
|
|
|
glog.Errorf("decode %s hardlink %d: %v", entry.FullPath, entry.HardLinkId, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fsw *FilerStoreWrapper) DeleteHardLink(ctx context.Context, hardLinkId HardLinkId) error {
|
2020-09-24 18:11:42 +00:00
|
|
|
key := hardLinkId
|
2020-09-24 16:43:00 +00:00
|
|
|
value, err := fsw.KvGet(ctx, key)
|
|
|
|
if err == ErrKvNotFound {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
entry := &Entry{}
|
|
|
|
if err = entry.DecodeAttributesAndChunks(value); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
entry.HardLinkCounter--
|
|
|
|
if entry.HardLinkCounter <= 0 {
|
2020-12-11 03:55:28 +00:00
|
|
|
glog.V(4).Infof("DeleteHardLink KvDelete %v", key)
|
2020-09-24 16:43:00 +00:00
|
|
|
return fsw.KvDelete(ctx, key)
|
|
|
|
}
|
|
|
|
|
|
|
|
newBlob, encodeErr := entry.EncodeAttributesAndChunks()
|
|
|
|
if encodeErr != nil {
|
|
|
|
return encodeErr
|
|
|
|
}
|
|
|
|
|
2020-12-11 03:55:28 +00:00
|
|
|
glog.V(4).Infof("DeleteHardLink KvPut %v", key)
|
2020-09-24 16:43:00 +00:00
|
|
|
return fsw.KvPut(ctx, key, newBlob)
|
|
|
|
|
|
|
|
}
|