mirror of
https://github.com/octoleo/syncthing.git
synced 2024-11-08 22:31:04 +00:00
So, in a funny plot twist, it turns out that WriteFile in Go 1.13 doesn't actually set the read only bit on Windows when called with permissions 0444 so my test was broken. With an improved test it turns out that Rename does not, in fact, overwrite a read-only file on Windows. This adds a fix for that. (Rename might get improved in Go 1.15...)
This commit is contained in:
parent
4b17c511f9
commit
59b1b0e1dc
@ -9,6 +9,7 @@ package osutil
|
|||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"runtime"
|
||||||
|
|
||||||
"github.com/syncthing/syncthing/lib/fs"
|
"github.com/syncthing/syncthing/lib/fs"
|
||||||
)
|
)
|
||||||
@ -92,7 +93,15 @@ func (w *AtomicWriter) Close() error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := w.fs.Rename(w.next.Name(), w.path); err != nil {
|
err := w.fs.Rename(w.next.Name(), w.path)
|
||||||
|
if runtime.GOOS == "windows" && fs.IsPermission(err) {
|
||||||
|
// On Windows, we might not be allowed to rename over the file
|
||||||
|
// because it's read-only. Get us some write permissions and try
|
||||||
|
// again.
|
||||||
|
_ = w.fs.Chmod(w.path, 0644)
|
||||||
|
err = w.fs.Rename(w.next.Name(), w.path)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
w.err = err
|
w.err = err
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -10,6 +10,7 @@ import (
|
|||||||
"bytes"
|
"bytes"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -52,27 +53,45 @@ func TestCreateAtomicCreate(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestCreateAtomicReplace(t *testing.T) {
|
func TestCreateAtomicReplace(t *testing.T) {
|
||||||
testCreateAtomicReplace(t, 0644)
|
testCreateAtomicReplace(t, 0666)
|
||||||
}
|
}
|
||||||
func TestCreateAtomicReplaceReadOnly(t *testing.T) {
|
func TestCreateAtomicReplaceReadOnly(t *testing.T) {
|
||||||
testCreateAtomicReplace(t, 0400)
|
testCreateAtomicReplace(t, 0444) // windows compatible read-only bits
|
||||||
}
|
}
|
||||||
|
|
||||||
func testCreateAtomicReplace(t *testing.T, oldPerms os.FileMode) {
|
func testCreateAtomicReplace(t *testing.T, oldPerms os.FileMode) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
|
||||||
os.RemoveAll("testdata")
|
testdir, err := ioutil.TempDir("", "syncthing")
|
||||||
defer os.RemoveAll("testdata")
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
testfile := filepath.Join(testdir, "testfile")
|
||||||
|
|
||||||
if err := os.Mkdir("testdata", 0755); err != nil {
|
os.RemoveAll(testdir)
|
||||||
|
defer os.RemoveAll(testdir)
|
||||||
|
|
||||||
|
if err := os.Mkdir(testdir, 0755); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := ioutil.WriteFile("testdata/file", []byte("some old data"), oldPerms); err != nil {
|
if err := ioutil.WriteFile(testfile, []byte("some old data"), oldPerms); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
w, err := CreateAtomic("testdata/file")
|
// Go < 1.14 has a bug in WriteFile where it does not use the requested
|
||||||
|
// permissions on Windows. Chmod to make sure.
|
||||||
|
if err := os.Chmod(testfile, oldPerms); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// Trust, but verify.
|
||||||
|
if info, err := os.Stat(testfile); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
} else if info.Mode() != oldPerms {
|
||||||
|
t.Fatalf("Wrong perms 0%o", info.Mode())
|
||||||
|
}
|
||||||
|
|
||||||
|
w, err := CreateAtomic(testfile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@ -84,7 +103,7 @@ func testCreateAtomicReplace(t *testing.T, oldPerms os.FileMode) {
|
|||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
bs, err := ioutil.ReadFile("testdata/file")
|
bs, err := ioutil.ReadFile(testfile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user