fix(filer.backup): local sink readonly permission (#8907)
This commit is contained in:
@@ -93,7 +93,11 @@ func (localsink *LocalSink) CreateEntry(key string, entry *filer_pb.Entry, signa
|
|||||||
}
|
}
|
||||||
|
|
||||||
mode := os.FileMode(entry.Attributes.FileMode)
|
mode := os.FileMode(entry.Attributes.FileMode)
|
||||||
dstFile, err := os.OpenFile(util.ToShortFileName(key), os.O_RDWR|os.O_CREATE|os.O_TRUNC, mode)
|
shortFileName := util.ToShortFileName(key)
|
||||||
|
if err := os.Remove(shortFileName); err != nil && !os.IsNotExist(err) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
dstFile, err := os.OpenFile(shortFileName, os.O_RDWR|os.O_CREATE|os.O_TRUNC, mode)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|||||||
74
weed/replication/sink/localsink/local_sink_test.go
Normal file
74
weed/replication/sink/localsink/local_sink_test.go
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
package localsink
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
|
||||||
|
"github.com/seaweedfs/seaweedfs/weed/replication/source"
|
||||||
|
)
|
||||||
|
|
||||||
|
// TestCreateEntry_OverwriteReadOnlyFile reproduces a bug where
|
||||||
|
// filer.backup receives a create event (0-byte, mode 0400) followed by
|
||||||
|
// an update event (with chunks) for the same file. The update event
|
||||||
|
// calls CreateEntry again, which fails with "permission denied" because
|
||||||
|
// OpenFile with O_RDWR cannot open the 0400 file created by the first
|
||||||
|
// event.
|
||||||
|
func TestCreateEntry_OverwriteReadOnlyFile(t *testing.T) {
|
||||||
|
tmpDir := t.TempDir()
|
||||||
|
sink := &LocalSink{}
|
||||||
|
sink.initialize(tmpDir, false)
|
||||||
|
sink.SetSourceFiler(&source.FilerSource{})
|
||||||
|
|
||||||
|
key := filepath.Join(tmpDir, "objects", "5c", "9fb207")
|
||||||
|
|
||||||
|
// Create event: 0-byte file with mode 0400 (metadata only, no chunks)
|
||||||
|
createEntry := &filer_pb.Entry{
|
||||||
|
Attributes: &filer_pb.FuseAttributes{
|
||||||
|
FileMode: uint32(0400),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
if err := os.MkdirAll(filepath.Dir(key), 0755); err != nil {
|
||||||
|
t.Fatalf("mkdir: %v", err)
|
||||||
|
}
|
||||||
|
if err := sink.CreateEntry(key, createEntry, nil); err != nil {
|
||||||
|
t.Fatalf("CreateEntry (create event) failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify: 0-byte, 0400 file exists
|
||||||
|
fi, err := os.Stat(key)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("stat after create event: %v", err)
|
||||||
|
}
|
||||||
|
if fi.Size() != 0 {
|
||||||
|
t.Errorf("expected 0 bytes after create event, got %d", fi.Size())
|
||||||
|
}
|
||||||
|
if fi.Mode().Perm() != 0400 {
|
||||||
|
t.Fatalf("expected 0400 after create event, got %o", fi.Mode().Perm())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update event: same file, now with content (simulated by entry.Content)
|
||||||
|
updateEntry := &filer_pb.Entry{
|
||||||
|
Attributes: &filer_pb.FuseAttributes{
|
||||||
|
FileMode: uint32(0400),
|
||||||
|
FileSize: 552,
|
||||||
|
},
|
||||||
|
Content: []byte("git object data placeholder"),
|
||||||
|
}
|
||||||
|
if err := sink.CreateEntry(key, updateEntry, nil); err != nil {
|
||||||
|
t.Fatalf("CreateEntry (update event) failed on read-only file: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify: file has content and correct permissions
|
||||||
|
fi, err = os.Stat(key)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("stat after update event: %v", err)
|
||||||
|
}
|
||||||
|
if fi.Size() == 0 {
|
||||||
|
t.Errorf("expected non-zero size after update event, got 0")
|
||||||
|
}
|
||||||
|
if fi.Mode().Perm() != 0400 {
|
||||||
|
t.Errorf("expected 0400 after update event, got %o", fi.Mode().Perm())
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user