Skip to content

Commit

Permalink
os: return an error when the argument of Mkdir on Windows is os.DevNull
Browse files Browse the repository at this point in the history
Test added.

Fixes golang#24556

Change-Id: I4d1cd4513142edeea1a983fbfde46c2fccecab2a
Reviewed-on: https://go-review.googlesource.com/c/go/+/186139
Run-TryBot: Alex Brainman <alex.brainman@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Alex Brainman <alex.brainman@gmail.com>
  • Loading branch information
iwdgo authored and t4n6a1ka committed Sep 5, 2019
1 parent f84face commit bdda2db
Show file tree
Hide file tree
Showing 3 changed files with 39 additions and 20 deletions.
21 changes: 21 additions & 0 deletions src/os/file.go
Original file line number Diff line number Diff line change
Expand Up @@ -228,6 +228,9 @@ func (f *File) WriteString(s string) (n int, err error) {
// bits (before umask).
// If there is an error, it will be of type *PathError.
func Mkdir(name string, perm FileMode) error {
if runtime.GOOS == "windows" && isWindowsNulName(name) {
return &PathError{"mkdir", name, syscall.ENOTDIR}
}
e := syscall.Mkdir(fixLongPath(name), syscallMode(perm))

if e != nil {
Expand Down Expand Up @@ -560,3 +563,21 @@ func (f *File) SyscallConn() (syscall.RawConn, error) {
}
return newRawConn(f)
}

// isWindowsNulName reports whether name is os.DevNull ('NUL') on Windows.
// True is returned if name is 'NUL' whatever the case.
func isWindowsNulName(name string) bool {
if len(name) != 3 {
return false
}
if name[0] != 'n' && name[0] != 'N' {
return false
}
if name[1] != 'u' && name[1] != 'U' {
return false
}
if name[2] != 'l' && name[2] != 'L' {
return false
}
return true
}
16 changes: 16 additions & 0 deletions src/os/os_windows_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1168,3 +1168,19 @@ func TestWindowsReadlink(t *testing.T) {
mklink(t, "relfilelink", "file")
testReadlink(t, "relfilelink", "file")
}

// os.Mkdir(os.DevNull) fails.
func TestMkdirDevNull(t *testing.T) {
err := os.Mkdir(os.DevNull, 777)
oserr, ok := err.(*os.PathError)
if !ok {
t.Fatalf("error (%T) is not *os.PathError", err)
}
errno, ok := oserr.Err.(syscall.Errno)
if !ok {
t.Fatalf("error (%T) is not syscall.Errno", oserr)
}
if errno != syscall.ENOTDIR {
t.Fatalf("error %d is not syscall.ENOTDIR", errno)
}
}
22 changes: 2 additions & 20 deletions src/os/stat_windows.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,24 +10,6 @@ import (
"unsafe"
)

// isNulName reports whether name is NUL file name.
// For example, it returns true for both "NUL" and "nul".
func isNulName(name string) bool {
if len(name) != 3 {
return false
}
if name[0] != 'n' && name[0] != 'N' {
return false
}
if name[1] != 'u' && name[1] != 'U' {
return false
}
if name[2] != 'l' && name[2] != 'L' {
return false
}
return true
}

// Stat returns the FileInfo structure describing file.
// If there is an error, it will be of type *PathError.
func (file *File) Stat() (FileInfo, error) {
Expand All @@ -39,7 +21,7 @@ func (file *File) Stat() (FileInfo, error) {
// I don't know any better way to do that for directory
return Stat(file.dirinfo.path)
}
if isNulName(file.name) {
if isWindowsNulName(file.name) {
return &devNullStat, nil
}

Expand All @@ -65,7 +47,7 @@ func stat(funcname, name string, createFileAttrs uint32) (FileInfo, error) {
if len(name) == 0 {
return nil, &PathError{funcname, name, syscall.Errno(syscall.ERROR_PATH_NOT_FOUND)}
}
if isNulName(name) {
if isWindowsNulName(name) {
return &devNullStat, nil
}
namep, err := syscall.UTF16PtrFromString(fixLongPath(name))
Expand Down

0 comments on commit bdda2db

Please sign in to comment.