mirror of
https://github.com/minio/minio.git
synced 2025-01-12 15:33:22 -05:00
server: Fixes for various conditions
- Fix distributed branch to be able to run FS version. - Fix distributed branch to be able to run XL local disks. - Ignore initialization failures of notification and bucket policies, the codepath should load whatever is possible.
This commit is contained in:
parent
67b8080144
commit
ba2ba328da
@ -265,12 +265,6 @@ func (api objectAPIHandlers) DeleteMultipleObjectsHandler(w http.ResponseWriter,
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
objectAPI := api.ObjectAPI()
|
|
||||||
if objectAPI == nil {
|
|
||||||
writeErrorResponse(w, r, ErrInternalError, r.URL.Path)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var wg = &sync.WaitGroup{} // Allocate a new wait group.
|
var wg = &sync.WaitGroup{} // Allocate a new wait group.
|
||||||
var dErrs = make([]error, len(deleteObjects.Objects))
|
var dErrs = make([]error, len(deleteObjects.Objects))
|
||||||
|
|
||||||
@ -279,7 +273,7 @@ func (api objectAPIHandlers) DeleteMultipleObjectsHandler(w http.ResponseWriter,
|
|||||||
wg.Add(1)
|
wg.Add(1)
|
||||||
go func(i int, obj ObjectIdentifier) {
|
go func(i int, obj ObjectIdentifier) {
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
dErr := api.ObjectAPI.DeleteObject(bucket, obj.ObjectName)
|
dErr := objectAPI.DeleteObject(bucket, obj.ObjectName)
|
||||||
if dErr != nil {
|
if dErr != nil {
|
||||||
dErrs[i] = dErr
|
dErrs[i] = dErr
|
||||||
}
|
}
|
||||||
|
@ -68,25 +68,34 @@ func loadAllBucketPolicies(objAPI ObjectLayer) (policies map[string]*bucketPolic
|
|||||||
buckets, err := objAPI.ListBuckets()
|
buckets, err := objAPI.ListBuckets()
|
||||||
errorIf(err, "Unable to list buckets.")
|
errorIf(err, "Unable to list buckets.")
|
||||||
err = errorCause(err)
|
err = errorCause(err)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
policies = make(map[string]*bucketPolicy)
|
policies = make(map[string]*bucketPolicy)
|
||||||
|
var pErrs []error
|
||||||
// Loads bucket policy.
|
// Loads bucket policy.
|
||||||
for _, bucket := range buckets {
|
for _, bucket := range buckets {
|
||||||
var policy *bucketPolicy
|
policy, pErr := readBucketPolicy(bucket.Name, objAPI)
|
||||||
policy, err = readBucketPolicy(bucket.Name, objAPI)
|
if pErr != nil {
|
||||||
if err != nil {
|
switch pErr.(type) {
|
||||||
switch err.(type) {
|
|
||||||
case BucketPolicyNotFound:
|
case BucketPolicyNotFound:
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
return nil, err
|
pErrs = append(pErrs, pErr)
|
||||||
|
// Continue to load other bucket policies if possible.
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
policies[bucket.Name] = policy
|
policies[bucket.Name] = policy
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Look for any errors occurred while reading bucket policies.
|
||||||
|
for _, pErr := range pErrs {
|
||||||
|
if pErr != nil {
|
||||||
|
return policies, pErr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Success.
|
// Success.
|
||||||
return policies, nil
|
return policies, nil
|
||||||
}
|
}
|
||||||
@ -96,6 +105,7 @@ func initBucketPolicies(objAPI ObjectLayer) error {
|
|||||||
if objAPI == nil {
|
if objAPI == nil {
|
||||||
return errInvalidArgument
|
return errInvalidArgument
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read all bucket policies.
|
// Read all bucket policies.
|
||||||
policies, err := loadAllBucketPolicies(objAPI)
|
policies, err := loadAllBucketPolicies(objAPI)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -130,22 +140,22 @@ func readBucketPolicyJSON(bucket string, objAPI ObjectLayer) (bucketPolicyReader
|
|||||||
}
|
}
|
||||||
policyPath := pathJoin(bucketConfigPrefix, bucket, policyJSON)
|
policyPath := pathJoin(bucketConfigPrefix, bucket, policyJSON)
|
||||||
objInfo, err := objAPI.GetObjectInfo(minioMetaBucket, policyPath)
|
objInfo, err := objAPI.GetObjectInfo(minioMetaBucket, policyPath)
|
||||||
errorIf(err, "Unable to get policy for the bucket %s.", bucket)
|
|
||||||
err = errorCause(err)
|
err = errorCause(err)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if _, ok := err.(ObjectNotFound); ok {
|
if _, ok := err.(ObjectNotFound); ok {
|
||||||
return nil, BucketPolicyNotFound{Bucket: bucket}
|
return nil, BucketPolicyNotFound{Bucket: bucket}
|
||||||
}
|
}
|
||||||
|
errorIf(err, "Unable to load policy for the bucket %s.", bucket)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
var buffer bytes.Buffer
|
var buffer bytes.Buffer
|
||||||
err = objAPI.GetObject(minioMetaBucket, policyPath, 0, objInfo.Size, &buffer)
|
err = objAPI.GetObject(minioMetaBucket, policyPath, 0, objInfo.Size, &buffer)
|
||||||
errorIf(err, "Unable to get policy for the bucket %s.", bucket)
|
|
||||||
err = errorCause(err)
|
err = errorCause(err)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if _, ok := err.(ObjectNotFound); ok {
|
if _, ok := err.(ObjectNotFound); ok {
|
||||||
return nil, BucketPolicyNotFound{Bucket: bucket}
|
return nil, BucketPolicyNotFound{Bucket: bucket}
|
||||||
}
|
}
|
||||||
|
errorIf(err, "Unable to load policy for the bucket %s.", bucket)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -227,7 +227,6 @@ func loadNotificationConfig(bucket string, objAPI ObjectLayer) (*notificationCon
|
|||||||
// Construct the notification config path.
|
// Construct the notification config path.
|
||||||
notificationConfigPath := path.Join(bucketConfigPrefix, bucket, bucketNotificationConfig)
|
notificationConfigPath := path.Join(bucketConfigPrefix, bucket, bucketNotificationConfig)
|
||||||
objInfo, err := objAPI.GetObjectInfo(minioMetaBucket, notificationConfigPath)
|
objInfo, err := objAPI.GetObjectInfo(minioMetaBucket, notificationConfigPath)
|
||||||
errorIf(err, "Unable to get bucket-notification for butkcet %s", bucket)
|
|
||||||
err = errorCause(err)
|
err = errorCause(err)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// 'notification.xml' not found return 'errNoSuchNotifications'.
|
// 'notification.xml' not found return 'errNoSuchNotifications'.
|
||||||
@ -236,12 +235,12 @@ func loadNotificationConfig(bucket string, objAPI ObjectLayer) (*notificationCon
|
|||||||
case ObjectNotFound:
|
case ObjectNotFound:
|
||||||
return nil, errNoSuchNotifications
|
return nil, errNoSuchNotifications
|
||||||
}
|
}
|
||||||
|
errorIf(err, "Unable to load bucket-notification for bucket %s", bucket)
|
||||||
// Returns error for other errors.
|
// Returns error for other errors.
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
var buffer bytes.Buffer
|
var buffer bytes.Buffer
|
||||||
err = objAPI.GetObject(minioMetaBucket, notificationConfigPath, 0, objInfo.Size, &buffer)
|
err = objAPI.GetObject(minioMetaBucket, notificationConfigPath, 0, objInfo.Size, &buffer)
|
||||||
errorIf(err, "Unable to get bucket-notification for butkcet %s", bucket)
|
|
||||||
err = errorCause(err)
|
err = errorCause(err)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// 'notification.xml' not found return 'errNoSuchNotifications'.
|
// 'notification.xml' not found return 'errNoSuchNotifications'.
|
||||||
@ -250,6 +249,7 @@ func loadNotificationConfig(bucket string, objAPI ObjectLayer) (*notificationCon
|
|||||||
case ObjectNotFound:
|
case ObjectNotFound:
|
||||||
return nil, errNoSuchNotifications
|
return nil, errNoSuchNotifications
|
||||||
}
|
}
|
||||||
|
errorIf(err, "Unable to load bucket-notification for bucket %s", bucket)
|
||||||
// Returns error for other errors.
|
// Returns error for other errors.
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -277,13 +277,12 @@ func loadAllBucketNotifications(objAPI ObjectLayer) (map[string]*notificationCon
|
|||||||
|
|
||||||
// Loads all bucket notifications.
|
// Loads all bucket notifications.
|
||||||
for _, bucket := range buckets {
|
for _, bucket := range buckets {
|
||||||
var nCfg *notificationConfig
|
nCfg, nErr := loadNotificationConfig(bucket.Name, objAPI)
|
||||||
nCfg, err = loadNotificationConfig(bucket.Name, objAPI)
|
if nErr != nil {
|
||||||
if err != nil {
|
if nErr == errNoSuchNotifications {
|
||||||
if err == errNoSuchNotifications {
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
return nil, err
|
return nil, nErr
|
||||||
}
|
}
|
||||||
configs[bucket.Name] = nCfg
|
configs[bucket.Name] = nCfg
|
||||||
}
|
}
|
||||||
|
@ -189,7 +189,7 @@ func loadAllFormats(bootstrapDisks []StorageAPI) ([]*formatConfigV1, []error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Return all formats and nil
|
// Return all formats and nil
|
||||||
return formatConfigs, nil
|
return formatConfigs, sErrs
|
||||||
}
|
}
|
||||||
|
|
||||||
// genericFormatCheck - validates and returns error.
|
// genericFormatCheck - validates and returns error.
|
||||||
|
@ -319,7 +319,7 @@ func (fs fsObjects) getObjectInfo(bucket, object string) (ObjectInfo, error) {
|
|||||||
}
|
}
|
||||||
fsMeta, err := readFSMetadata(fs.storage, minioMetaBucket, path.Join(bucketMetaPrefix, bucket, object, fsMetaJSONFile))
|
fsMeta, err := readFSMetadata(fs.storage, minioMetaBucket, path.Join(bucketMetaPrefix, bucket, object, fsMetaJSONFile))
|
||||||
// Ignore error if the metadata file is not found, other errors must be returned.
|
// Ignore error if the metadata file is not found, other errors must be returned.
|
||||||
if errorCause(err) != errFileNotFound {
|
if err != nil && errorCause(err) != errFileNotFound {
|
||||||
return ObjectInfo{}, toObjectErr(err, bucket, object)
|
return ObjectInfo{}, toObjectErr(err, bucket, object)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -530,7 +530,7 @@ func (fs fsObjects) ListObjects(bucket, prefix, marker, delimiter string, maxKey
|
|||||||
return FileInfo{}, traceError(err)
|
return FileInfo{}, traceError(err)
|
||||||
}
|
}
|
||||||
fsMeta, mErr := readFSMetadata(fs.storage, minioMetaBucket, path.Join(bucketMetaPrefix, bucket, entry, fsMetaJSONFile))
|
fsMeta, mErr := readFSMetadata(fs.storage, minioMetaBucket, path.Join(bucketMetaPrefix, bucket, entry, fsMetaJSONFile))
|
||||||
if errorCause(mErr) != errFileNotFound {
|
if mErr != nil && errorCause(mErr) != errFileNotFound {
|
||||||
return FileInfo{}, traceError(mErr)
|
return FileInfo{}, traceError(mErr)
|
||||||
}
|
}
|
||||||
if len(fsMeta.Meta) == 0 {
|
if len(fsMeta.Meta) == 0 {
|
||||||
|
@ -104,13 +104,8 @@ const (
|
|||||||
func prepForInit(disks []string, sErrs []error, diskCount int) InitActions {
|
func prepForInit(disks []string, sErrs []error, diskCount int) InitActions {
|
||||||
// Count errors by error value.
|
// Count errors by error value.
|
||||||
errMap := make(map[error]int)
|
errMap := make(map[error]int)
|
||||||
// If loadAllFormats returned successfully
|
for _, err := range sErrs {
|
||||||
if sErrs == nil {
|
errMap[err]++
|
||||||
errMap[nil] = diskCount
|
|
||||||
} else {
|
|
||||||
for _, err := range sErrs {
|
|
||||||
errMap[err]++
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
quorum := diskCount/2 + 1
|
quorum := diskCount/2 + 1
|
||||||
|
@ -17,7 +17,6 @@
|
|||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
@ -33,16 +32,19 @@ func newObjectLayerFn() ObjectLayer {
|
|||||||
|
|
||||||
// newObjectLayer - initialize any object layer depending on the number of disks.
|
// newObjectLayer - initialize any object layer depending on the number of disks.
|
||||||
func newObjectLayer(disks, ignoredDisks []string) (ObjectLayer, error) {
|
func newObjectLayer(disks, ignoredDisks []string) (ObjectLayer, error) {
|
||||||
|
var objAPI ObjectLayer
|
||||||
|
var err error
|
||||||
if len(disks) == 1 {
|
if len(disks) == 1 {
|
||||||
exportPath := disks[0]
|
|
||||||
// Initialize FS object layer.
|
// Initialize FS object layer.
|
||||||
return newFSObjects(exportPath)
|
objAPI, err = newFSObjects(disks[0])
|
||||||
|
} else {
|
||||||
|
// Initialize XL object layer.
|
||||||
|
objAPI, err = newXLObjects(disks, ignoredDisks)
|
||||||
}
|
}
|
||||||
// Initialize XL object layer.
|
if err != nil {
|
||||||
objAPI, err := newXLObjects(disks, ignoredDisks)
|
return nil, err
|
||||||
if err == errXLWriteQuorum {
|
|
||||||
return objAPI, errors.New("Disks are different with last minio server run.")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Migrate bucket policy from configDir to .minio.sys/buckets/
|
// Migrate bucket policy from configDir to .minio.sys/buckets/
|
||||||
err = migrateBucketPolicyConfig(objAPI)
|
err = migrateBucketPolicyConfig(objAPI)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -58,8 +60,10 @@ func newObjectLayer(disks, ignoredDisks []string) (ObjectLayer, error) {
|
|||||||
|
|
||||||
// Register the callback that should be called when the process shuts down.
|
// Register the callback that should be called when the process shuts down.
|
||||||
globalShutdownCBs.AddObjectLayerCB(func() errCode {
|
globalShutdownCBs.AddObjectLayerCB(func() errCode {
|
||||||
if sErr := objAPI.Shutdown(); sErr != nil {
|
if objAPI != nil {
|
||||||
return exitFailure
|
if sErr := objAPI.Shutdown(); sErr != nil {
|
||||||
|
return exitFailure
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return exitSuccess
|
return exitSuccess
|
||||||
})
|
})
|
||||||
@ -68,14 +72,12 @@ func newObjectLayer(disks, ignoredDisks []string) (ObjectLayer, error) {
|
|||||||
err = initEventNotifier(objAPI)
|
err = initEventNotifier(objAPI)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errorIf(err, "Unable to initialize event notification.")
|
errorIf(err, "Unable to initialize event notification.")
|
||||||
return nil, err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialize and load bucket policies.
|
// Initialize and load bucket policies.
|
||||||
err = initBucketPolicies(objAPI)
|
err = initBucketPolicies(objAPI)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errorIf(err, "Unable to load all bucket policies.")
|
errorIf(err, "Unable to load all bucket policies.")
|
||||||
return nil, err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Success.
|
// Success.
|
||||||
|
Loading…
Reference in New Issue
Block a user