| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | /* | 
					
						
							|  |  |  |  * MinIO Cloud Storage, (C) 2020 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" | 
					
						
							| 
									
										
										
										
											2020-10-31 00:33:16 +08:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2021-02-12 02:22:03 +08:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2021-02-12 02:22:03 +08:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 	"path" | 
					
						
							| 
									
										
										
										
											2021-02-27 01:52:27 +08:00
										 |  |  | 	pathutil "path" | 
					
						
							| 
									
										
										
										
											2021-01-14 01:44:11 +08:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2020-11-19 02:28:22 +08:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/minio/minio/cmd/logger" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-12 02:22:03 +08:00
										 |  |  | func renameAllBucketMetacache(epPath string) error { | 
					
						
							|  |  |  | 	// Rename all previous `.minio.sys/buckets/<bucketname>/.metacache` to
 | 
					
						
							|  |  |  | 	// to `.minio.sys/tmp/` for deletion.
 | 
					
						
							| 
									
										
										
										
											2021-02-18 07:34:42 +08:00
										 |  |  | 	return readDirFn(pathJoin(epPath, minioMetaBucket, bucketMetaPrefix), func(name string, typ os.FileMode) error { | 
					
						
							| 
									
										
										
										
											2021-02-12 02:22:03 +08:00
										 |  |  | 		if typ == os.ModeDir { | 
					
						
							| 
									
										
										
										
											2021-02-27 01:52:27 +08:00
										 |  |  | 			tmpMetacacheOld := pathutil.Join(epPath, minioMetaTmpDeletedBucket, mustGetUUID()) | 
					
						
							| 
									
										
										
										
											2021-02-12 02:22:03 +08:00
										 |  |  | 			if err := renameAll(pathJoin(epPath, minioMetaBucket, metacachePrefixForID(name, slashSeparator)), | 
					
						
							|  |  |  | 				tmpMetacacheOld); err != nil && err != errFileNotFound { | 
					
						
							|  |  |  | 				return fmt.Errorf("unable to rename (%s -> %s) %w", | 
					
						
							|  |  |  | 					pathJoin(epPath, minioMetaBucket+metacachePrefixForID(minioMetaBucket, slashSeparator)), | 
					
						
							|  |  |  | 					tmpMetacacheOld, | 
					
						
							|  |  |  | 					osErrToFileErr(err)) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | // listPath will return the requested entries.
 | 
					
						
							|  |  |  | // If no more entries are in the listing io.EOF is returned,
 | 
					
						
							|  |  |  | // otherwise nil or an unexpected error is returned.
 | 
					
						
							|  |  |  | // The listPathOptions given will be checked and modified internally.
 | 
					
						
							|  |  |  | // Required important fields are Bucket, Prefix, Separator.
 | 
					
						
							|  |  |  | // Other important fields are Limit, Marker.
 | 
					
						
							|  |  |  | // List ID always derived from the Marker.
 | 
					
						
							| 
									
										
										
										
											2020-12-02 05:50:33 +08:00
										 |  |  | func (z *erasureServerPools) listPath(ctx context.Context, o listPathOptions) (entries metaCacheEntriesSorted, err error) { | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 	if err := checkListObjsArgs(ctx, o.Bucket, o.Prefix, o.Marker, z); err != nil { | 
					
						
							|  |  |  | 		return entries, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Marker is set validate pre-condition.
 | 
					
						
							|  |  |  | 	if o.Marker != "" && o.Prefix != "" { | 
					
						
							|  |  |  | 		// Marker not common with prefix is not implemented. Send an empty response
 | 
					
						
							|  |  |  | 		if !HasPrefix(o.Marker, o.Prefix) { | 
					
						
							|  |  |  | 			return entries, io.EOF | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// With max keys of zero we have reached eof, return right here.
 | 
					
						
							|  |  |  | 	if o.Limit == 0 { | 
					
						
							|  |  |  | 		return entries, io.EOF | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// For delimiter and prefix as '/' we do not list anything at all
 | 
					
						
							|  |  |  | 	// 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/...'
 | 
					
						
							| 
									
										
										
										
											2021-01-14 01:44:11 +08:00
										 |  |  | 	if strings.HasPrefix(o.Prefix, SlashSeparator) { | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 		return entries, io.EOF | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Over flowing count - reset to maxObjectList.
 | 
					
						
							|  |  |  | 	if o.Limit < 0 || o.Limit > maxObjectList { | 
					
						
							|  |  |  | 		o.Limit = maxObjectList | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// If delimiter is slashSeparator we must return directories of
 | 
					
						
							|  |  |  | 	// the non-recursive scan unless explicitly requested.
 | 
					
						
							|  |  |  | 	o.IncludeDirectories = o.Separator == slashSeparator | 
					
						
							|  |  |  | 	if (o.Separator == slashSeparator || o.Separator == "") && !o.Recursive { | 
					
						
							|  |  |  | 		o.Recursive = o.Separator != slashSeparator | 
					
						
							|  |  |  | 		o.Separator = slashSeparator | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		// Default is recursive, if delimiter is set then list non recursive.
 | 
					
						
							|  |  |  | 		o.Recursive = true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Decode and get the optional list id from the marker.
 | 
					
						
							|  |  |  | 	o.Marker, o.ID = parseMarker(o.Marker) | 
					
						
							|  |  |  | 	o.Create = o.ID == "" | 
					
						
							|  |  |  | 	if o.ID == "" { | 
					
						
							|  |  |  | 		o.ID = mustGetUUID() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	o.BaseDir = baseDirFromPrefix(o.Prefix) | 
					
						
							| 
									
										
										
										
											2020-12-16 03:25:36 +08:00
										 |  |  | 	if o.discardResult { | 
					
						
							| 
									
										
										
										
											2020-11-14 08:58:20 +08:00
										 |  |  | 		// Override for single object.
 | 
					
						
							|  |  |  | 		o.BaseDir = o.Prefix | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-16 03:25:36 +08:00
										 |  |  | 	// For very small recursive listings, don't same cache.
 | 
					
						
							|  |  |  | 	// Attempts to avoid expensive listings to run for a long
 | 
					
						
							|  |  |  | 	// while when clients aren't interested in results.
 | 
					
						
							|  |  |  | 	// If the client DOES resume the listing a full cache
 | 
					
						
							|  |  |  | 	// will be generated due to the marker without ID and this check failing.
 | 
					
						
							|  |  |  | 	if o.Limit < 10 && o.Marker == "" && o.Create && o.Recursive { | 
					
						
							|  |  |  | 		o.discardResult = true | 
					
						
							| 
									
										
										
										
											2021-02-25 14:24:38 +08:00
										 |  |  | 		o.Transient = true | 
					
						
							| 
									
										
										
										
											2020-12-16 03:25:36 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 	var cache metacache | 
					
						
							|  |  |  | 	// If we don't have a list id we must ask the server if it has a cache or create a new.
 | 
					
						
							|  |  |  | 	if o.Create { | 
					
						
							|  |  |  | 		o.CurrentCycle = intDataUpdateTracker.current() | 
					
						
							|  |  |  | 		o.OldestCycle = globalNotificationSys.findEarliestCleanBloomFilter(ctx, path.Join(o.Bucket, o.BaseDir)) | 
					
						
							|  |  |  | 		var cache metacache | 
					
						
							|  |  |  | 		rpc := globalNotificationSys.restClientFromHash(o.Bucket) | 
					
						
							| 
									
										
										
										
											2020-11-04 04:47:52 +08:00
										 |  |  | 		if isReservedOrInvalidBucket(o.Bucket, false) { | 
					
						
							|  |  |  | 			rpc = nil | 
					
						
							| 
									
										
										
										
											2021-02-25 14:24:38 +08:00
										 |  |  | 			o.Transient = true | 
					
						
							| 
									
										
										
										
											2020-11-04 04:47:52 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-11-19 02:44:18 +08:00
										 |  |  | 		// Apply prefix filter if enabled.
 | 
					
						
							|  |  |  | 		o.SetFilter() | 
					
						
							| 
									
										
										
										
											2021-02-25 14:24:38 +08:00
										 |  |  | 		if rpc == nil || o.Transient { | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 			// Local
 | 
					
						
							| 
									
										
										
										
											2020-11-04 04:47:52 +08:00
										 |  |  | 			cache = localMetacacheMgr.findCache(ctx, o) | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2020-11-19 02:28:22 +08:00
										 |  |  | 			ctx, cancel := context.WithTimeout(ctx, 5*time.Second) | 
					
						
							|  |  |  | 			defer cancel() | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 			c, err := rpc.GetMetacacheListing(ctx, o) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2020-11-19 02:28:22 +08:00
										 |  |  | 				if errors.Is(err, context.Canceled) { | 
					
						
							| 
									
										
										
										
											2020-10-31 00:33:16 +08:00
										 |  |  | 					// Context is canceled, return at once.
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:53:48 +08:00
										 |  |  | 					// request canceled, no entries to return
 | 
					
						
							|  |  |  | 					return entries, io.EOF | 
					
						
							| 
									
										
										
										
											2020-10-31 00:33:16 +08:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2020-11-19 02:28:22 +08:00
										 |  |  | 				if !errors.Is(err, context.DeadlineExceeded) { | 
					
						
							|  |  |  | 					logger.LogIf(ctx, err) | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-02-25 14:24:38 +08:00
										 |  |  | 				o.Transient = true | 
					
						
							| 
									
										
										
										
											2020-11-04 04:47:52 +08:00
										 |  |  | 				cache = localMetacacheMgr.findCache(ctx, o) | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				cache = *c | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if cache.fileNotFound { | 
					
						
							| 
									
										
										
										
											2020-11-04 00:53:48 +08:00
										 |  |  | 			// No cache found, no entries found.
 | 
					
						
							|  |  |  | 			return entries, io.EOF | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		// Only create if we created a new.
 | 
					
						
							|  |  |  | 		o.Create = o.ID == cache.id | 
					
						
							|  |  |  | 		o.ID = cache.id | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var mu sync.Mutex | 
					
						
							|  |  |  | 	var wg sync.WaitGroup | 
					
						
							|  |  |  | 	var errs []error | 
					
						
							|  |  |  | 	allAtEOF := true | 
					
						
							|  |  |  | 	mu.Lock() | 
					
						
							|  |  |  | 	// Ask all sets and merge entries.
 | 
					
						
							| 
									
										
										
										
											2021-01-07 01:35:47 +08:00
										 |  |  | 	for _, pool := range z.serverPools { | 
					
						
							|  |  |  | 		for _, set := range pool.sets { | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 			wg.Add(1) | 
					
						
							|  |  |  | 			go func(i int, set *erasureObjects) { | 
					
						
							|  |  |  | 				defer wg.Done() | 
					
						
							|  |  |  | 				e, err := set.listPath(ctx, o) | 
					
						
							|  |  |  | 				mu.Lock() | 
					
						
							|  |  |  | 				defer mu.Unlock() | 
					
						
							|  |  |  | 				if err == nil { | 
					
						
							|  |  |  | 					allAtEOF = false | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				errs[i] = err | 
					
						
							|  |  |  | 				entries.merge(e, -1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// Resolve non-trivial conflicts
 | 
					
						
							|  |  |  | 				entries.deduplicate(func(existing, other *metaCacheEntry) (replace bool) { | 
					
						
							|  |  |  | 					if existing.isDir() { | 
					
						
							|  |  |  | 						return false | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					eFIV, err := existing.fileInfo(o.Bucket) | 
					
						
							|  |  |  | 					if err != nil { | 
					
						
							|  |  |  | 						return true | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					oFIV, err := existing.fileInfo(o.Bucket) | 
					
						
							|  |  |  | 					if err != nil { | 
					
						
							|  |  |  | 						return false | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					return oFIV.ModTime.After(eFIV.ModTime) | 
					
						
							|  |  |  | 				}) | 
					
						
							|  |  |  | 				if entries.len() > o.Limit { | 
					
						
							|  |  |  | 					allAtEOF = false | 
					
						
							|  |  |  | 					entries.truncate(o.Limit) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}(len(errs), set) | 
					
						
							|  |  |  | 			errs = append(errs, nil) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	mu.Unlock() | 
					
						
							|  |  |  | 	wg.Wait() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if isAllNotFound(errs) { | 
					
						
							|  |  |  | 		// All sets returned not found.
 | 
					
						
							| 
									
										
										
										
											2020-11-19 02:28:22 +08:00
										 |  |  | 		go func() { | 
					
						
							|  |  |  | 			// Update master cache with that information.
 | 
					
						
							|  |  |  | 			cache.status = scanStateSuccess | 
					
						
							|  |  |  | 			cache.fileNotFound = true | 
					
						
							|  |  |  | 			o.updateMetacacheListing(cache, globalNotificationSys.restClientFromHash(o.Bucket)) | 
					
						
							|  |  |  | 		}() | 
					
						
							| 
									
										
										
										
											2020-11-04 00:53:48 +08:00
										 |  |  | 		// cache returned not found, entries truncated.
 | 
					
						
							|  |  |  | 		return entries, io.EOF | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, err := range errs { | 
					
						
							|  |  |  | 		if err == nil { | 
					
						
							|  |  |  | 			allAtEOF = false | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-10-29 04:19:53 +08:00
										 |  |  | 		if err.Error() == io.EOF.Error() { | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		logger.LogIf(ctx, err) | 
					
						
							|  |  |  | 		return entries, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	truncated := entries.len() > o.Limit || !allAtEOF | 
					
						
							|  |  |  | 	entries.truncate(o.Limit) | 
					
						
							| 
									
										
										
										
											2020-12-16 03:25:36 +08:00
										 |  |  | 	if !o.discardResult { | 
					
						
							|  |  |  | 		entries.listID = o.ID | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-10-29 00:18:35 +08:00
										 |  |  | 	if !truncated { | 
					
						
							|  |  |  | 		return entries, io.EOF | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return entries, nil | 
					
						
							|  |  |  | } |