/* * MinIO Cloud Storage, (C) 2018 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 ( "context" "fmt" "hash/crc32" "io" "net/http" "sort" "strings" "sync" "time" "github.com/minio/minio/cmd/logger" "github.com/minio/minio/pkg/bpool" "github.com/minio/minio/pkg/lifecycle" "github.com/minio/minio/pkg/madmin" "github.com/minio/minio/pkg/policy" "github.com/minio/minio/pkg/sync/errgroup" ) // setsStorageAPI is encapsulated type for Close() type setsStorageAPI [][]StorageAPI func (s setsStorageAPI) Close() error { for i := 0; i < len(s); i++ { for _, disk := range s[i] { if disk == nil { continue } disk.Close() } } return nil } // xlSets implements ObjectLayer combining a static list of erasure coded // object sets. NOTE: There is no dynamic scaling allowed or intended in // current design. type xlSets struct { sets []*xlObjects // Reference format. format *formatXLV3 // xlDisks mutex to lock xlDisks. xlDisksMu sync.RWMutex // Re-ordered list of disks per set. xlDisks setsStorageAPI // List of endpoints provided on the command line. endpoints EndpointList // Total number of sets and the number of disks per set. setCount, drivesPerSet int // Done channel to control monitoring loop. disksConnectDoneCh chan struct{} // Distribution algorithm of choice. distributionAlgo string // Merge tree walk pool *MergeWalkPool } // isConnected - checks if the endpoint is connected or not. func (s *xlSets) isConnected(endpoint Endpoint) bool { s.xlDisksMu.RLock() defer s.xlDisksMu.RUnlock() for i := 0; i < s.setCount; i++ { for j := 0; j < s.drivesPerSet; j++ { if s.xlDisks[i][j] == nil { continue } var endpointStr string if endpoint.IsLocal { endpointStr = endpoint.Path } else { endpointStr = endpoint.String() } if s.xlDisks[i][j].String() != endpointStr { continue } return s.xlDisks[i][j].IsOnline() } } return false } // Initializes a new StorageAPI from the endpoint argument, returns // StorageAPI and also `format` which exists on the disk. func connectEndpoint(endpoint Endpoint) (StorageAPI, *formatXLV3, error) { disk, err := newStorageAPI(endpoint) if err != nil { return nil, nil, err } format, err := loadFormatXL(disk) if err != nil { // Close the internal connection to avoid connection leaks. disk.Close() return nil, nil, err } return disk, format, nil } // findDiskIndex - returns the i,j'th position of the input `format` against the reference // format, after successful validation. func findDiskIndex(refFormat, format *formatXLV3) (int, int, error) { if err := formatXLV3Check(refFormat, format); err != nil { return 0, 0, err } if format.XL.This == offlineDiskUUID { return -1, -1, fmt.Errorf("diskID: %s is offline", format.XL.This) } for i := 0; i < len(refFormat.XL.Sets); i++ { for j := 0; j < len(refFormat.XL.Sets[0]); j++ { if refFormat.XL.Sets[i][j] == format.XL.This { return i, j, nil } } } return -1, -1, fmt.Errorf("diskID: %s not found", format.XL.This) } // Re initializes all disks based on the reference format, this function is // only used by HealFormat and ReloadFormat calls. func (s *xlSets) reInitDisks(refFormat *formatXLV3, storageDisks []StorageAPI, formats []*formatXLV3) [][]StorageAPI { xlDisks := make([][]StorageAPI, s.setCount) for i := 0; i < len(refFormat.XL.Sets); i++ { xlDisks[i] = make([]StorageAPI, s.drivesPerSet) } for k := range storageDisks { if storageDisks[k] == nil || formats[k] == nil { continue } i, j, err := findDiskIndex(refFormat, formats[k]) if err != nil { reqInfo := (&logger.ReqInfo{}).AppendTags("storageDisk", storageDisks[i].String()) ctx := logger.SetReqInfo(context.Background(), reqInfo) logger.LogIf(ctx, err) continue } xlDisks[i][j] = storageDisks[k] } return xlDisks } // connectDisksWithQuorum is same as connectDisks but waits // for quorum number of formatted disks to be online in // any given sets. func (s *xlSets) connectDisksWithQuorum() { var onlineDisks int for onlineDisks < len(s.endpoints)/2 { for _, endpoint := range s.endpoints { if s.isConnected(endpoint) { continue } disk, format, err := connectEndpoint(endpoint) if err != nil { printEndpointError(endpoint, err) continue } i, j, err := findDiskIndex(s.format, format) if err != nil { // Close the internal connection to avoid connection leaks. disk.Close() printEndpointError(endpoint, err) continue } s.xlDisks[i][j] = disk onlineDisks++ } // Sleep for a while - so that we don't go into // 100% CPU when half the disks are online. time.Sleep(500 * time.Millisecond) } } // connectDisks - attempt to connect all the endpoints, loads format // and re-arranges the disks in proper position. func (s *xlSets) connectDisks() { for _, endpoint := range s.endpoints { if s.isConnected(endpoint) { continue } disk, format, err := connectEndpoint(endpoint) if err != nil { printEndpointError(endpoint, err) continue } i, j, err := findDiskIndex(s.format, format) if err != nil { // Close the internal connection to avoid connection leaks. disk.Close() printEndpointError(endpoint, err) continue } s.xlDisksMu.Lock() s.xlDisks[i][j] = disk s.xlDisksMu.Unlock() } } // monitorAndConnectEndpoints this is a monitoring loop to keep track of disconnected // endpoints by reconnecting them and making sure to place them into right position in // the set topology, this monitoring happens at a given monitoring interval. func (s *xlSets) monitorAndConnectEndpoints(monitorInterval time.Duration) { ticker := time.NewTicker(monitorInterval) // Stop the timer. defer ticker.Stop() for { select { case <-GlobalServiceDoneCh: return case <-s.disksConnectDoneCh: return case <-ticker.C: s.connectDisks() } } } // GetDisks returns a closure for a given set, which provides list of disks per set. func (s *xlSets) GetDisks(setIndex int) func() []StorageAPI { return func() []StorageAPI { s.xlDisksMu.Lock() defer s.xlDisksMu.Unlock() disks := make([]StorageAPI, s.drivesPerSet) copy(disks, s.xlDisks[setIndex]) return disks } } const defaultMonitorConnectEndpointInterval = time.Second * 10 // Set to 10 secs. // Initialize new set of erasure coded sets. func newXLSets(endpoints EndpointList, format *formatXLV3, setCount int, drivesPerSet int) (ObjectLayer, error) { // Initialize the XL sets instance. s := &xlSets{ sets: make([]*xlObjects, setCount), xlDisks: make([][]StorageAPI, setCount), endpoints: endpoints, setCount: setCount, drivesPerSet: drivesPerSet, format: format, disksConnectDoneCh: make(chan struct{}), distributionAlgo: format.XL.DistributionAlgo, pool: NewMergeWalkPool(globalMergeLookupTimeout), } mutex := newNSLock(globalIsDistXL) // Initialize byte pool once for all sets, bpool size is set to // setCount * drivesPerSet with each memory upto blockSizeV1. bp := bpool.NewBytePoolCap(setCount*drivesPerSet, blockSizeV1, blockSizeV1*2) for i := 0; i < len(format.XL.Sets); i++ { s.xlDisks[i] = make([]StorageAPI, drivesPerSet) // Initialize xl objects for a given set. s.sets[i] = &xlObjects{ getDisks: s.GetDisks(i), nsMutex: mutex, bp: bp, } go s.sets[i].cleanupStaleMultipartUploads(context.Background(), GlobalMultipartCleanupInterval, GlobalMultipartExpiry, GlobalServiceDoneCh) } // Connect disks right away, but wait until we have `format.json` quorum. s.connectDisksWithQuorum() // Start the disk monitoring and connect routine. go s.monitorAndConnectEndpoints(defaultMonitorConnectEndpointInterval) return s, nil } // StorageInfo - combines output of StorageInfo across all erasure coded object sets. func (s *xlSets) StorageInfo(ctx context.Context) StorageInfo { var storageInfo StorageInfo var wg sync.WaitGroup storageInfos := make([]StorageInfo, len(s.sets)) storageInfo.Backend.Type = BackendErasure for index, set := range s.sets { wg.Add(1) go func(id int, set *xlObjects) { defer wg.Done() storageInfos[id] = set.StorageInfo(ctx) }(index, set) } // Wait for the go routines. wg.Wait() for _, lstorageInfo := range storageInfos { storageInfo.Used += lstorageInfo.Used storageInfo.Total += lstorageInfo.Total storageInfo.Available += lstorageInfo.Available storageInfo.Backend.OnlineDisks += lstorageInfo.Backend.OnlineDisks storageInfo.Backend.OfflineDisks += lstorageInfo.Backend.OfflineDisks } scData, scParity := getRedundancyCount(standardStorageClass, s.drivesPerSet) storageInfo.Backend.StandardSCData = scData storageInfo.Backend.StandardSCParity = scParity rrSCData, rrSCparity := getRedundancyCount(reducedRedundancyStorageClass, s.drivesPerSet) storageInfo.Backend.RRSCData = rrSCData storageInfo.Backend.RRSCParity = rrSCparity storageInfo.Backend.Sets = make([][]madmin.DriveInfo, s.setCount) for i := range storageInfo.Backend.Sets { storageInfo.Backend.Sets[i] = make([]madmin.DriveInfo, s.drivesPerSet) } storageDisks, dErrs := initDisksWithErrors(s.endpoints) defer closeStorageDisks(storageDisks) formats, sErrs := loadFormatXLAll(storageDisks) combineStorageErrors := func(diskErrs []error, storageErrs []error) []error { for index, err := range diskErrs { if err != nil { storageErrs[index] = err } } return storageErrs } errs := combineStorageErrors(dErrs, sErrs) drivesInfo := formatsToDrivesInfo(s.endpoints, formats, errs) refFormat, err := getFormatXLInQuorum(formats) if err != nil { // Ignore errors here, since this call cannot do anything at // this point. too many disks are down already. return storageInfo } // fill all the available/online endpoints for _, drive := range drivesInfo { if drive.UUID == "" { continue } for i := range refFormat.XL.Sets { for j, driveUUID := range refFormat.XL.Sets[i] { if driveUUID == drive.UUID { storageInfo.Backend.Sets[i][j] = drive } } } } // fill all the offline, missing endpoints as well. for _, drive := range drivesInfo { if drive.UUID == "" { for i := range storageInfo.Backend.Sets { for j := range storageInfo.Backend.Sets[i] { if storageInfo.Backend.Sets[i][j].Endpoint == drive.Endpoint { continue } if storageInfo.Backend.Sets[i][j].Endpoint == "" { storageInfo.Backend.Sets[i][j] = drive break } } } } } return storageInfo } // Shutdown shutsdown all erasure coded sets in parallel // returns error upon first error. func (s *xlSets) Shutdown(ctx context.Context) error { g := errgroup.WithNErrs(len(s.sets)) for index := range s.sets { index := index g.Go(func() error { return s.sets[index].Shutdown(ctx) }, index) } for _, err := range g.Wait() { if err != nil { return err } } return nil } // MakeBucketLocation - creates a new bucket across all sets simultaneously // even if one of the sets fail to create buckets, we proceed to undo a // successful operation. func (s *xlSets) MakeBucketWithLocation(ctx context.Context, bucket, location string) error { g := errgroup.WithNErrs(len(s.sets)) // Create buckets in parallel across all sets. for index := range s.sets { index := index g.Go(func() error { return s.sets[index].MakeBucketWithLocation(ctx, bucket, location) }, index) } errs := g.Wait() // Upon even a single write quorum error we undo all previously created buckets. for _, err := range errs { if err != nil { if _, ok := err.(InsufficientWriteQuorum); ok { undoMakeBucketSets(bucket, s.sets, errs) } return err } } // Success. return nil } // This function is used to undo a successful MakeBucket operation. func undoMakeBucketSets(bucket string, sets []*xlObjects, errs []error) { g := errgroup.WithNErrs(len(sets)) // Undo previous make bucket entry on all underlying sets. for index := range sets { index := index if errs[index] == nil { g.Go(func() error { return sets[index].DeleteBucket(context.Background(), bucket) }, index) } } // Wait for all delete bucket to finish. g.Wait() } // hashes the key returning an integer based on the input algorithm. // This function currently supports // - CRCMOD // - all new algos. func crcHashMod(key string, cardinality int) int { if cardinality <= 0 { return -1 } keyCrc := crc32.Checksum([]byte(key), crc32.IEEETable) return int(keyCrc % uint32(cardinality)) } func hashKey(algo string, key string, cardinality int) int { switch algo { case formatXLVersionV2DistributionAlgo: return crcHashMod(key, cardinality) default: // Unknown algorithm returns -1, also if cardinality is lesser than 0. return -1 } } // Returns always a same erasure coded set for a given input. func (s *xlSets) getHashedSetIndex(input string) int { return hashKey(s.distributionAlgo, input, len(s.sets)) } // Returns always a same erasure coded set for a given input. func (s *xlSets) getHashedSet(input string) (set *xlObjects) { return s.sets[s.getHashedSetIndex(input)] } // GetBucketInfo - returns bucket info from one of the erasure coded set. func (s *xlSets) GetBucketInfo(ctx context.Context, bucket string) (bucketInfo BucketInfo, err error) { return s.getHashedSet(bucket).GetBucketInfo(ctx, bucket) } // ListObjectsV2 lists all objects in bucket filtered by prefix func (s *xlSets) ListObjectsV2(ctx context.Context, bucket, prefix, continuationToken, delimiter string, maxKeys int, fetchOwner bool, startAfter string) (result ListObjectsV2Info, err error) { marker := continuationToken if marker == "" { marker = startAfter } loi, err := s.ListObjects(ctx, bucket, prefix, marker, delimiter, maxKeys) if err != nil { return result, err } listObjectsV2Info := ListObjectsV2Info{ IsTruncated: loi.IsTruncated, ContinuationToken: continuationToken, NextContinuationToken: loi.NextMarker, Objects: loi.Objects, Prefixes: loi.Prefixes, } return listObjectsV2Info, err } // SetBucketPolicy persist the new policy on the bucket. func (s *xlSets) SetBucketPolicy(ctx context.Context, bucket string, policy *policy.Policy) error { return savePolicyConfig(ctx, s, bucket, policy) } // GetBucketPolicy will return a policy on a bucket func (s *xlSets) GetBucketPolicy(ctx context.Context, bucket string) (*policy.Policy, error) { return getPolicyConfig(s, bucket) } // DeleteBucketPolicy deletes all policies on bucket func (s *xlSets) DeleteBucketPolicy(ctx context.Context, bucket string) error { return removePolicyConfig(ctx, s, bucket) } // SetBucketLifecycle sets lifecycle on bucket func (s *xlSets) SetBucketLifecycle(ctx context.Context, bucket string, lifecycle *lifecycle.Lifecycle) error { return saveLifecycleConfig(ctx, s, bucket, lifecycle) } // GetBucketLifecycle will get lifecycle on bucket func (s *xlSets) GetBucketLifecycle(ctx context.Context, bucket string) (*lifecycle.Lifecycle, error) { return getLifecycleConfig(s, bucket) } // DeleteBucketLifecycle deletes all lifecycle on bucket func (s *xlSets) DeleteBucketLifecycle(ctx context.Context, bucket string) error { return removeLifecycleConfig(ctx, s, bucket) } // IsNotificationSupported returns whether bucket notification is applicable for this layer. func (s *xlSets) IsNotificationSupported() bool { return s.getHashedSet("").IsNotificationSupported() } // IsListenBucketSupported returns whether listen bucket notification is applicable for this layer. func (s *xlSets) IsListenBucketSupported() bool { return true } // IsEncryptionSupported returns whether server side encryption is implemented for this layer. func (s *xlSets) IsEncryptionSupported() bool { return s.getHashedSet("").IsEncryptionSupported() } // IsCompressionSupported returns whether compression is applicable for this layer. func (s *xlSets) IsCompressionSupported() bool { return s.getHashedSet("").IsCompressionSupported() } // DeleteBucket - deletes a bucket on all sets simultaneously, // even if one of the sets fail to delete buckets, we proceed to // undo a successful operation. func (s *xlSets) DeleteBucket(ctx context.Context, bucket string) error { g := errgroup.WithNErrs(len(s.sets)) // Delete buckets in parallel across all sets. for index := range s.sets { index := index g.Go(func() error { return s.sets[index].DeleteBucket(ctx, bucket) }, index) } errs := g.Wait() // For any write quorum failure, we undo all the delete buckets operation // by creating all the buckets again. for _, err := range errs { if err != nil { if _, ok := err.(InsufficientWriteQuorum); ok { undoDeleteBucketSets(bucket, s.sets, errs) } return err } } // Delete all bucket metadata. deleteBucketMetadata(ctx, bucket, s) // Success. return nil } // This function is used to undo a successful DeleteBucket operation. func undoDeleteBucketSets(bucket string, sets []*xlObjects, errs []error) { g := errgroup.WithNErrs(len(sets)) // Undo previous delete bucket on all underlying sets. for index := range sets { index := index if errs[index] == nil { g.Go(func() error { return sets[index].MakeBucketWithLocation(context.Background(), bucket, "") }, index) } } g.Wait() } // List all buckets from one of the set, we are not doing merge // sort here just for simplification. As per design it is assumed // that all buckets are present on all sets. func (s *xlSets) ListBuckets(ctx context.Context) (buckets []BucketInfo, err error) { // Always lists from the same set signified by the empty string. return s.getHashedSet("").ListBuckets(ctx) } // --- Object Operations --- // GetObjectNInfo - returns object info and locked object ReadCloser func (s *xlSets) GetObjectNInfo(ctx context.Context, bucket, object string, rs *HTTPRangeSpec, h http.Header, lockType LockType, opts ObjectOptions) (gr *GetObjectReader, err error) { return s.getHashedSet(object).GetObjectNInfo(ctx, bucket, object, rs, h, lockType, opts) } // GetObject - reads an object from the hashedSet based on the object name. func (s *xlSets) GetObject(ctx context.Context, bucket, object string, startOffset int64, length int64, writer io.Writer, etag string, opts ObjectOptions) error { return s.getHashedSet(object).GetObject(ctx, bucket, object, startOffset, length, writer, etag, opts) } // PutObject - writes an object to hashedSet based on the object name. func (s *xlSets) PutObject(ctx context.Context, bucket string, object string, data *PutObjReader, opts ObjectOptions) (objInfo ObjectInfo, err error) { return s.getHashedSet(object).PutObject(ctx, bucket, object, data, opts) } // GetObjectInfo - reads object metadata from the hashedSet based on the object name. func (s *xlSets) GetObjectInfo(ctx context.Context, bucket, object string, opts ObjectOptions) (objInfo ObjectInfo, err error) { return s.getHashedSet(object).GetObjectInfo(ctx, bucket, object, opts) } // DeleteObject - deletes an object from the hashedSet based on the object name. func (s *xlSets) DeleteObject(ctx context.Context, bucket string, object string) (err error) { return s.getHashedSet(object).DeleteObject(ctx, bucket, object) } // DeleteObjects - bulk delete of objects // Bulk delete is only possible within one set. For that purpose // objects are group by set first, and then bulk delete is invoked // for each set, the error response of each delete will be returned func (s *xlSets) DeleteObjects(ctx context.Context, bucket string, objects []string) ([]error, error) { type delObj struct { // Set index associated to this object setIndex int // Original index from the list of arguments // where this object is passed origIndex int // Object name name string } // Transform []delObj to the list of object names toNames := func(delObjs []delObj) []string { names := make([]string, len(delObjs)) for i, obj := range delObjs { names[i] = obj.name } return names } // The result of delete operation on all passed objects var delErrs = make([]error, len(objects)) // A map between a set and its associated objects var objSetMap = make(map[int][]delObj) // Group objects by set index for i, object := range objects { index := s.getHashedSetIndex(object) objSetMap[index] = append(objSetMap[index], delObj{setIndex: index, origIndex: i, name: object}) } // Invoke bulk delete on objects per set and save // the result of the delete operation for _, objsGroup := range objSetMap { errs, err := s.getHashedSet(objsGroup[0].name).DeleteObjects(ctx, bucket, toNames(objsGroup)) if err != nil { return nil, err } for i, obj := range objsGroup { delErrs[obj.origIndex] = errs[i] } } return delErrs, nil } // CopyObject - copies objects from one hashedSet to another hashedSet, on server side. func (s *xlSets) CopyObject(ctx context.Context, srcBucket, srcObject, destBucket, destObject string, srcInfo ObjectInfo, srcOpts, dstOpts ObjectOptions) (objInfo ObjectInfo, err error) { srcSet := s.getHashedSet(srcObject) destSet := s.getHashedSet(destObject) // Check if this request is only metadata update. cpSrcDstSame := isStringEqual(pathJoin(srcBucket, srcObject), pathJoin(destBucket, destObject)) if cpSrcDstSame && srcInfo.metadataOnly { return srcSet.CopyObject(ctx, srcBucket, srcObject, destBucket, destObject, srcInfo, srcOpts, dstOpts) } if !cpSrcDstSame { objectDWLock := destSet.nsMutex.NewNSLock(ctx, destBucket, destObject) if err := objectDWLock.GetLock(globalObjectTimeout); err != nil { return objInfo, err } defer objectDWLock.Unlock() } putOpts := ObjectOptions{ServerSideEncryption: dstOpts.ServerSideEncryption, UserDefined: srcInfo.UserDefined} return destSet.putObject(ctx, destBucket, destObject, srcInfo.PutObjReader, putOpts) } // Returns function "listDir" of the type listDirFunc. // disks - used for doing disk.ListDir(). Sets passes set of disks. func listDirSetsFactory(ctx context.Context, sets ...*xlObjects) ListDirFunc { listDirInternal := func(bucket, prefixDir, prefixEntry string, disks []StorageAPI) (mergedEntries []string) { var diskEntries = make([][]string, len(disks)) var wg sync.WaitGroup for index, disk := range disks { if disk == nil { continue } wg.Add(1) go func(index int, disk StorageAPI) { defer wg.Done() diskEntries[index], _ = disk.ListDir(bucket, prefixDir, -1, xlMetaJSONFile) }(index, disk) } wg.Wait() // Find elements in entries which are not in mergedEntries for _, entries := range diskEntries { var newEntries []string for _, entry := range entries { idx := sort.SearchStrings(mergedEntries, entry) // if entry is already present in mergedEntries don't add. if idx < len(mergedEntries) && mergedEntries[idx] == entry { continue } newEntries = append(newEntries, entry) } if len(newEntries) > 0 { // Merge the entries and sort it. mergedEntries = append(mergedEntries, newEntries...) sort.Strings(mergedEntries) } } return mergedEntries } // listDir - lists all the entries at a given prefix and given entry in the prefix. listDir := func(bucket, prefixDir, prefixEntry string) (mergedEntries []string) { for _, set := range sets { var newEntries []string // Find elements in entries which are not in mergedEntries for _, entry := range listDirInternal(bucket, prefixDir, prefixEntry, set.getLoadBalancedDisks()) { idx := sort.SearchStrings(mergedEntries, entry) // if entry is already present in mergedEntries don't add. if idx < len(mergedEntries) && mergedEntries[idx] == entry { continue } newEntries = append(newEntries, entry) } if len(newEntries) > 0 { // Merge the entries and sort it. mergedEntries = append(mergedEntries, newEntries...) sort.Strings(mergedEntries) } } return filterMatchingPrefix(mergedEntries, prefixEntry) } return listDir } // FileInfoCh - file info channel type FileInfoCh struct { Ch chan FileInfo Prev FileInfo Valid bool } // Pop - pops a cached entry if any, or from the cached channel. func (f *FileInfoCh) Pop() (fi FileInfo, ok bool) { if f.Valid { f.Valid = false return f.Prev, true } // No cached entries found, read from channel f.Prev, ok = <-f.Ch return f.Prev, ok } // Push - cache an entry, for Pop() later. func (f *FileInfoCh) Push(fi FileInfo) { f.Prev = fi f.Valid = true } // Calculate least entry across multiple FileInfo channels, additionally // returns a boolean to indicate if the caller needs to call again. func leastEntry(entriesCh []FileInfoCh, readQuorum int) (FileInfo, bool) { var entriesValid = make([]bool, len(entriesCh)) var entries = make([]FileInfo, len(entriesCh)) for i := range entriesCh { entries[i], entriesValid[i] = entriesCh[i].Pop() } var isTruncated = false for _, valid := range entriesValid { if !valid { continue } isTruncated = true break } var lentry FileInfo var found bool for i, valid := range entriesValid { if !valid { continue } if !found { lentry = entries[i] found = true continue } if entries[i].Name < lentry.Name { lentry = entries[i] } } // We haven't been able to find any least entry, // this would mean that we don't have valid. if !found { return lentry, isTruncated } leastEntryCount := 0 for i, valid := range entriesValid { if !valid { continue } // Entries are duplicated across disks, // we should simply skip such entries. if lentry.Name == entries[i].Name && lentry.ModTime.Equal(entries[i].ModTime) { leastEntryCount++ continue } // Push all entries which are lexically higher // and will be returned later in Pop() entriesCh[i].Push(entries[i]) } if readQuorum < 0 { return lentry, isTruncated } quorum := lentry.Quorum if quorum == 0 { quorum = readQuorum } if leastEntryCount >= quorum { return lentry, isTruncated } return leastEntry(entriesCh, readQuorum) } // mergeEntriesCh - merges FileInfo channel to entries upto maxKeys. func mergeEntriesCh(entriesCh []FileInfoCh, maxKeys int, readQuorum int) (entries FilesInfo) { var i = 0 for { fi, valid := leastEntry(entriesCh, readQuorum) if !valid { break } if i == maxKeys { entries.IsTruncated = true // Re-insert the last entry so it can be // listed in the next listing iteration. for j := range entriesCh { if !entriesCh[j].Valid { entriesCh[j].Push(fi) } } break } entries.Files = append(entries.Files, fi) i++ } return entries } // Starts a walk channel across all disks and returns a slice. func (s *xlSets) startMergeWalks(ctx context.Context, bucket, prefix, marker string, recursive bool, endWalkCh chan struct{}) []FileInfoCh { var entryChs []FileInfoCh for _, set := range s.sets { for _, disk := range set.getDisks() { if disk == nil { // Disk can be offline continue } entryCh, err := disk.Walk(bucket, prefix, marker, recursive, xlMetaJSONFile, readMetadata, endWalkCh) if err != nil { // Disk walk returned error, ignore it. continue } entryChs = append(entryChs, FileInfoCh{ Ch: entryCh, }) } } return entryChs } func (s *xlSets) listObjectsNonSlash(ctx context.Context, bucket, prefix, marker, delimiter string, maxKeys int) (loi ListObjectsInfo, err error) { endWalkCh := make(chan struct{}) defer close(endWalkCh) recursive := true entryChs := s.startMergeWalks(context.Background(), bucket, prefix, "", recursive, endWalkCh) readQuorum := s.drivesPerSet / 2 var objInfos []ObjectInfo var eof bool var prevPrefix string for { if len(objInfos) == maxKeys { break } result, ok := leastEntry(entryChs, readQuorum) if !ok { eof = true break } var objInfo ObjectInfo index := strings.Index(strings.TrimPrefix(result.Name, prefix), delimiter) if index == -1 { objInfo = ObjectInfo{ IsDir: false, Bucket: bucket, Name: result.Name, ModTime: result.ModTime, Size: result.Size, ContentType: result.Metadata["content-type"], ContentEncoding: result.Metadata["content-encoding"], } // Extract etag from metadata. objInfo.ETag = extractETag(result.Metadata) // All the parts per object. objInfo.Parts = result.Parts // etag/md5Sum has already been extracted. We need to // remove to avoid it from appearing as part of // response headers. e.g, X-Minio-* or X-Amz-*. objInfo.UserDefined = cleanMetadata(result.Metadata) // Update storage class if sc, ok := result.Metadata[amzStorageClass]; ok { objInfo.StorageClass = sc } else { objInfo.StorageClass = globalMinioDefaultStorageClass } } else { index = len(prefix) + index + len(delimiter) currPrefix := result.Name[:index] if currPrefix == prevPrefix { continue } prevPrefix = currPrefix objInfo = ObjectInfo{ Bucket: bucket, Name: currPrefix, IsDir: true, } } if objInfo.Name <= marker { continue } objInfos = append(objInfos, objInfo) } result := ListObjectsInfo{} for _, objInfo := range objInfos { if objInfo.IsDir { result.Prefixes = append(result.Prefixes, objInfo.Name) continue } result.Objects = append(result.Objects, objInfo) } if !eof { result.IsTruncated = true if len(objInfos) > 0 { result.NextMarker = objInfos[len(objInfos)-1].Name } } return result, nil } // ListObjects - implements listing of objects across disks, each disk is indepenently // walked and merged at this layer. Resulting value through the merge process sends // the data in lexically sorted order. func (s *xlSets) listObjects(ctx context.Context, bucket, prefix, marker, delimiter string, maxKeys int, heal bool) (loi ListObjectsInfo, err error) { if delimiter != SlashSeparator && delimiter != "" { // "heal" option passed can be ignored as the heal-listing does not send non-standard delimiter. return s.listObjectsNonSlash(ctx, bucket, prefix, marker, delimiter, maxKeys) } if err = checkListObjsArgs(ctx, bucket, prefix, marker, delimiter, s); err != nil { return loi, err } // Marker is set validate pre-condition. if marker != "" { // Marker not common with prefix is not implemented. Send an empty response if !hasPrefix(marker, prefix) { return loi, nil } } // With max keys of zero we have reached eof, return right here. if maxKeys == 0 { return loi, nil } // For delimiter and prefix as '/' we do not list anything at all // since according to s3 spec we stop at the 'delimiter' // along // with the prefix. On a flat namespace with 'prefix' // as '/' we don't have any entries, since all the keys are // of form 'keyName/...' if delimiter == SlashSeparator && prefix == SlashSeparator { return loi, nil } // Over flowing count - reset to maxObjectList. if maxKeys < 0 || maxKeys > maxObjectList { maxKeys = maxObjectList } // Default is recursive, if delimiter is set then list non recursive. recursive := true if delimiter == SlashSeparator { recursive = false } entryChs, endWalkCh := s.pool.Release(listParams{bucket, recursive, marker, prefix, heal}) if entryChs == nil { endWalkCh = make(chan struct{}) entryChs = s.startMergeWalks(context.Background(), bucket, prefix, marker, recursive, endWalkCh) } readQuorum := s.drivesPerSet / 2 if heal { readQuorum = -1 } entries := mergeEntriesCh(entryChs, maxKeys, readQuorum) if len(entries.Files) == 0 { return loi, nil } loi.IsTruncated = entries.IsTruncated if loi.IsTruncated { loi.NextMarker = entries.Files[len(entries.Files)-1].Name } for _, entry := range entries.Files { var objInfo ObjectInfo if hasSuffix(entry.Name, SlashSeparator) { if !recursive { loi.Prefixes = append(loi.Prefixes, entry.Name) continue } objInfo = ObjectInfo{ Bucket: bucket, Name: entry.Name, IsDir: true, } } else { objInfo = ObjectInfo{ IsDir: false, Bucket: bucket, Name: entry.Name, ModTime: entry.ModTime, Size: entry.Size, ContentType: entry.Metadata["content-type"], ContentEncoding: entry.Metadata["content-encoding"], } // Extract etag from metadata. objInfo.ETag = extractETag(entry.Metadata) // All the parts per object. objInfo.Parts = entry.Parts // etag/md5Sum has already been extracted. We need to // remove to avoid it from appearing as part of // response headers. e.g, X-Minio-* or X-Amz-*. objInfo.UserDefined = cleanMetadata(entry.Metadata) // Update storage class if sc, ok := entry.Metadata[amzStorageClass]; ok { objInfo.StorageClass = sc } else { objInfo.StorageClass = globalMinioDefaultStorageClass } } loi.Objects = append(loi.Objects, objInfo) } if loi.IsTruncated { s.pool.Set(listParams{bucket, recursive, loi.NextMarker, prefix, heal}, entryChs, endWalkCh) } return loi, nil } // ListObjects - implements listing of objects across disks, each disk is indepenently // walked and merged at this layer. Resulting value through the merge process sends // the data in lexically sorted order. func (s *xlSets) ListObjects(ctx context.Context, bucket, prefix, marker, delimiter string, maxKeys int) (loi ListObjectsInfo, err error) { return s.listObjects(ctx, bucket, prefix, marker, delimiter, maxKeys, false) } func (s *xlSets) ListMultipartUploads(ctx context.Context, bucket, prefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (result ListMultipartsInfo, err error) { // In list multipart uploads we are going to treat input prefix as the object, // this means that we are not supporting directory navigation. return s.getHashedSet(prefix).ListMultipartUploads(ctx, bucket, prefix, keyMarker, uploadIDMarker, delimiter, maxUploads) } // Initiate a new multipart upload on a hashedSet based on object name. func (s *xlSets) NewMultipartUpload(ctx context.Context, bucket, object string, opts ObjectOptions) (uploadID string, err error) { return s.getHashedSet(object).NewMultipartUpload(ctx, bucket, object, opts) } // Copies a part of an object from source hashedSet to destination hashedSet. func (s *xlSets) CopyObjectPart(ctx context.Context, srcBucket, srcObject, destBucket, destObject string, uploadID string, partID int, startOffset int64, length int64, srcInfo ObjectInfo, srcOpts, dstOpts ObjectOptions) (partInfo PartInfo, err error) { destSet := s.getHashedSet(destObject) return destSet.PutObjectPart(ctx, destBucket, destObject, uploadID, partID, NewPutObjReader(srcInfo.Reader, nil, nil), dstOpts) } // PutObjectPart - writes part of an object to hashedSet based on the object name. func (s *xlSets) PutObjectPart(ctx context.Context, bucket, object, uploadID string, partID int, data *PutObjReader, opts ObjectOptions) (info PartInfo, err error) { return s.getHashedSet(object).PutObjectPart(ctx, bucket, object, uploadID, partID, data, opts) } // ListObjectParts - lists all uploaded parts to an object in hashedSet. func (s *xlSets) ListObjectParts(ctx context.Context, bucket, object, uploadID string, partNumberMarker int, maxParts int, opts ObjectOptions) (result ListPartsInfo, err error) { return s.getHashedSet(object).ListObjectParts(ctx, bucket, object, uploadID, partNumberMarker, maxParts, opts) } // Aborts an in-progress multipart operation on hashedSet based on the object name. func (s *xlSets) AbortMultipartUpload(ctx context.Context, bucket, object, uploadID string) error { return s.getHashedSet(object).AbortMultipartUpload(ctx, bucket, object, uploadID) } // CompleteMultipartUpload - completes a pending multipart transaction, on hashedSet based on object name. func (s *xlSets) CompleteMultipartUpload(ctx context.Context, bucket, object, uploadID string, uploadedParts []CompletePart, opts ObjectOptions) (objInfo ObjectInfo, err error) { return s.getHashedSet(object).CompleteMultipartUpload(ctx, bucket, object, uploadID, uploadedParts, opts) } /* All disks online ----------------- - All Unformatted - format all and return success. - Some Unformatted - format all and return success. - Any JBOD inconsistent - return failure - Some are corrupt (missing format.json) - return failure - Any unrecognized disks - return failure Some disks are offline and we have quorum. ----------------- - Some unformatted - format all and return success, treat disks offline as corrupted. - Any JBOD inconsistent - return failure - Some are corrupt (missing format.json) - Any unrecognized disks - return failure No read quorum ----------------- failure for all cases. // Pseudo code for managing `format.json`. // Generic checks. if (no quorum) return error if (any disk is corrupt) return error // Always error if (jbod inconsistent) return error // Always error. if (disks not recognized) // Always error. // Specific checks. if (all disks online) if (all disks return format.json) if (jbod consistent) if (all disks recognized) return else if (all disks return format.json not found) return error else (some disks return format.json not found) (heal format) return fi fi else if (some disks return format.json not found) // Offline disks are marked as dead. (heal format) // Offline disks should be marked as dead. return success fi fi */ func formatsToDrivesInfo(endpoints EndpointList, formats []*formatXLV3, sErrs []error) (beforeDrives []madmin.DriveInfo) { beforeDrives = make([]madmin.DriveInfo, len(endpoints)) // Existing formats are available (i.e. ok), so save it in // result, also populate disks to be healed. for i, format := range formats { drive := endpoints.GetString(i) var state = madmin.DriveStateCorrupt switch { case format != nil: state = madmin.DriveStateOk case sErrs[i] == errUnformattedDisk: state = madmin.DriveStateMissing case sErrs[i] == errDiskNotFound: state = madmin.DriveStateOffline } beforeDrives[i] = madmin.DriveInfo{ UUID: func() string { if format != nil { return format.XL.This } return "" }(), Endpoint: drive, State: state, } } return beforeDrives } // Reloads the format from the disk, usually called by a remote peer notifier while // healing in a distributed setup. func (s *xlSets) ReloadFormat(ctx context.Context, dryRun bool) (err error) { // Acquire lock on format.json formatLock := s.getHashedSet(formatConfigFile).nsMutex.NewNSLock(ctx, minioMetaBucket, formatConfigFile) if err = formatLock.GetRLock(globalHealingTimeout); err != nil { return err } defer formatLock.RUnlock() storageDisks, err := initStorageDisks(s.endpoints) if err != nil { return err } defer func(storageDisks []StorageAPI) { if err != nil { closeStorageDisks(storageDisks) } }(storageDisks) formats, sErrs := loadFormatXLAll(storageDisks) if err = checkFormatXLValues(formats); err != nil { return err } for index, sErr := range sErrs { if sErr != nil { // Look for acceptable heal errors, for any other // errors we should simply quit and return. if _, ok := formatHealErrors[sErr]; !ok { return fmt.Errorf("Disk %s: %s", s.endpoints[index], sErr) } } } refFormat, err := getFormatXLInQuorum(formats) if err != nil { return err } // kill the monitoring loop such that we stop writing // to indicate that we will re-initialize everything // with new format. s.disksConnectDoneCh <- struct{}{} // Replace the new format. s.format = refFormat s.xlDisksMu.Lock() { // Close all existing disks. s.xlDisks.Close() // Re initialize disks, after saving the new reference format. s.xlDisks = s.reInitDisks(refFormat, storageDisks, formats) } s.xlDisksMu.Unlock() // Restart monitoring loop to monitor reformatted disks again. go s.monitorAndConnectEndpoints(defaultMonitorConnectEndpointInterval) return nil } // If it is a single node XL and all disks are root disks, it is most likely a test setup, else it is a production setup. // On a test setup we allow creation of format.json on root disks to help with dev/testing. func isTestSetup(infos []DiskInfo, errs []error) bool { rootDiskCount := 0 for i := range errs { if errs[i] != nil { // On error it is safer to assume that this is not a test setup. return false } if infos[i].RootDisk { rootDiskCount++ } } // It is a test setup if all disks are root disks. return rootDiskCount == len(infos) } func getAllDiskInfos(storageDisks []StorageAPI) ([]DiskInfo, []error) { infos := make([]DiskInfo, len(storageDisks)) errs := make([]error, len(storageDisks)) var wg sync.WaitGroup for i := range storageDisks { if storageDisks[i] == nil { errs[i] = errDiskNotFound continue } wg.Add(1) go func(i int) { defer wg.Done() infos[i], errs[i] = storageDisks[i].DiskInfo() }(i) } wg.Wait() return infos, errs } // Mark root disks as down so as not to heal them. func markRootDisksAsDown(storageDisks []StorageAPI) { infos, errs := getAllDiskInfos(storageDisks) if isTestSetup(infos, errs) { // Allow healing of disks for test setups to help with testing. return } for i := range storageDisks { if errs[i] != nil { storageDisks[i] = nil continue } if infos[i].RootDisk { // We should not heal on root disk. i.e in a situation where the minio-administrator has unmounted a // defective drive we should not heal a path on the root disk. storageDisks[i] = nil } } } // HealFormat - heals missing `format.json` on fresh unformatted disks. // TODO: In future support corrupted disks missing format.json but has erasure // coded data in it. func (s *xlSets) HealFormat(ctx context.Context, dryRun bool) (res madmin.HealResultItem, err error) { // Acquire lock on format.json formatLock := s.getHashedSet(formatConfigFile).nsMutex.NewNSLock(ctx, minioMetaBucket, formatConfigFile) if err = formatLock.GetLock(globalHealingTimeout); err != nil { return madmin.HealResultItem{}, err } defer formatLock.Unlock() storageDisks, err := initStorageDisks(s.endpoints) if err != nil { return madmin.HealResultItem{}, err } defer func(storageDisks []StorageAPI) { if err != nil { closeStorageDisks(storageDisks) } }(storageDisks) markRootDisksAsDown(storageDisks) formats, sErrs := loadFormatXLAll(storageDisks) if err = checkFormatXLValues(formats); err != nil { return madmin.HealResultItem{}, err } // Prepare heal-result res = madmin.HealResultItem{ Type: madmin.HealItemMetadata, Detail: "disk-format", DiskCount: s.setCount * s.drivesPerSet, SetCount: s.setCount, } // Fetch all the drive info status. beforeDrives := formatsToDrivesInfo(s.endpoints, formats, sErrs) res.After.Drives = make([]madmin.HealDriveInfo, len(beforeDrives)) res.Before.Drives = make([]madmin.HealDriveInfo, len(beforeDrives)) // Copy "after" drive state too from before. for k, v := range beforeDrives { res.Before.Drives[k] = madmin.HealDriveInfo(v) res.After.Drives[k] = madmin.HealDriveInfo(v) } for index, sErr := range sErrs { if sErr != nil { // Look for acceptable heal errors, for any other // errors we should simply quit and return. if _, ok := formatHealErrors[sErr]; !ok { return res, fmt.Errorf("Disk %s: %s", s.endpoints[index], sErr) } } } if !hasAnyErrorsUnformatted(sErrs) { // No unformatted disks found disks are either offline // or online, no healing is required. return res, errNoHealRequired } // All disks are unformatted, return quorum error. if shouldInitXLDisks(sErrs) { return res, errXLReadQuorum } refFormat, err := getFormatXLInQuorum(formats) if err != nil { return res, err } // Mark all UUIDs which might be offline, use list // of formats to mark them appropriately. markUUIDsOffline(refFormat, formats) // Initialize a new set of set formats which will be written to disk. newFormatSets := newHealFormatSets(refFormat, s.setCount, s.drivesPerSet, formats, sErrs) // Look for all offline/unformatted disks in our reference format, // such that we can fill them up with new UUIDs, this looping also // ensures that the replaced disks allocated evenly across all sets. // Making sure that the redundancy is not lost. for i := range refFormat.XL.Sets { for j := range refFormat.XL.Sets[i] { if refFormat.XL.Sets[i][j] == offlineDiskUUID { for l := range newFormatSets[i] { if newFormatSets[i][l] == nil { continue } if newFormatSets[i][l].XL.This == "" { newFormatSets[i][l].XL.This = mustGetUUID() refFormat.XL.Sets[i][j] = newFormatSets[i][l].XL.This for m, v := range res.After.Drives { if v.Endpoint == s.endpoints.GetString(i*s.drivesPerSet+l) { res.After.Drives[m].UUID = newFormatSets[i][l].XL.This res.After.Drives[m].State = madmin.DriveStateOk } } break } } } } } if !dryRun { var tmpNewFormats = make([]*formatXLV3, s.setCount*s.drivesPerSet) for i := range newFormatSets { for j := range newFormatSets[i] { if newFormatSets[i][j] == nil { continue } tmpNewFormats[i*s.drivesPerSet+j] = newFormatSets[i][j] tmpNewFormats[i*s.drivesPerSet+j].XL.Sets = refFormat.XL.Sets } } // Initialize meta volume, if volume already exists ignores it, all disks which // are not found are ignored as well. if err = initFormatXLMetaVolume(storageDisks, tmpNewFormats); err != nil { return madmin.HealResultItem{}, fmt.Errorf("Unable to initialize '.minio.sys' meta volume, %s", err) } // Save formats `format.json` across all disks. if err = saveFormatXLAll(ctx, storageDisks, tmpNewFormats); err != nil { return madmin.HealResultItem{}, err } // kill the monitoring loop such that we stop writing // to indicate that we will re-initialize everything // with new format. s.disksConnectDoneCh <- struct{}{} // Replace with new reference format. s.format = refFormat s.xlDisksMu.Lock() { // Disconnect/relinquish all existing disks. s.xlDisks.Close() // Re initialize disks, after saving the new reference format. s.xlDisks = s.reInitDisks(refFormat, storageDisks, tmpNewFormats) } s.xlDisksMu.Unlock() // Restart our monitoring loop to start monitoring newly formatted disks. go s.monitorAndConnectEndpoints(defaultMonitorConnectEndpointInterval) } return res, nil } // HealBucket - heals inconsistent buckets and bucket metadata on all sets. func (s *xlSets) HealBucket(ctx context.Context, bucket string, dryRun, remove bool) (result madmin.HealResultItem, err error) { bucketLock := globalNSMutex.NewNSLock(ctx, bucket, "") if err := bucketLock.GetLock(globalHealingTimeout); err != nil { return result, err } defer bucketLock.Unlock() // Initialize heal result info result = madmin.HealResultItem{ Type: madmin.HealItemBucket, Bucket: bucket, DiskCount: s.setCount * s.drivesPerSet, SetCount: s.setCount, } for _, s := range s.sets { var healResult madmin.HealResultItem healResult, err = s.HealBucket(ctx, bucket, dryRun, remove) if err != nil { return result, err } result.Before.Drives = append(result.Before.Drives, healResult.Before.Drives...) result.After.Drives = append(result.After.Drives, healResult.After.Drives...) } for _, endpoint := range s.endpoints { var foundBefore bool for _, v := range result.Before.Drives { if endpoint.IsLocal { if v.Endpoint == endpoint.Path { foundBefore = true } } else { if v.Endpoint == endpoint.String() { foundBefore = true } } } if !foundBefore { result.Before.Drives = append(result.Before.Drives, madmin.HealDriveInfo{ UUID: "", Endpoint: endpoint.String(), State: madmin.DriveStateOffline, }) } var foundAfter bool for _, v := range result.After.Drives { if endpoint.IsLocal { if v.Endpoint == endpoint.Path { foundAfter = true } } else { if v.Endpoint == endpoint.String() { foundAfter = true } } } if !foundAfter { result.After.Drives = append(result.After.Drives, madmin.HealDriveInfo{ UUID: "", Endpoint: endpoint.String(), State: madmin.DriveStateOffline, }) } } // Check if we had quorum to write, if not return an appropriate error. _, afterDriveOnline := result.GetOnlineCounts() if afterDriveOnline < ((s.setCount*s.drivesPerSet)/2)+1 { return result, toObjectErr(errXLWriteQuorum, bucket) } return result, nil } // HealObject - heals inconsistent object on a hashedSet based on object name. func (s *xlSets) HealObject(ctx context.Context, bucket, object string, dryRun, remove bool, scanMode madmin.HealScanMode) (madmin.HealResultItem, error) { return s.getHashedSet(object).HealObject(ctx, bucket, object, dryRun, remove, scanMode) } // Lists all buckets which need healing. func (s *xlSets) ListBucketsHeal(ctx context.Context) ([]BucketInfo, error) { listBuckets := []BucketInfo{} var healBuckets = map[string]BucketInfo{} for _, set := range s.sets { buckets, _, err := listAllBuckets(set.getDisks()) if err != nil { return nil, err } for _, currBucket := range buckets { healBuckets[currBucket.Name] = BucketInfo(currBucket) } } for _, bucketInfo := range healBuckets { listBuckets = append(listBuckets, bucketInfo) } return listBuckets, nil } // HealObjects - Heal all objects recursively at a specified prefix, any // dangling objects deleted as well automatically. func (s *xlSets) HealObjects(ctx context.Context, bucket, prefix string, healObjectFn func(string, string) error) (err error) { recursive := true endWalkCh := make(chan struct{}) listDir := listDirSetsFactory(ctx, s.sets...) walkResultCh := startTreeWalk(ctx, bucket, prefix, "", recursive, listDir, endWalkCh) for { walkResult, ok := <-walkResultCh if !ok { break } if err := healObjectFn(bucket, walkResult.entry); err != nil { return toObjectErr(err, bucket, walkResult.entry) } if walkResult.end { break } } return nil } func (s *xlSets) ListObjectsHeal(ctx context.Context, bucket, prefix, marker, delimiter string, maxKeys int) (loi ListObjectsInfo, err error) { return s.listObjects(ctx, bucket, prefix, marker, delimiter, maxKeys, true) }