mirror of
https://github.com/AdguardTeam/AdGuardHome.git
synced 2024-12-15 11:22:49 +03:00
698b963e11
Squashed commit of the following:
commit 6ce649c06398cf8a6f8e1a90f560fa8205f6500e
Merge: 1c6327e5d 996c6b3ee
Author: Ainar Garipov <A.Garipov@AdGuard.COM>
Date: Tue Jul 25 17:42:01 2023 +0300
Merge branch 'master' into imp-filter-upd
commit 1c6327e5d4c04393abc5d4d3e4b8568d4c6eca23
Author: Ainar Garipov <A.Garipov@AdGuard.COM>
Date: Fri Jul 21 17:32:47 2023 +0300
all: imp code; use renameio/v2 consistently
commit 1669288c9b662d1310f83a4e0d3f1f60731188cd
Author: Ainar Garipov <A.Garipov@AdGuard.COM>
Date: Fri Jul 21 16:26:17 2023 +0300
all: add renameioutil; imp flt upd
75 lines
1.7 KiB
Go
75 lines
1.7 KiB
Go
//go:build windows
|
|
|
|
package aghrenameio
|
|
|
|
import (
|
|
"fmt"
|
|
"io/fs"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"github.com/AdguardTeam/golibs/errors"
|
|
)
|
|
|
|
// pendingFile is a wrapper around [*os.File] calling [os.Rename] in its Close
|
|
// method.
|
|
type pendingFile struct {
|
|
file *os.File
|
|
targetPath string
|
|
}
|
|
|
|
// type check
|
|
var _ PendingFile = (*pendingFile)(nil)
|
|
|
|
// Cleanup implements the [PendingFile] interface for *pendingFile.
|
|
func (f *pendingFile) Cleanup() (err error) {
|
|
closeErr := f.file.Close()
|
|
err = os.Remove(f.file.Name())
|
|
|
|
// Put closeErr into the deferred error because that's where it is usually
|
|
// expected.
|
|
return errors.WithDeferred(err, closeErr)
|
|
}
|
|
|
|
// CloseReplace implements the [PendingFile] interface for *pendingFile.
|
|
func (f *pendingFile) CloseReplace() (err error) {
|
|
err = f.file.Close()
|
|
if err != nil {
|
|
return fmt.Errorf("closing: %w", err)
|
|
}
|
|
|
|
err = os.Rename(f.file.Name(), f.targetPath)
|
|
if err != nil {
|
|
return fmt.Errorf("renaming: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Write implements the [PendingFile] interface for *pendingFile.
|
|
func (f *pendingFile) Write(b []byte) (n int, err error) {
|
|
return f.file.Write(b)
|
|
}
|
|
|
|
// NewPendingFile is a wrapper around [os.CreateTemp].
|
|
//
|
|
// f.Close must be called to finish the renaming.
|
|
func newPendingFile(filePath string, mode fs.FileMode) (f PendingFile, err error) {
|
|
// Use the same directory as the file itself, because moves across
|
|
// filesystems can be especially problematic.
|
|
file, err := os.CreateTemp(filepath.Dir(filePath), "")
|
|
if err != nil {
|
|
return nil, fmt.Errorf("opening pending file: %w", err)
|
|
}
|
|
|
|
err = file.Chmod(mode)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("preparing pending file: %w", err)
|
|
}
|
|
|
|
return &pendingFile{
|
|
file: file,
|
|
targetPath: filePath,
|
|
}, nil
|
|
}
|