2018-04-08 08:02:30 -04:00
|
|
|
package restorer
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
"sync"
|
|
|
|
|
2020-10-05 15:38:39 +02:00
|
|
|
"github.com/cespare/xxhash/v2"
|
2020-08-15 17:45:05 +02:00
|
|
|
"github.com/restic/restic/internal/debug"
|
2021-02-02 17:43:40 +03:00
|
|
|
"github.com/restic/restic/internal/fs"
|
2018-04-08 08:02:30 -04:00
|
|
|
)
|
|
|
|
|
2019-11-27 07:22:38 -05:00
|
|
|
// writes blobs to target files.
|
|
|
|
// multiple files can be written to concurrently.
|
|
|
|
// multiple blobs can be concurrently written to the same file.
|
|
|
|
// TODO I am not 100% convinced this is necessary, i.e. it may be okay
|
2022-08-19 19:12:26 +02:00
|
|
|
// to use multiple os.File to write to the same target file
|
2018-04-08 08:02:30 -04:00
|
|
|
type filesWriter struct {
|
2019-11-27 07:22:38 -05:00
|
|
|
buckets []filesWriterBucket
|
2018-04-08 08:02:30 -04:00
|
|
|
}
|
|
|
|
|
2019-11-27 07:22:38 -05:00
|
|
|
type filesWriterBucket struct {
|
|
|
|
lock sync.Mutex
|
2020-02-26 21:48:05 +01:00
|
|
|
files map[string]*partialFile
|
|
|
|
}
|
|
|
|
|
|
|
|
type partialFile struct {
|
|
|
|
*os.File
|
2022-08-07 17:56:14 +02:00
|
|
|
users int // Reference count.
|
2022-08-07 17:26:46 +02:00
|
|
|
sparse bool
|
2019-11-27 07:22:38 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func newFilesWriter(count int) *filesWriter {
|
|
|
|
buckets := make([]filesWriterBucket, count)
|
|
|
|
for b := 0; b < count; b++ {
|
2020-02-26 21:48:05 +01:00
|
|
|
buckets[b].files = make(map[string]*partialFile)
|
2019-11-27 07:22:38 -05:00
|
|
|
}
|
2019-02-24 21:50:40 -08:00
|
|
|
return &filesWriter{
|
2019-11-27 07:22:38 -05:00
|
|
|
buckets: buckets,
|
2019-02-24 21:50:40 -08:00
|
|
|
}
|
2018-04-08 08:02:30 -04:00
|
|
|
}
|
|
|
|
|
2022-08-07 17:26:46 +02:00
|
|
|
func (w *filesWriter) writeToFile(path string, blob []byte, offset int64, createSize int64, sparse bool) error {
|
2019-11-27 07:22:38 -05:00
|
|
|
bucket := &w.buckets[uint(xxhash.Sum64String(path))%uint(len(w.buckets))]
|
|
|
|
|
2020-02-26 21:48:05 +01:00
|
|
|
acquireWriter := func() (*partialFile, error) {
|
2019-11-27 07:22:38 -05:00
|
|
|
bucket.lock.Lock()
|
|
|
|
defer bucket.lock.Unlock()
|
|
|
|
|
|
|
|
if wr, ok := bucket.files[path]; ok {
|
2020-02-26 21:48:05 +01:00
|
|
|
bucket.files[path].users++
|
2019-02-24 21:50:40 -08:00
|
|
|
return wr, nil
|
2018-04-08 08:02:30 -04:00
|
|
|
}
|
2024-02-22 17:31:20 -07:00
|
|
|
var f *os.File
|
|
|
|
var err error
|
2020-08-15 17:45:05 +02:00
|
|
|
if createSize >= 0 {
|
2024-02-22 17:31:20 -07:00
|
|
|
if f, err = os.OpenFile(path, os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0600); err != nil {
|
|
|
|
if fs.IsAccessDenied(err) {
|
|
|
|
// If file is readonly, clear the readonly flag by resetting the
|
|
|
|
// permissions of the file and try again
|
|
|
|
// as the metadata will be set again in the second pass and the
|
|
|
|
// readonly flag will be applied again if needed.
|
|
|
|
if err = fs.ResetPermissions(path); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if f, err = os.OpenFile(path, os.O_TRUNC|os.O_WRONLY, 0600); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if f, err = os.OpenFile(path, os.O_WRONLY, 0600); err != nil {
|
2018-04-08 08:02:30 -04:00
|
|
|
return nil, err
|
|
|
|
}
|
2019-11-27 07:22:38 -05:00
|
|
|
|
2022-08-07 17:26:46 +02:00
|
|
|
wr := &partialFile{File: f, users: 1, sparse: sparse}
|
2019-11-27 07:22:38 -05:00
|
|
|
bucket.files[path] = wr
|
|
|
|
|
2022-08-07 17:56:14 +02:00
|
|
|
if createSize >= 0 {
|
|
|
|
if sparse {
|
2022-09-04 11:23:31 +02:00
|
|
|
err = truncateSparse(f, createSize)
|
2022-08-07 17:56:14 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
} else {
|
2021-02-02 17:43:40 +03:00
|
|
|
err := fs.PreallocateFile(wr.File, createSize)
|
2022-08-07 17:56:14 +02:00
|
|
|
if err != nil {
|
|
|
|
// 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
|
|
|
|
// support preallocation or our parameters combination to the preallocate call
|
|
|
|
// This should yield a syscall.ENOTSUP error, but some other errors might also
|
|
|
|
// show up.
|
|
|
|
debug.Log("Failed to preallocate %v with size %v: %v", path, createSize, err)
|
|
|
|
}
|
2020-08-15 17:45:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-08 08:02:30 -04:00
|
|
|
return wr, nil
|
|
|
|
}
|
2019-11-27 07:22:38 -05:00
|
|
|
|
2020-02-26 21:48:05 +01:00
|
|
|
releaseWriter := func(wr *partialFile) error {
|
2019-11-27 07:22:38 -05:00
|
|
|
bucket.lock.Lock()
|
|
|
|
defer bucket.lock.Unlock()
|
|
|
|
|
2020-02-26 21:48:05 +01:00
|
|
|
if bucket.files[path].users == 1 {
|
2019-11-27 07:22:38 -05:00
|
|
|
delete(bucket.files, path)
|
|
|
|
return wr.Close()
|
2019-02-24 21:50:40 -08:00
|
|
|
}
|
2020-02-26 21:48:05 +01:00
|
|
|
bucket.files[path].users--
|
2019-11-27 07:22:38 -05:00
|
|
|
return nil
|
2019-02-24 21:50:40 -08:00
|
|
|
}
|
2018-04-08 08:02:30 -04:00
|
|
|
|
|
|
|
wr, err := acquireWriter()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-11-27 07:22:38 -05:00
|
|
|
|
|
|
|
_, err = wr.WriteAt(blob, offset)
|
|
|
|
|
2018-04-08 08:02:30 -04:00
|
|
|
if err != nil {
|
2021-01-30 19:35:46 +01:00
|
|
|
// ignore subsequent errors
|
|
|
|
_ = releaseWriter(wr)
|
2018-04-08 08:02:30 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-11-27 07:22:38 -05:00
|
|
|
return releaseWriter(wr)
|
2018-04-08 08:02:30 -04:00
|
|
|
}
|