mirror of
https://github.com/minio/minio.git
synced 2025-04-04 11:50:36 -04:00
posix: Return errDiskNotWritable during disk initialization. (#2048)
It can happen that minio server might not have writable permissions on the export paths command line. Fixes #2035
This commit is contained in:
parent
e5dd917c37
commit
d64c3fd464
15
fs-v1.go
15
fs-v1.go
@ -85,8 +85,21 @@ func newFSObjects(disk string) (ObjectLayer, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Attempt to create `.minio`.
|
||||||
|
err = storage.MakeVol(minioMetaBucket)
|
||||||
|
if err != nil {
|
||||||
|
switch err {
|
||||||
|
// Ignore the errors.
|
||||||
|
case errVolumeExists, errDiskNotFound, errFaultyDisk:
|
||||||
|
default:
|
||||||
|
return nil, toObjectErr(err, minioMetaBucket)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Runs house keeping code, like creating minioMetaBucket, cleaning up tmp files etc.
|
// Runs house keeping code, like creating minioMetaBucket, cleaning up tmp files etc.
|
||||||
fsHouseKeeping(storage)
|
if err = fsHouseKeeping(storage); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
// loading format.json from minioMetaBucket.
|
// loading format.json from minioMetaBucket.
|
||||||
// Note: The format.json content is ignored, reserved for future use.
|
// Note: The format.json content is ignored, reserved for future use.
|
||||||
|
@ -45,17 +45,10 @@ func registerShutdown(callback func()) {
|
|||||||
|
|
||||||
// House keeping code needed for FS.
|
// House keeping code needed for FS.
|
||||||
func fsHouseKeeping(storageDisk StorageAPI) error {
|
func fsHouseKeeping(storageDisk StorageAPI) error {
|
||||||
// Attempt to create `.minio`.
|
|
||||||
err := storageDisk.MakeVol(minioMetaBucket)
|
|
||||||
if err != nil {
|
|
||||||
if err != errVolumeExists && err != errDiskNotFound && err != errFaultyDisk {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Cleanup all temp entries upon start.
|
// Cleanup all temp entries upon start.
|
||||||
err = cleanupDir(storageDisk, minioMetaBucket, tmpMetaPrefix)
|
err := cleanupDir(storageDisk, minioMetaBucket, tmpMetaPrefix)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return toObjectErr(err, minioMetaBucket, tmpMetaPrefix)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -70,8 +63,8 @@ func newStorageAPI(disk string) (storage StorageAPI, err error) {
|
|||||||
return newRPCClient(disk)
|
return newRPCClient(disk)
|
||||||
}
|
}
|
||||||
|
|
||||||
// House keeping code needed for XL.
|
// Initializes meta volume on all input storage disks.
|
||||||
func xlHouseKeeping(storageDisks []StorageAPI) error {
|
func initMetaVolume(storageDisks []StorageAPI) error {
|
||||||
// This happens for the first time, but keep this here since this
|
// This happens for the first time, but keep this here since this
|
||||||
// is the only place where it can be made expensive optimizing all
|
// is the only place where it can be made expensive optimizing all
|
||||||
// other calls. Create minio meta volume, if it doesn't exist yet.
|
// other calls. Create minio meta volume, if it doesn't exist yet.
|
||||||
@ -93,17 +86,58 @@ func xlHouseKeeping(storageDisks []StorageAPI) error {
|
|||||||
|
|
||||||
// Attempt to create `.minio`.
|
// Attempt to create `.minio`.
|
||||||
err := disk.MakeVol(minioMetaBucket)
|
err := disk.MakeVol(minioMetaBucket)
|
||||||
if err != nil && err != errVolumeExists && err != errDiskNotFound && err != errFaultyDisk {
|
if err != nil {
|
||||||
errs[index] = err
|
switch err {
|
||||||
return
|
// Ignored errors.
|
||||||
|
case errVolumeExists, errDiskNotFound, errFaultyDisk:
|
||||||
|
default:
|
||||||
|
errs[index] = err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
}(index, disk)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait for all cleanup to finish.
|
||||||
|
wg.Wait()
|
||||||
|
|
||||||
|
// Return upon first error.
|
||||||
|
for _, err := range errs {
|
||||||
|
if err == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return toObjectErr(err, minioMetaBucket)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return success here.
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// House keeping code needed for XL.
|
||||||
|
func xlHouseKeeping(storageDisks []StorageAPI) error {
|
||||||
|
// This happens for the first time, but keep this here since this
|
||||||
|
// is the only place where it can be made expensive optimizing all
|
||||||
|
// other calls. Create metavolume.
|
||||||
|
var wg = &sync.WaitGroup{}
|
||||||
|
|
||||||
|
// Initialize errs to collect errors inside go-routine.
|
||||||
|
var errs = make([]error, len(storageDisks))
|
||||||
|
|
||||||
|
// Initialize all disks in parallel.
|
||||||
|
for index, disk := range storageDisks {
|
||||||
|
if disk == nil {
|
||||||
|
errs[index] = errDiskNotFound
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
wg.Add(1)
|
||||||
|
go func(index int, disk StorageAPI) {
|
||||||
|
// Indicate this wait group is done.
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
// Cleanup all temp entries upon start.
|
// Cleanup all temp entries upon start.
|
||||||
err = cleanupDir(disk, minioMetaBucket, tmpMetaPrefix)
|
err := cleanupDir(disk, minioMetaBucket, tmpMetaPrefix)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errs[index] = err
|
errs[index] = err
|
||||||
return
|
|
||||||
}
|
}
|
||||||
errs[index] = nil
|
|
||||||
}(index, disk)
|
}(index, disk)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -144,7 +144,7 @@ func Test32kUNCPath(t *testing.T) {
|
|||||||
// The following calculation was derived empirically. It is not exactly MAX_PATH - len(longDiskName)
|
// The following calculation was derived empirically. It is not exactly MAX_PATH - len(longDiskName)
|
||||||
// possibly due to expansion rules as mentioned here -
|
// possibly due to expansion rules as mentioned here -
|
||||||
// https://msdn.microsoft.com/en-us/library/windows/desktop/aa365247(v=vs.85).aspx#maxpath
|
// https://msdn.microsoft.com/en-us/library/windows/desktop/aa365247(v=vs.85).aspx#maxpath
|
||||||
remaining := 32767 - 25 - len(longDiskName)
|
remaining := 32767 - 25 - len(longDiskName) - 10
|
||||||
longDiskName = longDiskName + `\` + strings.Repeat("a", remaining)
|
longDiskName = longDiskName + `\` + strings.Repeat("a", remaining)
|
||||||
}
|
}
|
||||||
err = mkdirAll(longDiskName, 0777)
|
err = mkdirAll(longDiskName, 0777)
|
||||||
|
9
posix.go
9
posix.go
@ -218,8 +218,13 @@ func (s *posix) MakeVol(volume string) (err error) {
|
|||||||
}
|
}
|
||||||
// Make a volume entry, with mode 0777 mkdir honors system umask.
|
// Make a volume entry, with mode 0777 mkdir honors system umask.
|
||||||
err = os.Mkdir(preparePath(volumeDir), 0777)
|
err = os.Mkdir(preparePath(volumeDir), 0777)
|
||||||
if err != nil && os.IsExist(err) {
|
if err != nil {
|
||||||
return errVolumeExists
|
if os.IsExist(err) {
|
||||||
|
return errVolumeExists
|
||||||
|
} else if os.IsPermission(err) {
|
||||||
|
return errDiskAccessDenied
|
||||||
|
}
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
// Success
|
// Success
|
||||||
return nil
|
return nil
|
||||||
|
@ -19,6 +19,7 @@ package main
|
|||||||
import (
|
import (
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
|
"syscall"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -114,3 +115,46 @@ func TestReadAll(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TestNewPosix all the cases handled in posix storage layer initialization.
|
||||||
|
func TestNewPosix(t *testing.T) {
|
||||||
|
// Temporary dir name.
|
||||||
|
tmpDirName := os.TempDir() + "/" + "minio-" + nextSuffix()
|
||||||
|
// Temporary file name.
|
||||||
|
tmpFileName := os.TempDir() + "/" + "minio-" + nextSuffix()
|
||||||
|
f, _ := os.Create(tmpFileName)
|
||||||
|
f.Close()
|
||||||
|
defer os.Remove(tmpFileName)
|
||||||
|
|
||||||
|
// List of all tests for posix initialization.
|
||||||
|
testCases := []struct {
|
||||||
|
diskPath string
|
||||||
|
err error
|
||||||
|
}{
|
||||||
|
// Validates input argument cannot be empty.
|
||||||
|
{
|
||||||
|
"",
|
||||||
|
errInvalidArgument,
|
||||||
|
},
|
||||||
|
// Validates if the directory does not exist and
|
||||||
|
// gets automatically created.
|
||||||
|
{
|
||||||
|
tmpDirName,
|
||||||
|
nil,
|
||||||
|
},
|
||||||
|
// Validates if the disk exists as file and returns error
|
||||||
|
// not a directory.
|
||||||
|
{
|
||||||
|
tmpFileName,
|
||||||
|
syscall.ENOTDIR,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate all test cases.
|
||||||
|
for i, testCase := range testCases {
|
||||||
|
_, err := newPosix(testCase.diskPath)
|
||||||
|
if err != testCase.err {
|
||||||
|
t.Fatalf("Test %d failed wanted: %s, got: %s", i+1, err, testCase.err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1912,7 +1912,7 @@ func (s *TestSuiteFS) TestObjectValidMD5(c *C) {
|
|||||||
client = http.Client{}
|
client = http.Client{}
|
||||||
response, err = client.Do(request)
|
response, err = client.Do(request)
|
||||||
c.Assert(err, IsNil)
|
c.Assert(err, IsNil)
|
||||||
// exepcting a successfull upload.
|
// exepcting a successful upload.
|
||||||
c.Assert(response.StatusCode, Equals, http.StatusOK)
|
c.Assert(response.StatusCode, Equals, http.StatusOK)
|
||||||
objectName = "test-2-object"
|
objectName = "test-2-object"
|
||||||
buffer1 = bytes.NewReader(data)
|
buffer1 = bytes.NewReader(data)
|
||||||
|
@ -33,6 +33,9 @@ var errDiskFull = errors.New("disk path full")
|
|||||||
// errDiskNotFount - cannot find the underlying configured disk anymore.
|
// errDiskNotFount - cannot find the underlying configured disk anymore.
|
||||||
var errDiskNotFound = errors.New("disk not found")
|
var errDiskNotFound = errors.New("disk not found")
|
||||||
|
|
||||||
|
// errDiskAccessDenied - we don't have write permissions on disk.
|
||||||
|
var errDiskAccessDenied = errors.New("disk access denied")
|
||||||
|
|
||||||
// errFileNotFound - cannot find the file.
|
// errFileNotFound - cannot find the file.
|
||||||
var errFileNotFound = errors.New("file not found")
|
var errFileNotFound = errors.New("file not found")
|
||||||
|
|
||||||
|
@ -69,12 +69,12 @@ const (
|
|||||||
var randN uint32
|
var randN uint32
|
||||||
var randmu sync.Mutex
|
var randmu sync.Mutex
|
||||||
|
|
||||||
// reseed - returns a new seed everytime the function is called.
|
// reseed - returns a new seed every time the function is called.
|
||||||
func reseed() uint32 {
|
func reseed() uint32 {
|
||||||
return uint32(time.Now().UnixNano() + int64(os.Getpid()))
|
return uint32(time.Now().UnixNano() + int64(os.Getpid()))
|
||||||
}
|
}
|
||||||
|
|
||||||
// nextSuffix - provides a new unique suffix everytime the function is called.
|
// nextSuffix - provides a new unique suffix every time the function is called.
|
||||||
func nextSuffix() string {
|
func nextSuffix() string {
|
||||||
randmu.Lock()
|
randmu.Lock()
|
||||||
r := randN
|
r := randN
|
||||||
|
11
xl-v1.go
11
xl-v1.go
@ -122,9 +122,6 @@ func newXLObjects(disks []string) (ObjectLayer, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Runs house keeping code, like creating minioMetaBucket, cleaning up tmp files etc.
|
|
||||||
xlHouseKeeping(storageDisks)
|
|
||||||
|
|
||||||
// Attempt to load all `format.json`.
|
// Attempt to load all `format.json`.
|
||||||
formatConfigs, sErrs := loadAllFormats(storageDisks)
|
formatConfigs, sErrs := loadAllFormats(storageDisks)
|
||||||
|
|
||||||
@ -138,6 +135,9 @@ func newXLObjects(disks []string) (ObjectLayer, error) {
|
|||||||
// Handles different cases properly.
|
// Handles different cases properly.
|
||||||
switch reduceFormatErrs(sErrs, len(storageDisks)) {
|
switch reduceFormatErrs(sErrs, len(storageDisks)) {
|
||||||
case errUnformattedDisk:
|
case errUnformattedDisk:
|
||||||
|
if err := initMetaVolume(storageDisks); err != nil {
|
||||||
|
return nil, fmt.Errorf("Unable to initialize '.minio' meta volume, %s", err)
|
||||||
|
}
|
||||||
// All drives online but fresh, initialize format.
|
// All drives online but fresh, initialize format.
|
||||||
if err := initFormatXL(storageDisks); err != nil {
|
if err := initFormatXL(storageDisks); err != nil {
|
||||||
return nil, fmt.Errorf("Unable to initialize format, %s", err)
|
return nil, fmt.Errorf("Unable to initialize format, %s", err)
|
||||||
@ -153,6 +153,11 @@ func newXLObjects(disks []string) (ObjectLayer, error) {
|
|||||||
// FIXME.
|
// FIXME.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Runs house keeping code, like t, cleaning up tmp files etc.
|
||||||
|
if err := xlHouseKeeping(storageDisks); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
// Load saved XL format.json and validate.
|
// Load saved XL format.json and validate.
|
||||||
newPosixDisks, err := loadFormatXL(storageDisks)
|
newPosixDisks, err := loadFormatXL(storageDisks)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user