2019-01-12 04:58:27 +00:00
|
|
|
/*
|
|
|
|
Copyright 2017 The Kubernetes Authors.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package removeall
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"syscall"
|
|
|
|
|
2020-12-01 01:06:26 +00:00
|
|
|
"k8s.io/mount-utils"
|
2019-01-12 04:58:27 +00:00
|
|
|
)
|
|
|
|
|
2021-08-12 22:41:05 +00:00
|
|
|
// RemoveAllOneFilesystemCommon removes the path and any children it contains,
|
|
|
|
// using the provided remove function. It removes everything it can but returns
|
|
|
|
// the first error it encounters. If the path does not exist, RemoveAll
|
2019-01-12 04:58:27 +00:00
|
|
|
// returns nil (no error).
|
|
|
|
// It makes sure it does not cross mount boundary, i.e. it does *not* remove
|
|
|
|
// files from another filesystems. Like 'rm -rf --one-file-system'.
|
|
|
|
// It is copied from RemoveAll() sources, with IsLikelyNotMountPoint
|
2021-08-12 22:41:05 +00:00
|
|
|
func RemoveAllOneFilesystemCommon(mounter mount.Interface, path string, remove func(string) error) error {
|
2019-01-12 04:58:27 +00:00
|
|
|
// Simple case: if Remove works, we're done.
|
2021-08-12 22:41:05 +00:00
|
|
|
err := remove(path)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err == nil || os.IsNotExist(err) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, is this a directory we need to recurse into?
|
|
|
|
dir, serr := os.Lstat(path)
|
|
|
|
if serr != nil {
|
|
|
|
if serr, ok := serr.(*os.PathError); ok && (os.IsNotExist(serr.Err) || serr.Err == syscall.ENOTDIR) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return serr
|
|
|
|
}
|
|
|
|
if !dir.IsDir() {
|
2021-08-12 22:41:05 +00:00
|
|
|
// Not a directory; return the error from remove.
|
2019-01-12 04:58:27 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Directory.
|
|
|
|
isNotMount, err := mounter.IsLikelyNotMountPoint(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !isNotMount {
|
|
|
|
return fmt.Errorf("cannot delete directory %s: it is a mount point", path)
|
|
|
|
}
|
|
|
|
|
|
|
|
fd, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
// Race. It was deleted between the Lstat and Open.
|
|
|
|
// Return nil per RemoveAll's docs.
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove contents & return first error.
|
|
|
|
err = nil
|
|
|
|
for {
|
|
|
|
names, err1 := fd.Readdirnames(100)
|
|
|
|
for _, name := range names {
|
2021-08-12 22:41:05 +00:00
|
|
|
err1 := RemoveAllOneFilesystemCommon(mounter, path+string(os.PathSeparator)+name, remove)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err == nil {
|
|
|
|
err = err1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err1 == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
// If Readdirnames returned an error, use it.
|
|
|
|
if err == nil {
|
|
|
|
err = err1
|
|
|
|
}
|
|
|
|
if len(names) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close directory, because windows won't remove opened directory.
|
|
|
|
fd.Close()
|
|
|
|
|
|
|
|
// Remove directory.
|
2021-08-12 22:41:05 +00:00
|
|
|
err1 := remove(path)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err1 == nil || os.IsNotExist(err1) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
err = err1
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
2021-08-12 22:41:05 +00:00
|
|
|
|
|
|
|
// RemoveAllOneFilesystem removes the path and any children it contains, using
|
|
|
|
// the os.Remove function. It makes sure it does not cross mount boundaries,
|
|
|
|
// i.e. it returns an error rather than remove files from another filesystem.
|
|
|
|
// It removes everything it can but returns the first error it encounters.
|
|
|
|
// If the path does not exist, it returns nil (no error).
|
|
|
|
func RemoveAllOneFilesystem(mounter mount.Interface, path string) error {
|
|
|
|
return RemoveAllOneFilesystemCommon(mounter, path, os.Remove)
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveDirsOneFilesystem removes the path and any empty subdirectories it
|
|
|
|
// contains, using the syscall.Rmdir function. Unlike RemoveAllOneFilesystem,
|
|
|
|
// RemoveDirsOneFilesystem will remove only directories and returns an error if
|
|
|
|
// it encounters any files in the directory tree. It makes sure it does not
|
|
|
|
// cross mount boundaries, i.e. it returns an error rather than remove dirs
|
|
|
|
// from another filesystem. It removes everything it can but returns the first
|
|
|
|
// error it encounters. If the path does not exist, it returns nil (no error).
|
|
|
|
func RemoveDirsOneFilesystem(mounter mount.Interface, path string) error {
|
|
|
|
return RemoveAllOneFilesystemCommon(mounter, path, syscall.Rmdir)
|
|
|
|
}
|