Skip to content

Commit f44b615

Browse files
authored
Merge pull request #124 from HusterWan/zr/expose-func
feature: expose atomicWriterFile function as AtomicWriteFile
2 parents c7c5070 + 4469d34 commit f44b615

File tree

3 files changed

+56
-5
lines changed

3 files changed

+56
-5
lines changed

context.go

+6-2
Original file line numberDiff line numberDiff line change
@@ -12,11 +12,14 @@ import (
1212
"github.com/containerd/continuity/devices"
1313
driverpkg "github.com/containerd/continuity/driver"
1414
"github.com/containerd/continuity/pathdriver"
15+
1516
"github.com/opencontainers/go-digest"
1617
)
1718

1819
var (
19-
ErrNotFound = fmt.Errorf("not found")
20+
// ErrNotFound represents the resource not found
21+
ErrNotFound = fmt.Errorf("not found")
22+
// ErrNotSupported represents the resource not supported
2023
ErrNotSupported = fmt.Errorf("not supported")
2124
)
2225

@@ -36,6 +39,7 @@ type Context interface {
3639
// not under the given root.
3740
type SymlinkPath func(root, linkname, target string) (string, error)
3841

42+
// ContextOptions represents options to create a new context.
3943
type ContextOptions struct {
4044
Digester Digester
4145
Driver driverpkg.Driver
@@ -379,7 +383,7 @@ func (c *context) checkoutFile(fp string, rf RegularFile) error {
379383
}
380384
defer r.Close()
381385

382-
return atomicWriteFile(fp, r, rf)
386+
return atomicWriteFile(fp, r, rf.Size(), rf.Mode())
383387
}
384388

385389
// Apply the resource to the contexts. An error will be returned if the

ioutils.go

+11-3
Original file line numberDiff line numberDiff line change
@@ -1,26 +1,34 @@
11
package continuity
22

33
import (
4+
"bytes"
45
"io"
56
"io/ioutil"
67
"os"
78
"path/filepath"
89
)
910

11+
// AtomicWriteFile atomically writes data to a file by first writing to a
12+
// temp file and calling rename.
13+
func AtomicWriteFile(filename string, data []byte, perm os.FileMode) error {
14+
buf := bytes.NewBuffer(data)
15+
return atomicWriteFile(filename, buf, int64(len(data)), perm)
16+
}
17+
1018
// atomicWriteFile writes data to a file by first writing to a temp
1119
// file and calling rename.
12-
func atomicWriteFile(filename string, r io.Reader, rf RegularFile) error {
20+
func atomicWriteFile(filename string, r io.Reader, dataSize int64, perm os.FileMode) error {
1321
f, err := ioutil.TempFile(filepath.Dir(filename), ".tmp-"+filepath.Base(filename))
1422
if err != nil {
1523
return err
1624
}
17-
err = os.Chmod(f.Name(), rf.Mode())
25+
err = os.Chmod(f.Name(), perm)
1826
if err != nil {
1927
f.Close()
2028
return err
2129
}
2230
n, err := io.Copy(f, r)
23-
if err == nil && n < rf.Size() {
31+
if err == nil && n < dataSize {
2432
f.Close()
2533
return io.ErrShortWrite
2634
}

ioutils_test.go

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
package continuity
2+
3+
import (
4+
"bytes"
5+
"io/ioutil"
6+
"os"
7+
"path/filepath"
8+
"testing"
9+
)
10+
11+
func TestAtomicWriteFile(t *testing.T) {
12+
tmpDir, err := ioutil.TempDir("", "atomic-write-test")
13+
if err != nil {
14+
t.Fatalf("Error when creating temporary directory: %s", err)
15+
}
16+
defer os.RemoveAll(tmpDir)
17+
18+
expected := []byte("barbaz")
19+
if err := AtomicWriteFile(filepath.Join(tmpDir, "foo"), expected, 0666); err != nil {
20+
t.Fatalf("Error writing to file: %v", err)
21+
}
22+
23+
actual, err := ioutil.ReadFile(filepath.Join(tmpDir, "foo"))
24+
if err != nil {
25+
t.Fatalf("Error reading from file: %v", err)
26+
}
27+
28+
if bytes.Compare(actual, expected) != 0 {
29+
t.Fatalf("Data mismatch, expected %q, got %q", expected, actual)
30+
}
31+
32+
st, err := os.Stat(filepath.Join(tmpDir, "foo"))
33+
if err != nil {
34+
t.Fatalf("Error statting file: %v", err)
35+
}
36+
if expected := os.FileMode(0666); st.Mode() != expected {
37+
t.Fatalf("Mode mismatched, expected %o, got %o", expected, st.Mode())
38+
}
39+
}

0 commit comments

Comments
 (0)