Merge pull request #3261 from DRON-666/prealloc-packs

Prevent local backend file fragmentation by file preallocation.
This commit is contained in:
Michael Eischer 2023-06-09 11:50:46 +02:00 committed by GitHub
commit 237f32c651
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 26 additions and 11 deletions

View file

@ -0,0 +1,8 @@
Enhancement: Reduce file fragmentation for local backend
Before this change, local backend files could become fragmented.
Now restic will try to preallocate space for pack files to avoid
their fragmentation.
https://github.com/restic/restic/issues/2679
https://github.com/restic/restic/pull/3261

View file

@ -148,6 +148,13 @@ func (b *Local) Save(_ context.Context, h restic.Handle, rd restic.RewindReader)
} }
}(f) }(f)
// preallocate disk space
if size := rd.Length(); size > 0 {
if err := fs.PreallocateFile(f, size); err != nil {
debug.Log("Failed to preallocate %v with size %v: %v", finalname, size, err)
}
}
// save data, then sync // save data, then sync
wbytes, err := io.Copy(f, rd) wbytes, err := io.Copy(f, rd)
if err != nil { if err != nil {

View file

@ -1,4 +1,4 @@
package restorer package fs
import ( import (
"os" "os"
@ -6,7 +6,7 @@ import (
"golang.org/x/sys/unix" "golang.org/x/sys/unix"
) )
func preallocateFile(wr *os.File, size int64) error { func PreallocateFile(wr *os.File, size int64) error {
// try contiguous first // try contiguous first
fst := unix.Fstore_t{ fst := unix.Fstore_t{
Flags: unix.F_ALLOCATECONTIG | unix.F_ALLOCATEALL, Flags: unix.F_ALLOCATECONTIG | unix.F_ALLOCATEALL,

View file

@ -1,4 +1,4 @@
package restorer package fs
import ( import (
"os" "os"
@ -6,7 +6,7 @@ import (
"golang.org/x/sys/unix" "golang.org/x/sys/unix"
) )
func preallocateFile(wr *os.File, size int64) error { func PreallocateFile(wr *os.File, size int64) error {
if size <= 0 { if size <= 0 {
return nil return nil
} }

View file

@ -1,11 +1,11 @@
//go:build !linux && !darwin //go:build !linux && !darwin
// +build !linux,!darwin // +build !linux,!darwin
package restorer package fs
import "os" import "os"
func preallocateFile(wr *os.File, size int64) error { func PreallocateFile(wr *os.File, size int64) error {
// Maybe truncate can help? // Maybe truncate can help?
// Windows: This calls SetEndOfFile which preallocates space on disk // Windows: This calls SetEndOfFile which preallocates space on disk
return wr.Truncate(size) return wr.Truncate(size)

View file

@ -1,4 +1,4 @@
package restorer package fs
import ( import (
"os" "os"
@ -7,7 +7,6 @@ import (
"syscall" "syscall"
"testing" "testing"
"github.com/restic/restic/internal/fs"
"github.com/restic/restic/internal/test" "github.com/restic/restic/internal/test"
) )
@ -23,7 +22,7 @@ func TestPreallocate(t *testing.T) {
test.OK(t, wr.Close()) test.OK(t, wr.Close())
}() }()
err = preallocateFile(wr, i) err = PreallocateFile(wr, i)
if err == syscall.ENOTSUP { if err == syscall.ENOTSUP {
t.SkipNow() t.SkipNow()
} }
@ -32,7 +31,7 @@ func TestPreallocate(t *testing.T) {
fi, err := wr.Stat() fi, err := wr.Stat()
test.OK(t, err) test.OK(t, err)
efi := fs.ExtendedStat(fi) efi := ExtendedStat(fi)
test.Assert(t, efi.Size == i || efi.Blocks > 0, "Preallocated size of %v, got size %v block %v", i, efi.Size, efi.Blocks) test.Assert(t, efi.Size == i || efi.Blocks > 0, "Preallocated size of %v, got size %v block %v", i, efi.Size, efi.Blocks)
}) })
} }

View file

@ -6,6 +6,7 @@ import (
"github.com/cespare/xxhash/v2" "github.com/cespare/xxhash/v2"
"github.com/restic/restic/internal/debug" "github.com/restic/restic/internal/debug"
"github.com/restic/restic/internal/fs"
) )
// writes blobs to target files. // writes blobs to target files.
@ -72,7 +73,7 @@ func (w *filesWriter) writeToFile(path string, blob []byte, offset int64, create
return nil, err return nil, err
} }
} else { } else {
err := preallocateFile(wr.File, createSize) err := fs.PreallocateFile(wr.File, createSize)
if err != nil { if err != nil {
// Just log the preallocate error but don't let it cause the restore process to fail. // Just log the preallocate error but don't let it cause the restore process to fail.
// Preallocate might return an error if the filesystem (implementation) does not // Preallocate might return an error if the filesystem (implementation) does not