| 
									
										
										
										
											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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												fs: Break fs package to top-level and introduce ObjectAPI interface.
ObjectAPI interface brings in changes needed for XL ObjectAPI layer.
The new interface for any ObjectAPI layer is as below
```
// ObjectAPI interface.
type ObjectAPI interface {
        // Bucket resource API.
        DeleteBucket(bucket string) *probe.Error
        ListBuckets() ([]BucketInfo, *probe.Error)
        MakeBucket(bucket string) *probe.Error
        GetBucketInfo(bucket string) (BucketInfo, *probe.Error)
        // Bucket query API.
        ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsResult, *probe.Error)
        ListMultipartUploads(bucket string, resources BucketMultipartResourcesMetadata) (BucketMultipartResourcesMetadata, *probe.Error)
        // Object resource API.
        GetObject(bucket, object string, startOffset int64) (io.ReadCloser, *probe.Error)
        GetObjectInfo(bucket, object string) (ObjectInfo, *probe.Error)
        PutObject(bucket string, object string, size int64, data io.Reader, metadata map[string]string) (ObjectInfo, *probe.Error)
        DeleteObject(bucket, object string) *probe.Error
        // Object query API.
        NewMultipartUpload(bucket, object string) (string, *probe.Error)
        PutObjectPart(bucket, object, uploadID string, partID int, size int64, data io.Reader, md5Hex string) (string, *probe.Error)
        ListObjectParts(bucket, object string, resources ObjectResourcesMetadata) (ObjectResourcesMetadata, *probe.Error)
        CompleteMultipartUpload(bucket string, object string, uploadID string, parts []CompletePart) (ObjectInfo, *probe.Error)
        AbortMultipartUpload(bucket, object, uploadID string) *probe.Error
}
```
											
										 
											2016-03-31 07:15:28 +08:00
										 |  |  | package main | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 	"crypto/md5" | 
					
						
							|  |  |  | 	"encoding/hex" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"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
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/minio/minio/pkg/safe" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							| 
									
										
										
										
											2016-05-02 09:13:10 +08:00
										 |  |  | 	// Minio meta bucket.
 | 
					
						
							|  |  |  | 	minioMetaBucket = ".minio" | 
					
						
							| 
									
										
										
										
											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-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 { | 
					
						
							|  |  |  | 	if len(bucket) < 3 || len(bucket) > 63 { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if bucket[0] == '.' || bucket[len(bucket)-1] == '.' { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return validBucket.MatchString(bucket) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 ("\")
 | 
					
						
							|  |  |  | // - Caret ("^")
 | 
					
						
							|  |  |  | // - Grave accent / back tick ("`")
 | 
					
						
							|  |  |  | // - Vertical bar / pipe ("|")
 | 
					
						
							| 
									
										
										
										
											2016-02-20 08:04:29 +08:00
										 |  |  | func IsValidObjectName(object string) bool { | 
					
						
							|  |  |  | 	if len(object) > 1024 || len(object) == 0 { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !utf8.ValidString(object) { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-04-14 02:32:47 +08:00
										 |  |  | 	// Reject unsupported characters in object name.
 | 
					
						
							| 
									
										
										
										
											2016-04-30 11:19:08 +08:00
										 |  |  | 	return !strings.ContainsAny(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-04-17 03:48:41 +08:00
										 |  |  | 	// Prefix can be empty or "/".
 | 
					
						
							|  |  |  | 	if object == "" || object == "/" { | 
					
						
							| 
									
										
										
										
											2016-03-23 07:03:08 +08:00
										 |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Verify if prefix is a valid object name.
 | 
					
						
							|  |  |  | 	return IsValidObjectName(object) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											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-04-22 09:05:26 +08:00
										 |  |  | func pathJoin(s1 string, s2 string) string { | 
					
						
							| 
									
										
										
										
											2016-05-09 01:15:34 +08:00
										 |  |  | 	trailingSlash := "" | 
					
						
							|  |  |  | 	if strings.HasSuffix(s2, slashSeparator) { | 
					
						
							|  |  |  | 		trailingSlash = "/" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return path.Join(s1, s2) + trailingSlash | 
					
						
							| 
									
										
										
										
											2016-04-17 02:43:03 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Create an s3 compatible MD5sum for complete multipart transaction.
 | 
					
						
							| 
									
										
										
										
											2016-05-08 17:38:35 +08:00
										 |  |  | func completeMultipartMD5(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 { | 
					
						
							|  |  |  | 			return "", err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		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 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // safeCloseAndRemove - safely closes and removes underlying temporary
 | 
					
						
							|  |  |  | // file writer if possible.
 | 
					
						
							|  |  |  | func safeCloseAndRemove(writer io.WriteCloser) error { | 
					
						
							|  |  |  | 	// If writer is a safe file, Attempt to close and remove.
 | 
					
						
							|  |  |  | 	safeWriter, ok := writer.(*safe.File) | 
					
						
							|  |  |  | 	if ok { | 
					
						
							|  |  |  | 		return safeWriter.CloseAndRemove() | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-05-09 07:26:10 +08:00
										 |  |  | 	wCloser, ok := writer.(*waitCloser) | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 	if ok { | 
					
						
							| 
									
										
										
										
											2016-05-09 07:26:10 +08:00
										 |  |  | 		return wCloser.CloseWithError(errors.New("Close and error out.")) | 
					
						
							| 
									
										
										
										
											2016-04-30 05:24:10 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } |