| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | /* | 
					
						
							|  |  |  |  * Minio Cloud Storage, (C) 2015, 2016 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-19 07:23:42 +08:00
										 |  |  | package cmd | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 	"crypto/md5" | 
					
						
							|  |  |  | 	"encoding/hex" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2016-10-23 00:05:01 +08:00
										 |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2016-05-09 01:15:34 +08:00
										 |  |  | 	"path" | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | 	"regexp" | 
					
						
							| 
									
										
										
										
											2016-04-14 02:32:47 +08:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | 	"unicode/utf8" | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-21 11:48:47 +08:00
										 |  |  | 	"github.com/skyrings/skyring-common/tools/uuid" | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							| 
									
										
										
										
											2016-05-02 09:13:10 +08:00
										 |  |  | 	// Minio meta bucket.
 | 
					
						
							| 
									
										
										
										
											2016-07-13 06:21:29 +08:00
										 |  |  | 	minioMetaBucket = ".minio.sys" | 
					
						
							| 
									
										
										
										
											2016-05-04 07:10:24 +08:00
										 |  |  | 	// Multipart meta prefix.
 | 
					
						
							|  |  |  | 	mpartMetaPrefix = "multipart" | 
					
						
							|  |  |  | 	// Tmp meta prefix.
 | 
					
						
							|  |  |  | 	tmpMetaPrefix = "tmp" | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // validBucket regexp.
 | 
					
						
							|  |  |  | var validBucket = regexp.MustCompile(`^[a-z0-9][a-z0-9\.\-]{1,61}[a-z0-9]$`) | 
					
						
							| 
									
										
										
										
											2016-07-16 08:30:37 +08:00
										 |  |  | var isIPAddress = regexp.MustCompile(`^(\d+\.){3}\d+$`) | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-08 11:30:30 +08:00
										 |  |  | // IsValidBucketName verifies a bucket name in accordance with Amazon's
 | 
					
						
							|  |  |  | // requirements. It must be 3-63 characters long, can contain dashes
 | 
					
						
							|  |  |  | // and periods, but must begin and end with a lowercase letter or a number.
 | 
					
						
							|  |  |  | // See: http://docs.aws.amazon.com/AmazonS3/latest/dev/BucketRestrictions.html
 | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | func IsValidBucketName(bucket string) bool { | 
					
						
							| 
									
										
										
										
											2016-07-24 13:51:12 +08:00
										 |  |  | 	// Special case when bucket is equal to 'metaBucket'.
 | 
					
						
							|  |  |  | 	if bucket == minioMetaBucket { | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | 	if len(bucket) < 3 || len(bucket) > 63 { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if bucket[0] == '.' || bucket[len(bucket)-1] == '.' { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-07-16 08:30:37 +08:00
										 |  |  | 	return (validBucket.MatchString(bucket) && | 
					
						
							|  |  |  | 		!isIPAddress.MatchString(bucket) && | 
					
						
							|  |  |  | 		!strings.Contains(bucket, "..")) | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-08 11:30:30 +08:00
										 |  |  | // IsValidObjectName verifies an object name in accordance with Amazon's
 | 
					
						
							|  |  |  | // requirements. It cannot exceed 1024 characters and must be a valid UTF8
 | 
					
						
							|  |  |  | // string.
 | 
					
						
							| 
									
										
										
										
											2016-04-14 02:32:47 +08:00
										 |  |  | //
 | 
					
						
							|  |  |  | // See:
 | 
					
						
							|  |  |  | // http://docs.aws.amazon.com/AmazonS3/latest/dev/UsingMetadata.html
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // You should avoid the following characters in a key name because of
 | 
					
						
							|  |  |  | // significant special handling for consistency across all
 | 
					
						
							|  |  |  | // applications.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Rejects strings with following characters.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // - Backslash ("\")
 | 
					
						
							| 
									
										
										
										
											2016-05-14 02:43:06 +08:00
										 |  |  | //
 | 
					
						
							| 
									
										
										
										
											2016-10-01 07:56:36 +08:00
										 |  |  | // additionally minio does not support object names with trailing "/".
 | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | func IsValidObjectName(object string) bool { | 
					
						
							| 
									
										
										
										
											2016-05-14 02:43:06 +08:00
										 |  |  | 	if len(object) == 0 { | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-05-14 02:43:06 +08:00
										 |  |  | 	if strings.HasSuffix(object, slashSeparator) { | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-05-14 02:43:06 +08:00
										 |  |  | 	if strings.HasPrefix(object, slashSeparator) { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return IsValidObjectPrefix(object) | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-03-23 07:03:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // IsValidObjectPrefix verifies whether the prefix is a valid object name.
 | 
					
						
							|  |  |  | // Its valid to have a empty prefix.
 | 
					
						
							|  |  |  | func IsValidObjectPrefix(object string) bool { | 
					
						
							| 
									
										
										
										
											2016-05-14 02:43:06 +08:00
										 |  |  | 	if len(object) > 1024 { | 
					
						
							|  |  |  | 		return false | 
					
						
							| 
									
										
										
										
											2016-03-23 07:03:08 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-05-14 02:43:06 +08:00
										 |  |  | 	if !utf8.ValidString(object) { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Reject unsupported characters in object name.
 | 
					
						
							| 
									
										
										
										
											2016-10-01 07:56:36 +08:00
										 |  |  | 	if strings.ContainsAny(object, "\\") { | 
					
						
							| 
									
										
										
										
											2016-05-14 02:43:06 +08:00
										 |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true | 
					
						
							| 
									
										
										
										
											2016-03-23 07:03:08 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-17 02:43:03 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-22 09:05:26 +08:00
										 |  |  | // Slash separator.
 | 
					
						
							|  |  |  | const slashSeparator = "/" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // retainSlash - retains slash from a path.
 | 
					
						
							|  |  |  | func retainSlash(s string) string { | 
					
						
							|  |  |  | 	return strings.TrimSuffix(s, slashSeparator) + slashSeparator | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-09 01:15:34 +08:00
										 |  |  | // pathJoin - like path.Join() but retains trailing "/" of the last element
 | 
					
						
							| 
									
										
										
										
											2016-05-09 15:46:54 +08:00
										 |  |  | func pathJoin(elem ...string) string { | 
					
						
							| 
									
										
										
										
											2016-05-09 01:15:34 +08:00
										 |  |  | 	trailingSlash := "" | 
					
						
							| 
									
										
										
										
											2016-05-09 15:46:54 +08:00
										 |  |  | 	if len(elem) > 0 { | 
					
						
							|  |  |  | 		if strings.HasSuffix(elem[len(elem)-1], slashSeparator) { | 
					
						
							|  |  |  | 			trailingSlash = "/" | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-05-09 01:15:34 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-05-09 15:46:54 +08:00
										 |  |  | 	return path.Join(elem...) + trailingSlash | 
					
						
							| 
									
										
										
										
											2016-04-17 02:43:03 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-21 11:48:47 +08:00
										 |  |  | // getUUID() - get a unique uuid.
 | 
					
						
							|  |  |  | func getUUID() (uuidStr string) { | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		uuid, err := uuid.New() | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			errorIf(err, "Unable to initialize uuid") | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		uuidStr = uuid.String() | 
					
						
							|  |  |  | 		break | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return uuidStr | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | // Create an s3 compatible MD5sum for complete multipart transaction.
 | 
					
						
							| 
									
										
										
										
											2016-10-25 04:56:13 +08:00
										 |  |  | func getCompleteMultipartMD5(parts ...completePart) (string, error) { | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 	var finalMD5Bytes []byte | 
					
						
							| 
									
										
										
										
											2016-05-08 17:38:35 +08:00
										 |  |  | 	for _, part := range parts { | 
					
						
							|  |  |  | 		md5Bytes, err := hex.DecodeString(part.ETag) | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2016-08-26 00:39:01 +08:00
										 |  |  | 			return "", traceError(err) | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		finalMD5Bytes = append(finalMD5Bytes, md5Bytes...) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	md5Hasher := md5.New() | 
					
						
							|  |  |  | 	md5Hasher.Write(finalMD5Bytes) | 
					
						
							| 
									
										
										
										
											2016-05-08 17:38:35 +08:00
										 |  |  | 	s3MD5 := fmt.Sprintf("%s-%d", hex.EncodeToString(md5Hasher.Sum(nil)), len(parts)) | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 	return s3MD5, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // byBucketName is a collection satisfying sort.Interface.
 | 
					
						
							|  |  |  | type byBucketName []BucketInfo | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d byBucketName) Len() int           { return len(d) } | 
					
						
							|  |  |  | func (d byBucketName) Swap(i, j int)      { d[i], d[j] = d[j], d[i] } | 
					
						
							|  |  |  | func (d byBucketName) Less(i, j int) bool { return d[i].Name < d[j].Name } | 
					
						
							| 
									
										
										
										
											2016-10-23 00:05:01 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Copied from io.LimitReader()
 | 
					
						
							|  |  |  | // limitReader returns a Reader that reads from r
 | 
					
						
							|  |  |  | // but returns error after n bytes.
 | 
					
						
							|  |  |  | // The underlying implementation is a *LimitedReader.
 | 
					
						
							|  |  |  | type limitedReader struct { | 
					
						
							|  |  |  | 	R io.Reader // underlying reader
 | 
					
						
							| 
									
										
										
										
											2016-10-25 14:47:03 +08:00
										 |  |  | 	M int64     // min bytes remaining
 | 
					
						
							| 
									
										
										
										
											2016-10-23 00:05:01 +08:00
										 |  |  | 	N int64     // max bytes remaining
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-25 14:47:03 +08:00
										 |  |  | func limitReader(r io.Reader, m, n int64) io.Reader { return &limitedReader{r, m, n} } | 
					
						
							| 
									
										
										
										
											2016-10-23 00:05:01 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func (l *limitedReader) Read(p []byte) (n int, err error) { | 
					
						
							|  |  |  | 	n, err = l.R.Read(p) | 
					
						
							|  |  |  | 	l.N -= int64(n) | 
					
						
							| 
									
										
										
										
											2016-10-25 14:47:03 +08:00
										 |  |  | 	l.M -= int64(n) | 
					
						
							| 
									
										
										
										
											2016-10-23 00:05:01 +08:00
										 |  |  | 	if l.N < 0 { | 
					
						
							|  |  |  | 		// If more data is available than what is expected we return error.
 | 
					
						
							|  |  |  | 		return 0, errDataTooLarge | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-10-25 14:47:03 +08:00
										 |  |  | 	if err == io.EOF && l.M > 0 { | 
					
						
							|  |  |  | 		return 0, errDataTooSmall | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-10-23 00:05:01 +08:00
										 |  |  | 	return | 
					
						
							|  |  |  | } |