Implement os.RemoveAll()
TODO: enable test on windows once Readdir is implemented there
Этот коммит содержится в:
		
							родитель
							
								
									4417374b53
								
							
						
					
					
						коммит
						47a622a903
					
				
					 7 изменённых файлов: 676 добавлений и 7 удалений
				
			
		|  | @ -64,11 +64,6 @@ func Remove(path string) error { | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // RemoveAll is a stub, it is not implemented. |  | ||||||
| func RemoveAll(path string) error { |  | ||||||
| 	return ErrNotImplemented |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // Name returns the name of the file with which it was opened. | // Name returns the name of the file with which it was opened. | ||||||
| func (f *File) Name() string { | func (f *File) Name() string { | ||||||
| 	return f.name | 	return f.name | ||||||
|  |  | ||||||
|  | @ -1,5 +1,5 @@ | ||||||
| //go:build !baremetal && !js | //go:build !baremetal | ||||||
| // +build !baremetal,!js | // +build !baremetal | ||||||
| 
 | 
 | ||||||
| // Copyright 2009 The Go Authors. All rights reserved. | // Copyright 2009 The Go Authors. All rights reserved. | ||||||
| // Use of this source code is governed by a BSD-style | // Use of this source code is governed by a BSD-style | ||||||
|  | @ -60,3 +60,23 @@ func MkdirAll(path string, perm FileMode) error { | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // RemoveAll removes path and any children it contains. | ||||||
|  | // It removes everything it can but returns the first error | ||||||
|  | // it encounters. If the path does not exist, RemoveAll | ||||||
|  | // returns nil (no error). | ||||||
|  | // If there is an error, it will be of type *PathError. | ||||||
|  | func RemoveAll(path string) error { | ||||||
|  | 	return removeAll(path) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // endsWithDot reports whether the final component of path is ".". | ||||||
|  | func endsWithDot(path string) bool { | ||||||
|  | 	if path == "." { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  | 	if len(path) >= 2 && path[len(path)-1] == '.' && IsPathSeparator(path[len(path)-2]) { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										102
									
								
								src/os/read_test.go
									
										
									
									
									
										Обычный файл
									
								
							
							
						
						
									
										102
									
								
								src/os/read_test.go
									
										
									
									
									
										Обычный файл
									
								
							|  | @ -0,0 +1,102 @@ | ||||||
|  | //go:build !baremetal && !js && !wasi | ||||||
|  | // +build !baremetal,!js,!wasi | ||||||
|  | 
 | ||||||
|  | // Copyright 2009 The Go Authors. All rights reserved. | ||||||
|  | // Use of this source code is governed by a BSD-style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | package os_test | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	. "os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"testing" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func checkNamedSize(t *testing.T, path string, size int64) { | ||||||
|  | 	// TODO: this statement fails on wasi, possibly it objects to reading Stat on a symlink | ||||||
|  | 	dir, err := Stat(path) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("Stat %q (looking for size %d): %s", path, size, err) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	if dir.Size() != size { | ||||||
|  | 		t.Errorf("Stat %q: size %d want %d", path, dir.Size(), size) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestReadFile(t *testing.T) { | ||||||
|  | 	filename := "rumpelstilzchen" | ||||||
|  | 	contents, err := ReadFile(filename) | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Fatalf("ReadFile %s: error expected, none found", filename) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	filename = "read_test.go" | ||||||
|  | 	contents, err = ReadFile(filename) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("ReadFile %s: %v", filename, err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	checkNamedSize(t, filename, int64(len(contents))) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestWriteFile(t *testing.T) { | ||||||
|  | 	f, err := CreateTemp("", "ioutil-test") | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 	defer f.Close() | ||||||
|  | 	defer Remove(f.Name()) | ||||||
|  | 
 | ||||||
|  | 	msg := "Programming today is a race between software engineers striving to " + | ||||||
|  | 		"build bigger and better idiot-proof programs, and the Universe trying " + | ||||||
|  | 		"to produce bigger and better idiots. So far, the Universe is winning." | ||||||
|  | 
 | ||||||
|  | 	if err := WriteFile(f.Name(), []byte(msg), 0644); err != nil { | ||||||
|  | 		t.Fatalf("WriteFile %s: %v", f.Name(), err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	data, err := ReadFile(f.Name()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("ReadFile %s: %v", f.Name(), err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if string(data) != msg { | ||||||
|  | 		t.Fatalf("ReadFile: wrong data:\nhave %q\nwant %q", string(data), msg) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestReadOnlyWriteFile(t *testing.T) { | ||||||
|  | 	// TODO: also skip on wasi, where file permissions are ignored | ||||||
|  | 	if Getuid() == 0 { | ||||||
|  | 		t.Skipf("Root can write to read-only files anyway, so skip the read-only test.") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// We don't want to use CreateTemp directly, since that opens a file for us as 0600. | ||||||
|  | 	tempDir, err := MkdirTemp("", t.Name()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 	defer RemoveAll(tempDir) | ||||||
|  | 	filename := filepath.Join(tempDir, "blurp.txt") | ||||||
|  | 
 | ||||||
|  | 	shmorp := []byte("shmorp") | ||||||
|  | 	florp := []byte("florp") | ||||||
|  | 	err = WriteFile(filename, shmorp, 0444) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("WriteFile %s: %v", filename, err) | ||||||
|  | 	} | ||||||
|  | 	err = WriteFile(filename, florp, 0444) | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Fatalf("Expected an error when writing to read-only file %s", filename) | ||||||
|  | 	} | ||||||
|  | 	got, err := ReadFile(filename) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("ReadFile %s: %v", filename, err) | ||||||
|  | 	} | ||||||
|  | 	if !bytes.Equal(got, shmorp) { | ||||||
|  | 		t.Fatalf("want %s, got %s", shmorp, got) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										143
									
								
								src/os/removeall_noat.go
									
										
									
									
									
										Обычный файл
									
								
							
							
						
						
									
										143
									
								
								src/os/removeall_noat.go
									
										
									
									
									
										Обычный файл
									
								
							|  | @ -0,0 +1,143 @@ | ||||||
|  | // Copyright 2018 The Go Authors. All rights reserved. | ||||||
|  | // Use of this source code is governed by a BSD-style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | //go:build !baremetal && !js && !wasi | ||||||
|  | // +build !baremetal,!js,!wasi | ||||||
|  | 
 | ||||||
|  | package os | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"io" | ||||||
|  | 	"runtime" | ||||||
|  | 	"syscall" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func removeAll(path string) error { | ||||||
|  | 	if path == "" { | ||||||
|  | 		// fail silently to retain compatibility with previous behavior | ||||||
|  | 		// of RemoveAll. See issue 28830. | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// The rmdir system call permits removing "." on Plan 9, | ||||||
|  | 	// so we don't permit it to remain consistent with the | ||||||
|  | 	// "at" implementation of RemoveAll. | ||||||
|  | 	if endsWithDot(path) { | ||||||
|  | 		return &PathError{Op: "RemoveAll", Path: path, Err: syscall.EINVAL} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Simple case: if Remove works, we're done. | ||||||
|  | 	err := Remove(path) | ||||||
|  | 	if err == nil || IsNotExist(err) { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Otherwise, is this a directory we need to recurse into? | ||||||
|  | 	dir, serr := Lstat(path) | ||||||
|  | 	if serr != nil { | ||||||
|  | 		if serr, ok := serr.(*PathError); ok && (IsNotExist(serr.Err) || serr.Err == syscall.ENOTDIR) { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 		return serr | ||||||
|  | 	} | ||||||
|  | 	if !dir.IsDir() { | ||||||
|  | 		// Not a directory; return the error from Remove. | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Remove contents & return first error. | ||||||
|  | 	err = nil | ||||||
|  | 	for { | ||||||
|  | 		fd, err := Open(path) | ||||||
|  | 		if err != nil { | ||||||
|  | 			if IsNotExist(err) { | ||||||
|  | 				// Already deleted by someone else. | ||||||
|  | 				return nil | ||||||
|  | 			} | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		const reqSize = 1024 | ||||||
|  | 		var names []string | ||||||
|  | 		var readErr error | ||||||
|  | 
 | ||||||
|  | 		for { | ||||||
|  | 			numErr := 0 | ||||||
|  | 			names, readErr = fd.Readdirnames(reqSize) | ||||||
|  | 
 | ||||||
|  | 			for _, name := range names { | ||||||
|  | 				err1 := RemoveAll(path + string(PathSeparator) + name) | ||||||
|  | 				if err == nil { | ||||||
|  | 					err = err1 | ||||||
|  | 				} | ||||||
|  | 				if err1 != nil { | ||||||
|  | 					numErr++ | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			// If we can delete any entry, break to start new iteration. | ||||||
|  | 			// Otherwise, we discard current names, get next entries and try deleting them. | ||||||
|  | 			if numErr != reqSize { | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// Removing files from the directory may have caused | ||||||
|  | 		// the OS to reshuffle it. Simply calling Readdirnames | ||||||
|  | 		// again may skip some entries. The only reliable way | ||||||
|  | 		// to avoid this is to close and re-open the | ||||||
|  | 		// directory. See issue 20841. | ||||||
|  | 		fd.Close() | ||||||
|  | 
 | ||||||
|  | 		if readErr == io.EOF { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		// If Readdirnames returned an error, use it. | ||||||
|  | 		if err == nil { | ||||||
|  | 			err = readErr | ||||||
|  | 		} | ||||||
|  | 		if len(names) == 0 { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// We don't want to re-open unnecessarily, so if we | ||||||
|  | 		// got fewer than request names from Readdirnames, try | ||||||
|  | 		// simply removing the directory now. If that | ||||||
|  | 		// succeeds, we are done. | ||||||
|  | 		if len(names) < reqSize { | ||||||
|  | 			err1 := Remove(path) | ||||||
|  | 			if err1 == nil || IsNotExist(err1) { | ||||||
|  | 				return nil | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if err != nil { | ||||||
|  | 				// We got some error removing the | ||||||
|  | 				// directory contents, and since we | ||||||
|  | 				// read fewer names than we requested | ||||||
|  | 				// there probably aren't more files to | ||||||
|  | 				// remove. Don't loop around to read | ||||||
|  | 				// the directory again. We'll probably | ||||||
|  | 				// just get the same error. | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Remove directory. | ||||||
|  | 	err1 := Remove(path) | ||||||
|  | 	if err1 == nil || IsNotExist(err1) { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	if runtime.GOOS == "windows" && IsPermission(err1) { | ||||||
|  | 		if fs, err := Stat(path); err == nil { | ||||||
|  | 			if err = Chmod(path, FileMode(0200|int(fs.Mode()))); err == nil { | ||||||
|  | 				err1 = Remove(path) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if err == nil { | ||||||
|  | 		err = err1 | ||||||
|  | 	} | ||||||
|  | 	return err | ||||||
|  | } | ||||||
							
								
								
									
										16
									
								
								src/os/removeall_other.go
									
										
									
									
									
										Обычный файл
									
								
							
							
						
						
									
										16
									
								
								src/os/removeall_other.go
									
										
									
									
									
										Обычный файл
									
								
							|  | @ -0,0 +1,16 @@ | ||||||
|  | //go:build baremetal || js || wasi | ||||||
|  | // +build baremetal js wasi | ||||||
|  | 
 | ||||||
|  | // Copyright 2009 The Go Authors. All rights reserved. | ||||||
|  | // Use of this source code is governed by a BSD-style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | package os | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"syscall" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func removeAll(path string) error { | ||||||
|  | 	return &PathError{Op: "RemoveAll", Path: path, Err: syscall.ENOSYS} | ||||||
|  | } | ||||||
							
								
								
									
										390
									
								
								src/os/removeall_test.go
									
										
									
									
									
										Обычный файл
									
								
							
							
						
						
									
										390
									
								
								src/os/removeall_test.go
									
										
									
									
									
										Обычный файл
									
								
							|  | @ -0,0 +1,390 @@ | ||||||
|  | //go:build darwin || (linux && !baremetal && !js && !wasi) | ||||||
|  | // +build darwin linux,!baremetal,!js,!wasi | ||||||
|  | 
 | ||||||
|  | // TODO: implement ReadDir on windows | ||||||
|  | 
 | ||||||
|  | // Copyright 2018 The Go Authors. All rights reserved. | ||||||
|  | // Use of this source code is governed by a BSD-style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | package os_test | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	. "os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"runtime" | ||||||
|  | 	"testing" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestRemoveAll(t *testing.T) { | ||||||
|  | 	tmpDir, _ := os.MkdirTemp("", "TestRemoveAll") | ||||||
|  | 	if err := RemoveAll(""); err != nil { | ||||||
|  | 		t.Errorf("RemoveAll(\"\"): %v; want nil", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	file := filepath.Join(tmpDir, "file") | ||||||
|  | 	path := filepath.Join(tmpDir, "_TestRemoveAll_") | ||||||
|  | 	fpath := filepath.Join(path, "file") | ||||||
|  | 	dpath := filepath.Join(path, "dir") | ||||||
|  | 
 | ||||||
|  | 	// Make a regular file and remove | ||||||
|  | 	fd, err := Create(file) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("create %q: %s", file, err) | ||||||
|  | 	} | ||||||
|  | 	fd.Close() | ||||||
|  | 	if err = RemoveAll(file); err != nil { | ||||||
|  | 		t.Fatalf("RemoveAll %q (first): %s", file, err) | ||||||
|  | 	} | ||||||
|  | 	if _, err = Lstat(file); err == nil { | ||||||
|  | 		t.Fatalf("Lstat %q succeeded after RemoveAll (first)", file) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Make directory with 1 file and remove. | ||||||
|  | 	if err := MkdirAll(path, 0777); err != nil { | ||||||
|  | 		t.Fatalf("MkdirAll %q: %s", path, err) | ||||||
|  | 	} | ||||||
|  | 	fd, err = Create(fpath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("create %q: %s", fpath, err) | ||||||
|  | 	} | ||||||
|  | 	fd.Close() | ||||||
|  | 	if err = RemoveAll(path); err != nil { | ||||||
|  | 		t.Fatalf("RemoveAll %q (second): %s", path, err) | ||||||
|  | 	} | ||||||
|  | 	if _, err = Lstat(path); err == nil { | ||||||
|  | 		t.Fatalf("Lstat %q succeeded after RemoveAll (second)", path) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Make directory with file and subdirectory and remove. | ||||||
|  | 	if err = MkdirAll(dpath, 0777); err != nil { | ||||||
|  | 		t.Fatalf("MkdirAll %q: %s", dpath, err) | ||||||
|  | 	} | ||||||
|  | 	fd, err = Create(fpath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("create %q: %s", fpath, err) | ||||||
|  | 	} | ||||||
|  | 	fd.Close() | ||||||
|  | 	fd, err = Create(dpath + "/file") | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("create %q: %s", fpath, err) | ||||||
|  | 	} | ||||||
|  | 	fd.Close() | ||||||
|  | 	if err = RemoveAll(path); err != nil { | ||||||
|  | 		t.Fatalf("RemoveAll %q (third): %s", path, err) | ||||||
|  | 	} | ||||||
|  | 	if _, err := Lstat(path); err == nil { | ||||||
|  | 		t.Fatalf("Lstat %q succeeded after RemoveAll (third)", path) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Chmod is not supported under Windows and test fails as root. | ||||||
|  | 	if runtime.GOOS != "windows" && Getuid() != 0 { | ||||||
|  | 		// Make directory with file and subdirectory and trigger error. | ||||||
|  | 		if err = MkdirAll(dpath, 0777); err != nil { | ||||||
|  | 			t.Fatalf("MkdirAll %q: %s", dpath, err) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		for _, s := range []string{fpath, dpath + "/file1", path + "/zzz"} { | ||||||
|  | 			fd, err = Create(s) | ||||||
|  | 			if err != nil { | ||||||
|  | 				t.Fatalf("create %q: %s", s, err) | ||||||
|  | 			} | ||||||
|  | 			fd.Close() | ||||||
|  | 		} | ||||||
|  | 		if err = Chmod(dpath, 0); err != nil { | ||||||
|  | 			t.Fatalf("Chmod %q 0: %s", dpath, err) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// No error checking here: either RemoveAll | ||||||
|  | 		// will or won't be able to remove dpath; | ||||||
|  | 		// either way we want to see if it removes fpath | ||||||
|  | 		// and path/zzz. Reasons why RemoveAll might | ||||||
|  | 		// succeed in removing dpath as well include: | ||||||
|  | 		//	* running as root | ||||||
|  | 		//	* running on a file system without permissions (FAT) | ||||||
|  | 		RemoveAll(path) | ||||||
|  | 		Chmod(dpath, 0777) | ||||||
|  | 
 | ||||||
|  | 		for _, s := range []string{fpath, path + "/zzz"} { | ||||||
|  | 			if _, err = Lstat(s); err == nil { | ||||||
|  | 				t.Fatalf("Lstat %q succeeded after partial RemoveAll", s) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if err = RemoveAll(path); err != nil { | ||||||
|  | 		t.Fatalf("RemoveAll %q after partial RemoveAll: %s", path, err) | ||||||
|  | 	} | ||||||
|  | 	if _, err = Lstat(path); err == nil { | ||||||
|  | 		t.Fatalf("Lstat %q succeeded after RemoveAll (final)", path) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Test RemoveAll on a large directory. | ||||||
|  | func TestRemoveAllLarge(t *testing.T) { | ||||||
|  | 	if testing.Short() { | ||||||
|  | 		t.Skip("skipping in short mode") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	tmpDir, _ := os.MkdirTemp("", "TestRemoveAllLarge") | ||||||
|  | 	path := filepath.Join(tmpDir, "_TestRemoveAllLarge_") | ||||||
|  | 
 | ||||||
|  | 	// Make directory with 1000 files and remove. | ||||||
|  | 	if err := MkdirAll(path, 0777); err != nil { | ||||||
|  | 		t.Fatalf("MkdirAll %q: %s", path, err) | ||||||
|  | 	} | ||||||
|  | 	for i := 0; i < 1000; i++ { | ||||||
|  | 		fpath := fmt.Sprintf("%s/file%d", path, i) | ||||||
|  | 		fd, err := Create(fpath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			t.Fatalf("create %q: %s", fpath, err) | ||||||
|  | 		} | ||||||
|  | 		fd.Close() | ||||||
|  | 	} | ||||||
|  | 	if err := RemoveAll(path); err != nil { | ||||||
|  | 		t.Fatalf("RemoveAll %q: %s", path, err) | ||||||
|  | 	} | ||||||
|  | 	if _, err := Lstat(path); err == nil { | ||||||
|  | 		t.Fatalf("Lstat %q succeeded after RemoveAll", path) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestRemoveAllDot(t *testing.T) { | ||||||
|  | 	prevDir, err := Getwd() | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("Could not get wd: %s", err) | ||||||
|  | 	} | ||||||
|  | 	tempDir, err := os.MkdirTemp("", "TestRemoveAllDot-") | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("Could not create TempDir: %s", err) | ||||||
|  | 	} | ||||||
|  | 	defer RemoveAll(tempDir) | ||||||
|  | 
 | ||||||
|  | 	err = Chdir(tempDir) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("Could not chdir to tempdir: %s", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = RemoveAll(".") | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Errorf("RemoveAll succeed to remove .") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = Chdir(prevDir) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("Could not chdir %s: %s", prevDir, err) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestRemoveAllDotDot(t *testing.T) { | ||||||
|  | 	t.Parallel() | ||||||
|  | 
 | ||||||
|  | 	tempDir, _ := os.MkdirTemp("", "TestRemoveAllDotDot") | ||||||
|  | 	subdir := filepath.Join(tempDir, "x") | ||||||
|  | 	subsubdir := filepath.Join(subdir, "y") | ||||||
|  | 	if err := MkdirAll(subsubdir, 0777); err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 	if err := RemoveAll(filepath.Join(subsubdir, "..")); err != nil { | ||||||
|  | 		t.Error(err) | ||||||
|  | 	} | ||||||
|  | 	for _, dir := range []string{subsubdir, subdir} { | ||||||
|  | 		if _, err := Stat(dir); err == nil { | ||||||
|  | 			t.Errorf("%s: exists after RemoveAll", dir) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Issue #29178. | ||||||
|  | func TestRemoveReadOnlyDir(t *testing.T) { | ||||||
|  | 	t.Parallel() | ||||||
|  | 
 | ||||||
|  | 	tempDir, _ := os.MkdirTemp("", "TestRemoveReadOnlyDir") | ||||||
|  | 	subdir := filepath.Join(tempDir, "x") | ||||||
|  | 	if err := Mkdir(subdir, 0); err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// If an error occurs make it more likely that removing the | ||||||
|  | 	// temporary directory will succeed. | ||||||
|  | 	defer Chmod(subdir, 0777) | ||||||
|  | 
 | ||||||
|  | 	if err := RemoveAll(subdir); err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if _, err := Stat(subdir); err == nil { | ||||||
|  | 		t.Error("subdirectory was not removed") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Issue #29983. | ||||||
|  | func TestRemoveAllButReadOnlyAndPathError(t *testing.T) { | ||||||
|  | 	switch runtime.GOOS { | ||||||
|  | 	case "js", "windows": | ||||||
|  | 		t.Skipf("skipping test on %s", runtime.GOOS) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if Getuid() == 0 { | ||||||
|  | 		t.Skip("skipping test when running as root") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	t.Parallel() | ||||||
|  | 
 | ||||||
|  | 	tempDir, _ := os.MkdirTemp("", "TestRemoveAllButReadOnlyAndPathError") | ||||||
|  | 	dirs := []string{ | ||||||
|  | 		"a", | ||||||
|  | 		"a/x", | ||||||
|  | 		"a/x/1", | ||||||
|  | 		"b", | ||||||
|  | 		"b/y", | ||||||
|  | 		"b/y/2", | ||||||
|  | 		"c", | ||||||
|  | 		"c/z", | ||||||
|  | 		"c/z/3", | ||||||
|  | 	} | ||||||
|  | 	readonly := []string{ | ||||||
|  | 		"b", | ||||||
|  | 	} | ||||||
|  | 	inReadonly := func(d string) bool { | ||||||
|  | 		for _, ro := range readonly { | ||||||
|  | 			if d == ro { | ||||||
|  | 				return true | ||||||
|  | 			} | ||||||
|  | 			dd, _ := filepath.Split(d) | ||||||
|  | 			if filepath.Clean(dd) == ro { | ||||||
|  | 				return true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for _, dir := range dirs { | ||||||
|  | 		if err := Mkdir(filepath.Join(tempDir, dir), 0777); err != nil { | ||||||
|  | 			t.Fatal(err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	for _, dir := range readonly { | ||||||
|  | 		d := filepath.Join(tempDir, dir) | ||||||
|  | 		if err := Chmod(d, 0555); err != nil { | ||||||
|  | 			t.Fatal(err) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// Defer changing the mode back so that the deferred | ||||||
|  | 		// RemoveAll(tempDir) can succeed. | ||||||
|  | 		defer Chmod(d, 0777) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err := RemoveAll(tempDir) | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Fatal("RemoveAll succeeded unexpectedly") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// The error should be of type *PathError. | ||||||
|  | 	// see issue 30491 for details. | ||||||
|  | 	if pathErr, ok := err.(*PathError); ok { | ||||||
|  | 		want := filepath.Join(tempDir, "b", "y") | ||||||
|  | 		if pathErr.Path != want { | ||||||
|  | 			t.Errorf("RemoveAll(%q): err.Path=%q, want %q", tempDir, pathErr.Path, want) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		t.Errorf("RemoveAll(%q): error has type %T, want *fs.PathError", tempDir, err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for _, dir := range dirs { | ||||||
|  | 		_, err := Stat(filepath.Join(tempDir, dir)) | ||||||
|  | 		if inReadonly(dir) { | ||||||
|  | 			if err != nil { | ||||||
|  | 				t.Errorf("file %q was deleted but should still exist", dir) | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			if err == nil { | ||||||
|  | 				t.Errorf("file %q still exists but should have been deleted", dir) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestRemoveUnreadableDir(t *testing.T) { | ||||||
|  | 	switch runtime.GOOS { | ||||||
|  | 	case "js": | ||||||
|  | 		t.Skipf("skipping test on %s", runtime.GOOS) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if Getuid() == 0 { | ||||||
|  | 		t.Skip("skipping test when running as root") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	t.Parallel() | ||||||
|  | 
 | ||||||
|  | 	tempDir, _ := os.MkdirTemp("", "TestRemoveUnreadableDir") | ||||||
|  | 	target := filepath.Join(tempDir, "d0", "d1", "d2") | ||||||
|  | 	if err := MkdirAll(target, 0755); err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 	if err := Chmod(target, 0300); err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 	if err := RemoveAll(filepath.Join(tempDir, "d0")); err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Issue 29921 | ||||||
|  | func TestRemoveAllWithMoreErrorThanReqSize(t *testing.T) { | ||||||
|  | 	if testing.Short() { | ||||||
|  | 		t.Skip("skipping in short mode") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	tmpDir, _ := os.MkdirTemp("", "TestRemoveAllWithMoreErrorThanReqSize") | ||||||
|  | 	path := filepath.Join(tmpDir, "_TestRemoveAllWithMoreErrorThanReqSize_") | ||||||
|  | 
 | ||||||
|  | 	// Make directory with 1025 read-only files. | ||||||
|  | 	if err := MkdirAll(path, 0777); err != nil { | ||||||
|  | 		t.Fatalf("MkdirAll %q: %s", path, err) | ||||||
|  | 	} | ||||||
|  | 	for i := 0; i < 1025; i++ { | ||||||
|  | 		fpath := filepath.Join(path, fmt.Sprintf("file%d", i)) | ||||||
|  | 		fd, err := Create(fpath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			t.Fatalf("create %q: %s", fpath, err) | ||||||
|  | 		} | ||||||
|  | 		fd.Close() | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Make the parent directory read-only. On some platforms, this is what | ||||||
|  | 	// prevents os.Remove from removing the files within that directory. | ||||||
|  | 	if err := Chmod(path, 0555); err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 	defer Chmod(path, 0755) | ||||||
|  | 
 | ||||||
|  | 	// This call should not hang, even on a platform that disallows file deletion | ||||||
|  | 	// from read-only directories. | ||||||
|  | 	err := RemoveAll(path) | ||||||
|  | 
 | ||||||
|  | 	if Getuid() == 0 { | ||||||
|  | 		// On many platforms, root can remove files from read-only directories. | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	if err == nil { | ||||||
|  | 		if runtime.GOOS == "windows" { | ||||||
|  | 			// Marking a directory as read-only in Windows does not prevent the RemoveAll | ||||||
|  | 			// from creating or removing files within it. | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		t.Fatal("RemoveAll(<read-only directory>) = nil; want error") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	dir, err := Open(path) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  | 	defer dir.Close() | ||||||
|  | 
 | ||||||
|  | 	names, _ := dir.Readdirnames(1025) | ||||||
|  | 	if len(names) < 1025 { | ||||||
|  | 		t.Fatalf("RemoveAll(<read-only directory>) unexpectedly removed %d read-only files from that directory", 1025-len(names)) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | @ -2,6 +2,9 @@ | ||||||
| // Use of this source code is governed by a BSD-style | // Use of this source code is governed by a BSD-style | ||||||
| // license that can be found in the LICENSE file. | // license that can be found in the LICENSE file. | ||||||
| 
 | 
 | ||||||
|  | //go:build !baremetal && !js && !wasi | ||||||
|  | // +build !baremetal,!js,!wasi | ||||||
|  | 
 | ||||||
| package os_test | package os_test | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  |  | ||||||
		Загрузка…
	
	Создание таблицы
		
		Сослаться в новой задаче
	
	 Dan Kegel
						Dan Kegel