mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
6234af6680
Perform chmod before rename with the atomic file writer.
Ensure writeErr is set on short write and file is removed on write error.
Signed-off-by: Derek McGowan <derek@mcgstyle.net> (github: dmcgowan)
(cherry picked from commit 1cd7490281
)
82 lines
1.7 KiB
Go
82 lines
1.7 KiB
Go
package ioutils
|
|
|
|
import (
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
)
|
|
|
|
// NewAtomicFileWriter returns WriteCloser so that writing to it writes to a
|
|
// temporary file and closing it atomically changes the temporary file to
|
|
// destination path. Writing and closing concurrently is not allowed.
|
|
func NewAtomicFileWriter(filename string, perm os.FileMode) (io.WriteCloser, error) {
|
|
f, err := ioutil.TempFile(filepath.Dir(filename), ".tmp-"+filepath.Base(filename))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
abspath, err := filepath.Abs(filename)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &atomicFileWriter{
|
|
f: f,
|
|
fn: abspath,
|
|
perm: perm,
|
|
}, nil
|
|
}
|
|
|
|
// AtomicWriteFile atomically writes data to a file named by filename.
|
|
func AtomicWriteFile(filename string, data []byte, perm os.FileMode) error {
|
|
f, err := NewAtomicFileWriter(filename, perm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
n, err := f.Write(data)
|
|
if err == nil && n < len(data) {
|
|
err = io.ErrShortWrite
|
|
f.(*atomicFileWriter).writeErr = err
|
|
}
|
|
if err1 := f.Close(); err == nil {
|
|
err = err1
|
|
}
|
|
return err
|
|
}
|
|
|
|
type atomicFileWriter struct {
|
|
f *os.File
|
|
fn string
|
|
writeErr error
|
|
perm os.FileMode
|
|
}
|
|
|
|
func (w *atomicFileWriter) Write(dt []byte) (int, error) {
|
|
n, err := w.f.Write(dt)
|
|
if err != nil {
|
|
w.writeErr = err
|
|
}
|
|
return n, err
|
|
}
|
|
|
|
func (w *atomicFileWriter) Close() (retErr error) {
|
|
defer func() {
|
|
if retErr != nil || w.writeErr != nil {
|
|
os.Remove(w.f.Name())
|
|
}
|
|
}()
|
|
if err := w.f.Sync(); err != nil {
|
|
w.f.Close()
|
|
return err
|
|
}
|
|
if err := w.f.Close(); err != nil {
|
|
return err
|
|
}
|
|
if err := os.Chmod(w.f.Name(), w.perm); err != nil {
|
|
return err
|
|
}
|
|
if w.writeErr == nil {
|
|
return os.Rename(w.f.Name(), w.fn)
|
|
}
|
|
return nil
|
|
}
|