Commit 44cf595a authored by Alex Brainman's avatar Alex Brainman

path/filepath: return special error from EvalSymlinks

CL 155597 attempted to fix #29372. But it failed to make all new
test cases pass. Also CL 155597 broke some existing code
(see #29449 for details).

Make small adjustment to CL 155597 that fixes both #29372 and #29449.

Suggested by Ian.

Updates #29372
Fixes #29449

Change-Id: I9777a615514d3f152af5acb65fb1239e696607b6
Reviewed-on: https://go-review.googlesource.com/c/156398
Run-TryBot: Alex Brainman <alex.brainman@gmail.com>
Reviewed-by: default avatarIan Lance Taylor <iant@golang.org>
parent 94d9a204
...@@ -1382,27 +1382,18 @@ func TestIssue29372(t *testing.T) { ...@@ -1382,27 +1382,18 @@ func TestIssue29372(t *testing.T) {
path := f.Name() path := f.Name()
defer os.Remove(path) defer os.Remove(path)
isWin := runtime.GOOS == "windows"
pathSeparator := string(filepath.Separator) pathSeparator := string(filepath.Separator)
tests := []struct { tests := []string{
path string path + strings.Repeat(pathSeparator, 1),
skip bool path + strings.Repeat(pathSeparator, 2),
}{ path + strings.Repeat(pathSeparator, 1) + ".",
{path + strings.Repeat(pathSeparator, 1), false}, path + strings.Repeat(pathSeparator, 2) + ".",
{path + strings.Repeat(pathSeparator, 2), false}, path + strings.Repeat(pathSeparator, 1) + "..",
{path + strings.Repeat(pathSeparator, 1) + ".", false}, path + strings.Repeat(pathSeparator, 2) + "..",
{path + strings.Repeat(pathSeparator, 2) + ".", false},
// windows.GetFinalPathNameByHandle return the directory part with trailing dot dot
// C:\path\to\existing_dir\existing_file\.. returns C:\path\to\existing_dir
{path + strings.Repeat(pathSeparator, 1) + "..", isWin},
{path + strings.Repeat(pathSeparator, 2) + "..", isWin},
} }
for i, test := range tests { for i, test := range tests {
if test.skip { _, err = filepath.EvalSymlinks(test)
continue
}
_, err = filepath.EvalSymlinks(test.path)
if err != syscall.ENOTDIR { if err != syscall.ENOTDIR {
t.Fatalf("test#%d: want %q, got %q", i, syscall.ENOTDIR, err) t.Fatalf("test#%d: want %q, got %q", i, syscall.ENOTDIR, err)
} }
......
...@@ -536,17 +536,39 @@ func TestNTNamespaceSymlink(t *testing.T) { ...@@ -536,17 +536,39 @@ func TestNTNamespaceSymlink(t *testing.T) {
} }
target := strings.Trim(string(output), " \n\r") target := strings.Trim(string(output), " \n\r")
link := filepath.Join(tmpdir, "link") dirlink := filepath.Join(tmpdir, "dirlink")
output, err = exec.Command("cmd", "/c", "mklink", "/J", link, target).CombinedOutput() output, err = exec.Command("cmd", "/c", "mklink", "/J", dirlink, target).CombinedOutput()
if err != nil { if err != nil {
t.Fatalf("failed to run mklink %v %v: %v %q", link, target, err, output) t.Fatalf("failed to run mklink %v %v: %v %q", dirlink, target, err, output)
} }
got, err := filepath.EvalSymlinks(link) got, err := filepath.EvalSymlinks(dirlink)
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
if want := vol + `\`; got != want { if want := vol + `\`; got != want {
t.Errorf(`EvalSymlinks(%q): got %q, want %q`, link, got, want) t.Errorf(`EvalSymlinks(%q): got %q, want %q`, dirlink, got, want)
}
file := filepath.Join(tmpdir, "file")
err = ioutil.WriteFile(file, []byte(""), 0666)
if err != nil {
t.Fatal(err)
}
target += file[len(filepath.VolumeName(file)):]
filelink := filepath.Join(tmpdir, "filelink")
output, err = exec.Command("cmd", "/c", "mklink", filelink, target).CombinedOutput()
if err != nil {
t.Fatalf("failed to run mklink %v %v: %v %q", filelink, target, err, output)
}
got, err = filepath.EvalSymlinks(filelink)
if err != nil {
t.Fatal(err)
}
if want := file; got != want {
t.Errorf(`EvalSymlinks(%q): got %q, want %q`, filelink, got, want)
} }
} }
...@@ -8,7 +8,6 @@ import ( ...@@ -8,7 +8,6 @@ import (
"errors" "errors"
"os" "os"
"runtime" "runtime"
"syscall"
) )
func walkSymlinks(path string) (string, error) { func walkSymlinks(path string) (string, error) {
...@@ -79,7 +78,7 @@ func walkSymlinks(path string) (string, error) { ...@@ -79,7 +78,7 @@ func walkSymlinks(path string) (string, error) {
if fi.Mode()&os.ModeSymlink == 0 { if fi.Mode()&os.ModeSymlink == 0 {
if !fi.Mode().IsDir() && end < len(path) { if !fi.Mode().IsDir() && end < len(path) {
return "", syscall.ENOTDIR return "", slashAfterFilePathError
} }
continue continue
} }
......
...@@ -2,6 +2,15 @@ ...@@ -2,6 +2,15 @@
package filepath package filepath
import (
"syscall"
)
// walkSymlinks returns slashAfterFilePathError error for paths like
// //path/to/existing_file/ and /path/to/existing_file/. and /path/to/existing_file/..
var slashAfterFilePathError = syscall.ENOTDIR
func evalSymlinks(path string) (string, error) { func evalSymlinks(path string) (string, error) {
return walkSymlinks(path) return walkSymlinks(path)
} }
...@@ -159,18 +159,6 @@ func evalSymlinksUsingGetFinalPathNameByHandle(path string) (string, error) { ...@@ -159,18 +159,6 @@ func evalSymlinksUsingGetFinalPathNameByHandle(path string) (string, error) {
return "", errors.New("GetFinalPathNameByHandle returned unexpected path=" + s) return "", errors.New("GetFinalPathNameByHandle returned unexpected path=" + s)
} }
func symlinkOrDir(path string) (string, error) {
fi, err := os.Lstat(path)
if err != nil {
return "", err
}
if fi.Mode()&os.ModeSymlink == 0 && !fi.Mode().IsDir() {
return "", syscall.ENOTDIR
}
return path, nil
}
func samefile(path1, path2 string) bool { func samefile(path1, path2 string) bool {
fi1, err := os.Lstat(path1) fi1, err := os.Lstat(path1)
if err != nil { if err != nil {
...@@ -183,16 +171,20 @@ func samefile(path1, path2 string) bool { ...@@ -183,16 +171,20 @@ func samefile(path1, path2 string) bool {
return os.SameFile(fi1, fi2) return os.SameFile(fi1, fi2)
} }
// walkSymlinks returns slashAfterFilePathError error for paths like
// //path/to/existing_file/ and /path/to/existing_file/. and /path/to/existing_file/..
var slashAfterFilePathError = errors.New("attempting to walk past file path.")
func evalSymlinks(path string) (string, error) { func evalSymlinks(path string) (string, error) {
newpath, err := walkSymlinks(path) newpath, err := walkSymlinks(path)
if err == slashAfterFilePathError {
return "", syscall.ENOTDIR
}
if err != nil { if err != nil {
newpath2, err2 := evalSymlinksUsingGetFinalPathNameByHandle(path) newpath2, err2 := evalSymlinksUsingGetFinalPathNameByHandle(path)
if err2 == nil { if err2 == nil {
normPath, toNormErr := toNorm(newpath2, normBase) return toNorm(newpath2, normBase)
if toNormErr != nil {
return "", toNormErr
}
return symlinkOrDir(normPath)
} }
return "", err return "", err
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment