mirror of
https://github.com/minio/minio.git
synced 2025-01-12 15:33:22 -05:00
ad86454580
Continuing from PR 157ed65c35
Our posix.go implementation did not handle I/O errors
properly on the disks, this led to situations where
top-level callers such as ListObjects might return early
without even verifying all the available disks.
This commit tries to address this in Kubernetes, drbd/nbd based
persistent volumes which can disconnect under load and
result in the situations with disks return I/O errors.
This commit also simplifies listing operation, listing
never returns any error. We can avoid this since we pretty
much ignore most of the errors anyways. When objects are
accessed directly we return proper errors.
152 lines
3.4 KiB
Go
152 lines
3.4 KiB
Go
/*
|
|
* Minio Cloud Storage, (C) 2016, 2017 Minio, Inc.
|
|
*
|
|
* 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 cmd
|
|
|
|
import (
|
|
"os"
|
|
"runtime"
|
|
"syscall"
|
|
)
|
|
|
|
// Function not implemented error
|
|
func isSysErrNoSys(err error) bool {
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
switch pathErr.Err {
|
|
case syscall.ENOSYS:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Not supported error
|
|
func isSysErrOpNotSupported(err error) bool {
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
switch pathErr.Err {
|
|
case syscall.EOPNOTSUPP:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// No space left on device error
|
|
func isSysErrNoSpace(err error) bool {
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
switch pathErr.Err {
|
|
case syscall.ENOSPC:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Input/output error
|
|
func isSysErrIO(err error) bool {
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
switch pathErr.Err {
|
|
case syscall.EIO:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Check if the given error corresponds to EISDIR (is a directory).
|
|
func isSysErrIsDir(err error) bool {
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
switch pathErr.Err {
|
|
case syscall.EISDIR:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Check if the given error corresponds to ENOTDIR (is not a directory).
|
|
func isSysErrNotDir(err error) bool {
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
switch pathErr.Err {
|
|
case syscall.ENOTDIR:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Check if the given error corresponds to the ENAMETOOLONG (name too long).
|
|
func isSysErrTooLong(err error) bool {
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
switch pathErr.Err {
|
|
case syscall.ENAMETOOLONG:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Check if the given error corresponds to ENOTEMPTY for unix
|
|
// and ERROR_DIR_NOT_EMPTY for windows (directory not empty).
|
|
func isSysErrNotEmpty(err error) bool {
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
if runtime.GOOS == globalWindowsOSName {
|
|
if errno, _ok := pathErr.Err.(syscall.Errno); _ok && errno == 0x91 {
|
|
// ERROR_DIR_NOT_EMPTY
|
|
return true
|
|
}
|
|
}
|
|
switch pathErr.Err {
|
|
case syscall.ENOTEMPTY:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Check if the given error corresponds to the specific ERROR_PATH_NOT_FOUND for windows
|
|
func isSysErrPathNotFound(err error) bool {
|
|
if runtime.GOOS != globalWindowsOSName {
|
|
return false
|
|
}
|
|
if pathErr, ok := err.(*os.PathError); ok {
|
|
if errno, _ok := pathErr.Err.(syscall.Errno); _ok && errno == 0x03 {
|
|
// ERROR_PATH_NOT_FOUND
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Check if the given error corresponds to the specific ERROR_INVALID_HANDLE for windows
|
|
func isSysErrHandleInvalid(err error) bool {
|
|
if runtime.GOOS != globalWindowsOSName {
|
|
return false
|
|
}
|
|
// Check if err contains ERROR_INVALID_HANDLE errno
|
|
if errno, ok := err.(syscall.Errno); ok && errno == 0x6 {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func isSysErrCrossDevice(err error) bool {
|
|
if e, ok := err.(*os.LinkError); ok {
|
|
return e.Err == syscall.EXDEV
|
|
}
|
|
return false
|
|
}
|