2015-02-23 19:46:48 -05:00
|
|
|
/*
|
2015-07-24 20:51:40 -04:00
|
|
|
* Minio Cloud Storage, (C) 2015 Minio, Inc.
|
2015-02-23 19:46:48 -05:00
|
|
|
*
|
|
|
|
* 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,
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
2015-02-23 19:46:48 -05:00
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2016-08-18 19:23:42 -04:00
|
|
|
package cmd
|
2015-02-15 20:03:27 -05:00
|
|
|
|
|
|
|
import (
|
2016-03-12 19:08:15 -05:00
|
|
|
"encoding/hex"
|
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-30 19:15:28 -04:00
|
|
|
"encoding/xml"
|
2016-02-27 06:04:52 -05:00
|
|
|
"io"
|
2016-03-12 19:08:15 -05:00
|
|
|
"io/ioutil"
|
2015-02-15 20:03:27 -05:00
|
|
|
"net/http"
|
2016-02-07 06:37:54 -05:00
|
|
|
"net/url"
|
2016-07-26 22:10:02 -04:00
|
|
|
"path"
|
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-30 19:15:28 -04:00
|
|
|
"sort"
|
2015-05-04 02:16:10 -04:00
|
|
|
"strconv"
|
2016-02-27 06:04:52 -05:00
|
|
|
"strings"
|
2015-02-15 20:03:27 -05:00
|
|
|
|
2016-02-27 06:04:52 -05:00
|
|
|
mux "github.com/gorilla/mux"
|
2015-05-09 22:39:00 -04:00
|
|
|
)
|
|
|
|
|
2016-04-04 22:55:07 -04:00
|
|
|
// supportedGetReqParams - supported request parameters for GET presigned request.
|
2016-02-07 06:37:54 -05:00
|
|
|
var supportedGetReqParams = map[string]string{
|
|
|
|
"response-expires": "Expires",
|
|
|
|
"response-content-type": "Content-Type",
|
|
|
|
"response-cache-control": "Cache-Control",
|
2016-08-10 20:36:28 -04:00
|
|
|
"response-content-encoding": "Content-Encoding",
|
|
|
|
"response-content-language": "Content-Language",
|
2016-02-07 06:37:54 -05:00
|
|
|
"response-content-disposition": "Content-Disposition",
|
|
|
|
}
|
|
|
|
|
2016-02-28 21:10:37 -05:00
|
|
|
// setGetRespHeaders - set any requested parameters as response headers.
|
|
|
|
func setGetRespHeaders(w http.ResponseWriter, reqParams url.Values) {
|
2016-02-07 06:37:54 -05:00
|
|
|
for k, v := range reqParams {
|
|
|
|
if header, ok := supportedGetReqParams[k]; ok {
|
|
|
|
w.Header()[header] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-06 21:31:40 -04:00
|
|
|
// errAllowableNotFound - For an anon user, return 404 if have ListBucket, 403 otherwise
|
|
|
|
// this is in keeping with the permissions sections of the docs of both:
|
|
|
|
// HEAD Object: http://docs.aws.amazon.com/AmazonS3/latest/API/RESTObjectHEAD.html
|
|
|
|
// GET Object: http://docs.aws.amazon.com/AmazonS3/latest/API/RESTObjectGET.html
|
2016-08-10 23:10:48 -04:00
|
|
|
func errAllowableObjectNotFound(bucket string, r *http.Request) APIErrorCode {
|
2016-04-06 21:31:40 -04:00
|
|
|
if getRequestAuthType(r) == authTypeAnonymous {
|
|
|
|
//we care about the bucket as a whole, not a particular resource
|
|
|
|
url := *r.URL
|
|
|
|
url.Path = "/" + bucket
|
|
|
|
|
2016-08-10 23:10:48 -04:00
|
|
|
if s3Error := enforceBucketPolicy(bucket, "s3:ListBucket", &url); s3Error != ErrNone {
|
2016-04-06 21:31:40 -04:00
|
|
|
return ErrAccessDenied
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ErrNoSuchKey
|
|
|
|
}
|
|
|
|
|
2016-07-10 20:12:22 -04:00
|
|
|
// Simple way to convert a func to io.Writer type.
|
|
|
|
type funcToWriter func([]byte) (int, error)
|
|
|
|
|
|
|
|
func (f funcToWriter) Write(p []byte) (int, error) {
|
|
|
|
return f(p)
|
|
|
|
}
|
|
|
|
|
2015-06-30 23:15:48 -04:00
|
|
|
// GetObjectHandler - GET Object
|
2015-02-23 19:46:48 -05:00
|
|
|
// ----------
|
|
|
|
// This implementation of the GET operation retrieves object. To use GET,
|
|
|
|
// you must have READ access to the object.
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) GetObjectHandler(w http.ResponseWriter, r *http.Request) {
|
2015-04-22 19:28:13 -04:00
|
|
|
var object, bucket string
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2015-02-15 20:03:27 -05:00
|
|
|
bucket = vars["bucket"]
|
|
|
|
object = vars["object"]
|
2015-04-22 22:29:39 -04:00
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
// Fetch object stat info.
|
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 16:51:05 -05:00
|
|
|
if s3Error := checkRequestAuthType(r, bucket, "s3:GetObject", serverConfig.GetRegion()); s3Error != ErrNone {
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
return
|
2016-02-15 20:42:39 -05:00
|
|
|
}
|
2016-11-21 16:51:05 -05:00
|
|
|
|
2016-12-10 19:15:12 -05:00
|
|
|
// Lock the object before reading.
|
|
|
|
objectLock := globalNSMutex.NewNSLock(bucket, object)
|
|
|
|
objectLock.RLock()
|
|
|
|
defer objectLock.RUnlock()
|
|
|
|
|
2016-07-31 17:11:14 -04:00
|
|
|
objInfo, err := objectAPI.GetObjectInfo(bucket, object)
|
2015-09-19 06:20:07 -04:00
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to fetch object info.")
|
2016-05-05 23:24:29 -04:00
|
|
|
apiErr := toAPIErrorCode(err)
|
|
|
|
if apiErr == ErrNoSuchKey {
|
2016-08-10 23:10:48 -04:00
|
|
|
apiErr = errAllowableObjectNotFound(bucket, r)
|
2015-08-03 19:17:21 -04:00
|
|
|
}
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, apiErr, r.URL.Path)
|
2015-08-03 19:17:21 -04:00
|
|
|
return
|
|
|
|
}
|
2016-02-28 21:10:37 -05:00
|
|
|
|
2016-07-06 15:50:24 -04:00
|
|
|
// Get request range.
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
var hrange *httpRange
|
2016-07-10 20:12:22 -04:00
|
|
|
rangeHeader := r.Header.Get("Range")
|
|
|
|
if rangeHeader != "" {
|
|
|
|
if hrange, err = parseRequestRange(rangeHeader, objInfo.Size); err != nil {
|
|
|
|
// Handle only errInvalidRange
|
|
|
|
// Ignore other parse error and treat it as regular Get request like Amazon S3.
|
|
|
|
if err == errInvalidRange {
|
|
|
|
writeErrorResponse(w, r, ErrInvalidRange, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// log the error.
|
|
|
|
errorIf(err, "Invalid request range")
|
2016-07-06 15:50:24 -04:00
|
|
|
}
|
|
|
|
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
}
|
2016-02-07 06:37:54 -05:00
|
|
|
|
2016-07-10 20:12:22 -04:00
|
|
|
// Validate pre-conditions if any.
|
2016-07-11 22:24:34 -04:00
|
|
|
if checkPreconditions(w, r, objInfo) {
|
2016-06-26 21:10:08 -04:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-05-28 18:13:15 -04:00
|
|
|
// Get the object.
|
2016-07-06 15:50:24 -04:00
|
|
|
startOffset := int64(0)
|
|
|
|
length := objInfo.Size
|
|
|
|
if hrange != nil {
|
2016-07-08 10:46:49 -04:00
|
|
|
startOffset = hrange.offsetBegin
|
2016-07-06 15:50:24 -04:00
|
|
|
length = hrange.getLength()
|
2016-05-28 18:13:15 -04:00
|
|
|
}
|
2016-07-10 20:12:22 -04:00
|
|
|
// Indicates if any data was written to the http.ResponseWriter
|
|
|
|
dataWritten := false
|
|
|
|
// io.Writer type which keeps track if any data was written.
|
|
|
|
writer := funcToWriter(func(p []byte) (int, error) {
|
|
|
|
if !dataWritten {
|
|
|
|
// Set headers on the first write.
|
|
|
|
// Set standard object headers.
|
|
|
|
setObjectHeaders(w, objInfo, hrange)
|
|
|
|
|
|
|
|
// Set any additional requested response headers.
|
|
|
|
setGetRespHeaders(w, r.URL.Query())
|
|
|
|
|
|
|
|
dataWritten = true
|
|
|
|
}
|
|
|
|
return w.Write(p)
|
|
|
|
})
|
2016-07-24 01:51:12 -04:00
|
|
|
|
2016-07-03 19:58:21 -04:00
|
|
|
// Reads the object at startOffset and writes to mw.
|
2016-07-31 17:11:14 -04:00
|
|
|
if err := objectAPI.GetObject(bucket, object, startOffset, length, writer); err != nil {
|
2016-07-03 19:58:21 -04:00
|
|
|
errorIf(err, "Unable to write to client.")
|
2016-07-10 20:12:22 -04:00
|
|
|
if !dataWritten {
|
|
|
|
// Error response only if no data has been written to client yet. i.e if
|
|
|
|
// partial data has already been written before an error
|
2016-07-10 20:32:59 -04:00
|
|
|
// occurred then no point in setting StatusCode and
|
2016-07-10 20:12:22 -04:00
|
|
|
// sending error XML.
|
|
|
|
apiErr := toAPIErrorCode(err)
|
|
|
|
writeErrorResponse(w, r, apiErr, r.URL.Path)
|
|
|
|
}
|
2016-05-28 18:13:15 -04:00
|
|
|
return
|
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-30 19:15:28 -04:00
|
|
|
}
|
2016-07-10 20:12:22 -04:00
|
|
|
if !dataWritten {
|
|
|
|
// If ObjectAPI.GetObject did not return error and no data has
|
|
|
|
// been written it would mean that it is a 0-byte object.
|
|
|
|
// call wrter.Write(nil) to set appropriate headers.
|
|
|
|
writer.Write(nil)
|
|
|
|
}
|
2015-02-15 20:03:27 -05:00
|
|
|
}
|
|
|
|
|
2015-06-30 23:15:48 -04:00
|
|
|
// HeadObjectHandler - HEAD Object
|
2015-02-23 19:46:48 -05:00
|
|
|
// -----------
|
|
|
|
// The HEAD operation retrieves metadata from an object without returning the object itself.
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) HeadObjectHandler(w http.ResponseWriter, r *http.Request) {
|
2015-04-22 19:28:13 -04:00
|
|
|
var object, bucket string
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2015-02-15 20:03:27 -05:00
|
|
|
bucket = vars["bucket"]
|
|
|
|
object = vars["object"]
|
2015-07-02 23:31:22 -04:00
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 16:51:05 -05:00
|
|
|
if s3Error := checkRequestAuthType(r, bucket, "s3:GetObject", serverConfig.GetRegion()); s3Error != ErrNone {
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
2016-02-15 20:42:39 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-12-10 19:15:12 -05:00
|
|
|
// Lock the object before reading.
|
|
|
|
objectLock := globalNSMutex.NewNSLock(bucket, object)
|
|
|
|
objectLock.RLock()
|
|
|
|
defer objectLock.RUnlock()
|
|
|
|
|
2016-07-31 17:11:14 -04:00
|
|
|
objInfo, err := objectAPI.GetObjectInfo(bucket, object)
|
2015-09-19 06:20:07 -04:00
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to fetch object info.")
|
2016-05-05 23:24:29 -04:00
|
|
|
apiErr := toAPIErrorCode(err)
|
|
|
|
if apiErr == ErrNoSuchKey {
|
2016-08-10 23:10:48 -04:00
|
|
|
apiErr = errAllowableObjectNotFound(bucket, r)
|
2015-09-19 06:20:07 -04:00
|
|
|
}
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, apiErr, r.URL.Path)
|
2015-08-03 19:17:21 -04:00
|
|
|
return
|
|
|
|
}
|
2016-02-28 21:10:37 -05:00
|
|
|
|
2016-07-10 20:12:22 -04:00
|
|
|
// Validate pre-conditions if any.
|
2016-07-11 22:24:34 -04:00
|
|
|
if checkPreconditions(w, r, objInfo) {
|
2016-02-28 21:10:37 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-07-10 20:12:22 -04:00
|
|
|
// Set standard object headers.
|
|
|
|
setObjectHeaders(w, objInfo, nil)
|
2016-02-28 21:10:37 -05:00
|
|
|
|
2016-07-27 14:57:08 -04:00
|
|
|
// Successful response.
|
2015-09-19 06:20:07 -04:00
|
|
|
w.WriteHeader(http.StatusOK)
|
2015-02-15 20:03:27 -05:00
|
|
|
}
|
|
|
|
|
2016-02-27 06:04:52 -05:00
|
|
|
// CopyObjectHandler - Copy Object
|
|
|
|
// ----------
|
|
|
|
// This implementation of the PUT operation adds an object to a bucket
|
|
|
|
// while reading the object from another source.
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) CopyObjectHandler(w http.ResponseWriter, r *http.Request) {
|
2016-02-27 06:04:52 -05:00
|
|
|
vars := mux.Vars(r)
|
|
|
|
bucket := vars["bucket"]
|
|
|
|
object := vars["object"]
|
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 16:51:05 -05:00
|
|
|
if s3Error := checkRequestAuthType(r, bucket, "s3:PutObject", serverConfig.GetRegion()); s3Error != ErrNone {
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
2016-02-27 06:04:52 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-07-06 17:00:29 -04:00
|
|
|
// TODO: Reject requests where body/payload is present, for now we don't even read it.
|
2016-02-27 06:04:52 -05:00
|
|
|
|
|
|
|
// objectSource
|
2016-07-06 17:00:29 -04:00
|
|
|
objectSource, err := url.QueryUnescape(r.Header.Get("X-Amz-Copy-Source"))
|
|
|
|
if err != nil {
|
|
|
|
// Save unescaped string as is.
|
|
|
|
objectSource = r.Header.Get("X-Amz-Copy-Source")
|
|
|
|
}
|
2016-02-27 06:04:52 -05:00
|
|
|
|
|
|
|
// Skip the first element if it is '/', split the rest.
|
2016-08-16 10:57:14 -04:00
|
|
|
objectSource = strings.TrimPrefix(objectSource, "/")
|
2016-02-27 06:04:52 -05:00
|
|
|
splits := strings.SplitN(objectSource, "/", 2)
|
|
|
|
|
|
|
|
// Save sourceBucket and sourceObject extracted from url Path.
|
|
|
|
var sourceBucket, sourceObject string
|
|
|
|
if len(splits) == 2 {
|
|
|
|
sourceBucket = splits[0]
|
|
|
|
sourceObject = splits[1]
|
|
|
|
}
|
|
|
|
// If source object is empty, reply back error.
|
|
|
|
if sourceObject == "" {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrInvalidCopySource, r.URL.Path)
|
2016-02-27 06:04:52 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Source and destination objects cannot be same, reply back error.
|
|
|
|
if sourceObject == object && sourceBucket == bucket {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrInvalidCopyDest, r.URL.Path)
|
2016-02-27 06:04:52 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-12-10 19:15:12 -05:00
|
|
|
// Lock the object before reading.
|
|
|
|
objectRLock := globalNSMutex.NewNSLock(sourceBucket, sourceObject)
|
|
|
|
objectRLock.RLock()
|
|
|
|
defer objectRLock.RUnlock()
|
|
|
|
|
2016-07-31 17:11:14 -04:00
|
|
|
objInfo, err := objectAPI.GetObjectInfo(sourceBucket, sourceObject)
|
2016-02-27 06:04:52 -05:00
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to fetch object info.")
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), objectSource)
|
2016-02-27 06:04:52 -05:00
|
|
|
return
|
|
|
|
}
|
2016-07-09 15:13:40 -04:00
|
|
|
|
2016-07-11 22:24:34 -04:00
|
|
|
// Verify before x-amz-copy-source preconditions before continuing with CopyObject.
|
|
|
|
if checkCopyObjectPreconditions(w, r, objInfo) {
|
2016-03-16 15:57:29 -04:00
|
|
|
return
|
|
|
|
}
|
2016-02-27 06:04:52 -05:00
|
|
|
|
|
|
|
/// maximum Upload size for object in a single CopyObject operation.
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
if isMaxObjectSize(objInfo.Size) {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrEntityTooLarge, objectSource)
|
2016-02-27 06:04:52 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-09-02 15:18:35 -04:00
|
|
|
// Size of object.
|
|
|
|
size := objInfo.Size
|
|
|
|
|
2016-05-28 18:13:15 -04:00
|
|
|
pipeReader, pipeWriter := io.Pipe()
|
|
|
|
go func() {
|
|
|
|
startOffset := int64(0) // Read the whole file.
|
|
|
|
// Get the object.
|
2016-09-02 15:18:35 -04:00
|
|
|
gErr := objectAPI.GetObject(sourceBucket, sourceObject, startOffset, size, pipeWriter)
|
2016-05-28 18:13:15 -04:00
|
|
|
if gErr != nil {
|
|
|
|
errorIf(gErr, "Unable to read an object.")
|
|
|
|
pipeWriter.CloseWithError(gErr)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pipeWriter.Close() // Close.
|
|
|
|
}()
|
|
|
|
|
2016-05-19 20:10:08 -04:00
|
|
|
// Save other metadata if available.
|
2016-09-02 02:10:50 -04:00
|
|
|
metadata := objInfo.UserDefined
|
2016-07-22 23:31:45 -04:00
|
|
|
|
2016-10-15 09:20:56 -04:00
|
|
|
// Remove the etag from source metadata because if it was uploaded as a multipart object
|
|
|
|
// then its ETag will not be MD5sum of the object.
|
|
|
|
delete(metadata, "md5Sum")
|
2016-05-19 20:10:08 -04:00
|
|
|
|
2016-10-02 18:51:49 -04:00
|
|
|
sha256sum := ""
|
2016-12-10 19:15:12 -05:00
|
|
|
|
|
|
|
objectWLock := globalNSMutex.NewNSLock(bucket, object)
|
|
|
|
objectWLock.Lock()
|
|
|
|
defer objectWLock.Unlock()
|
|
|
|
|
2016-02-27 06:04:52 -05:00
|
|
|
// Create the object.
|
2016-10-02 18:51:49 -04:00
|
|
|
objInfo, err = objectAPI.PutObject(bucket, object, size, pipeReader, metadata, sha256sum)
|
2016-02-27 06:04:52 -05:00
|
|
|
if err != nil {
|
2016-09-06 05:23:32 -04:00
|
|
|
// Close the this end of the pipe upon error in PutObject.
|
|
|
|
pipeReader.CloseWithError(err)
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to create an object.")
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
2016-02-27 06:04:52 -05:00
|
|
|
return
|
|
|
|
}
|
2016-09-06 05:23:32 -04:00
|
|
|
// Explicitly close the reader, before fetching object info.
|
|
|
|
pipeReader.Close()
|
2016-04-16 15:48:41 -04:00
|
|
|
|
2016-09-02 15:18:35 -04:00
|
|
|
md5Sum := objInfo.MD5Sum
|
2016-04-16 15:48:41 -04:00
|
|
|
response := generateCopyObjectResponse(md5Sum, objInfo.ModTime)
|
2016-03-06 15:16:22 -05:00
|
|
|
encodedSuccessResponse := encodeResponse(response)
|
2016-02-27 06:04:52 -05:00
|
|
|
// write headers
|
|
|
|
setCommonHeaders(w)
|
|
|
|
// write success response.
|
|
|
|
writeSuccessResponse(w, encodedSuccessResponse)
|
2016-07-24 01:51:12 -04:00
|
|
|
|
2016-09-29 01:46:19 -04:00
|
|
|
// Notify object created event.
|
|
|
|
eventNotify(eventData{
|
|
|
|
Type: ObjectCreatedCopy,
|
|
|
|
Bucket: bucket,
|
|
|
|
ObjInfo: objInfo,
|
|
|
|
ReqParams: map[string]string{
|
|
|
|
"sourceIPAddress": r.RemoteAddr,
|
|
|
|
},
|
|
|
|
})
|
2016-02-27 06:04:52 -05:00
|
|
|
}
|
|
|
|
|
2015-06-30 23:15:48 -04:00
|
|
|
// PutObjectHandler - PUT Object
|
2015-02-23 19:46:48 -05:00
|
|
|
// ----------
|
|
|
|
// This implementation of the PUT operation adds an object to a bucket.
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) PutObjectHandler(w http.ResponseWriter, r *http.Request) {
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-09-30 17:32:13 -04:00
|
|
|
// X-Amz-Copy-Source shouldn't be set for this call.
|
2016-02-27 06:04:52 -05:00
|
|
|
if _, ok := r.Header["X-Amz-Copy-Source"]; ok {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrInvalidCopySource, r.URL.Path)
|
2016-02-27 06:04:52 -05:00
|
|
|
return
|
|
|
|
}
|
2016-09-30 17:32:13 -04:00
|
|
|
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2016-02-27 06:04:52 -05:00
|
|
|
bucket := vars["bucket"]
|
|
|
|
object := vars["object"]
|
2015-04-22 22:29:39 -04:00
|
|
|
|
2016-03-12 04:08:55 -05:00
|
|
|
// Get Content-Md5 sent by client and verify if valid
|
2016-03-12 19:08:15 -05:00
|
|
|
md5Bytes, err := checkValidMD5(r.Header.Get("Content-Md5"))
|
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to validate content-md5 format.")
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrInvalidDigest, r.URL.Path)
|
2015-04-22 19:28:13 -04:00
|
|
|
return
|
|
|
|
}
|
2016-09-30 17:32:13 -04:00
|
|
|
|
2015-12-28 02:00:36 -05:00
|
|
|
/// if Content-Length is unknown/missing, deny the request
|
2016-02-15 20:42:39 -05:00
|
|
|
size := r.ContentLength
|
2016-08-08 23:56:29 -04:00
|
|
|
rAuthType := getRequestAuthType(r)
|
|
|
|
if rAuthType == authTypeStreamingSigned {
|
|
|
|
sizeStr := r.Header.Get("x-amz-decoded-content-length")
|
|
|
|
size, err = strconv.ParseInt(sizeStr, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
errorIf(err, "Unable to parse `x-amz-decoded-content-length` into its integer value", sizeStr)
|
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2016-02-15 20:42:39 -05:00
|
|
|
if size == -1 && !contains(r.TransferEncoding, "chunked") {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrMissingContentLength, r.URL.Path)
|
2015-04-29 05:19:51 -04:00
|
|
|
return
|
|
|
|
}
|
2016-09-30 17:32:13 -04:00
|
|
|
|
2015-04-29 13:51:59 -04:00
|
|
|
/// maximum Upload size for objects in a single operation
|
2015-04-29 05:19:51 -04:00
|
|
|
if isMaxObjectSize(size) {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrEntityTooLarge, r.URL.Path)
|
2015-04-29 05:19:51 -04:00
|
|
|
return
|
|
|
|
}
|
2015-07-02 23:31:22 -04:00
|
|
|
|
2016-07-22 23:31:45 -04:00
|
|
|
// Extract metadata to be saved from incoming HTTP header.
|
|
|
|
metadata := extractMetadataFromHeader(r.Header)
|
2016-05-18 22:54:25 -04:00
|
|
|
// Make sure we hex encode md5sum here.
|
|
|
|
metadata["md5Sum"] = hex.EncodeToString(md5Bytes)
|
|
|
|
|
2016-10-02 18:51:49 -04:00
|
|
|
sha256sum := ""
|
|
|
|
|
2016-12-10 19:15:12 -05:00
|
|
|
// Lock the object.
|
|
|
|
objectLock := globalNSMutex.NewNSLock(bucket, object)
|
|
|
|
objectLock.Lock()
|
|
|
|
defer objectLock.Unlock()
|
|
|
|
|
2016-09-02 15:18:35 -04:00
|
|
|
var objInfo ObjectInfo
|
2016-08-08 23:56:29 -04:00
|
|
|
switch rAuthType {
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
default:
|
|
|
|
// For all unknown auth types return error.
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrAccessDenied, r.URL.Path)
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
return
|
|
|
|
case authTypeAnonymous:
|
2016-04-06 21:31:40 -04:00
|
|
|
// http://docs.aws.amazon.com/AmazonS3/latest/dev/using-with-s3-actions.html
|
2016-08-10 23:10:48 -04:00
|
|
|
if s3Error := enforceBucketPolicy(bucket, "s3:PutObject", r.URL); s3Error != ErrNone {
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Create anonymous object.
|
2016-10-02 18:51:49 -04:00
|
|
|
objInfo, err = objectAPI.PutObject(bucket, object, size, r.Body, metadata, sha256sum)
|
2016-08-08 23:56:29 -04:00
|
|
|
case authTypeStreamingSigned:
|
|
|
|
// Initialize stream signature verifier.
|
|
|
|
reader, s3Error := newSignV4ChunkedReader(r)
|
|
|
|
if s3Error != ErrNone {
|
2016-09-19 13:17:46 -04:00
|
|
|
errorIf(errSignatureMismatch, dumpRequest(r))
|
2016-08-08 23:56:29 -04:00
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
2016-10-02 18:51:49 -04:00
|
|
|
objInfo, err = objectAPI.PutObject(bucket, object, size, reader, metadata, sha256sum)
|
2016-09-30 17:32:13 -04:00
|
|
|
case authTypeSignedV2, authTypePresignedV2:
|
|
|
|
s3Error := isReqAuthenticatedV2(r)
|
|
|
|
if s3Error != ErrNone {
|
|
|
|
errorIf(errSignatureMismatch, dumpRequest(r))
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
2016-10-02 18:51:49 -04:00
|
|
|
objInfo, err = objectAPI.PutObject(bucket, object, size, r.Body, metadata, sha256sum)
|
2016-04-07 06:04:18 -04:00
|
|
|
case authTypePresigned, authTypeSigned:
|
2016-10-02 18:51:49 -04:00
|
|
|
if s3Error := reqSignatureV4Verify(r); s3Error != ErrNone {
|
|
|
|
errorIf(errSignatureMismatch, dumpRequest(r))
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !skipContentSha256Cksum(r) {
|
|
|
|
sha256sum = r.Header.Get("X-Amz-Content-Sha256")
|
|
|
|
}
|
2016-07-05 04:04:50 -04:00
|
|
|
// Create object.
|
2016-10-02 18:51:49 -04:00
|
|
|
objInfo, err = objectAPI.PutObject(bucket, object, size, r.Body, metadata, sha256sum)
|
2016-02-16 21:50:36 -05:00
|
|
|
}
|
2015-09-19 06:20:07 -04:00
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to create an object.")
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
2015-08-03 19:17:21 -04:00
|
|
|
return
|
|
|
|
}
|
2016-09-02 15:18:35 -04:00
|
|
|
w.Header().Set("ETag", "\""+objInfo.MD5Sum+"\"")
|
2016-01-08 03:40:06 -05:00
|
|
|
writeSuccessResponse(w, nil)
|
2016-07-24 01:51:12 -04:00
|
|
|
|
2016-09-29 01:46:19 -04:00
|
|
|
// Notify object created event.
|
|
|
|
eventNotify(eventData{
|
|
|
|
Type: ObjectCreatedPut,
|
|
|
|
Bucket: bucket,
|
|
|
|
ObjInfo: objInfo,
|
|
|
|
ReqParams: map[string]string{
|
|
|
|
"sourceIPAddress": r.RemoteAddr,
|
|
|
|
},
|
|
|
|
})
|
2015-02-15 20:03:27 -05:00
|
|
|
}
|
2015-05-07 22:55:30 -04:00
|
|
|
|
2016-04-12 15:45:15 -04:00
|
|
|
/// Multipart objectAPIHandlers
|
2015-06-08 14:06:06 -04:00
|
|
|
|
2016-10-06 16:34:33 -04:00
|
|
|
// NewMultipartUploadHandler - New multipart upload.
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) NewMultipartUploadHandler(w http.ResponseWriter, r *http.Request) {
|
2015-05-07 22:55:30 -04:00
|
|
|
var object, bucket string
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2015-05-07 22:55:30 -04:00
|
|
|
bucket = vars["bucket"]
|
|
|
|
object = vars["object"]
|
2015-07-02 23:31:22 -04:00
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 16:51:05 -05:00
|
|
|
if s3Error := checkRequestAuthType(r, bucket, "s3:PutObject", serverConfig.GetRegion()); s3Error != ErrNone {
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
2016-03-12 19:08:15 -05:00
|
|
|
return
|
2016-02-15 20:42:39 -05:00
|
|
|
}
|
|
|
|
|
2016-07-22 23:31:45 -04:00
|
|
|
// Extract metadata that needs to be saved.
|
|
|
|
metadata := extractMetadataFromHeader(r.Header)
|
2016-05-18 22:54:25 -04:00
|
|
|
|
2016-07-31 17:11:14 -04:00
|
|
|
uploadID, err := objectAPI.NewMultipartUpload(bucket, object, metadata)
|
2015-09-19 06:20:07 -04:00
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to initiate new multipart upload id.")
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
2015-08-03 19:17:21 -04:00
|
|
|
return
|
|
|
|
}
|
2015-09-19 06:20:07 -04:00
|
|
|
|
|
|
|
response := generateInitiateMultipartUploadResponse(bucket, object, uploadID)
|
2016-03-06 15:16:22 -05:00
|
|
|
encodedSuccessResponse := encodeResponse(response)
|
2015-09-19 06:20:07 -04:00
|
|
|
// write headers
|
2016-01-08 03:40:06 -05:00
|
|
|
setCommonHeaders(w)
|
|
|
|
// write success response.
|
|
|
|
writeSuccessResponse(w, encodedSuccessResponse)
|
2015-05-07 22:55:30 -04:00
|
|
|
}
|
|
|
|
|
2015-06-30 23:15:48 -04:00
|
|
|
// PutObjectPartHandler - Upload part
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) PutObjectPartHandler(w http.ResponseWriter, r *http.Request) {
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2015-10-16 22:09:35 -04:00
|
|
|
bucket := vars["bucket"]
|
|
|
|
object := vars["object"]
|
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-03-05 19:43:48 -05:00
|
|
|
// get Content-Md5 sent by client and verify if valid
|
2016-03-12 19:08:15 -05:00
|
|
|
md5Bytes, err := checkValidMD5(r.Header.Get("Content-Md5"))
|
|
|
|
if err != nil {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrInvalidDigest, r.URL.Path)
|
2015-10-16 22:09:35 -04:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-12-28 02:00:36 -05:00
|
|
|
/// if Content-Length is unknown/missing, throw away
|
2016-02-15 20:42:39 -05:00
|
|
|
size := r.ContentLength
|
2016-08-08 23:56:29 -04:00
|
|
|
|
|
|
|
rAuthType := getRequestAuthType(r)
|
|
|
|
// For auth type streaming signature, we need to gather a different content length.
|
|
|
|
if rAuthType == authTypeStreamingSigned {
|
|
|
|
sizeStr := r.Header.Get("x-amz-decoded-content-length")
|
|
|
|
size, err = strconv.ParseInt(sizeStr, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
errorIf(err, "Unable to parse `x-amz-decoded-content-length` into its integer value", sizeStr)
|
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2015-12-28 02:00:36 -05:00
|
|
|
if size == -1 {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrMissingContentLength, r.URL.Path)
|
2015-12-28 02:00:36 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-08 01:43:19 -04:00
|
|
|
/// maximum Upload size for multipart objects in a single operation
|
2015-05-07 22:55:30 -04:00
|
|
|
if isMaxObjectSize(size) {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrEntityTooLarge, r.URL.Path)
|
2015-05-07 22:55:30 -04:00
|
|
|
return
|
|
|
|
}
|
2015-04-30 19:29:03 -04:00
|
|
|
|
2016-02-15 20:42:39 -05:00
|
|
|
uploadID := r.URL.Query().Get("uploadId")
|
|
|
|
partIDString := r.URL.Query().Get("partNumber")
|
2015-05-09 22:39:00 -04:00
|
|
|
|
2016-04-29 17:24:10 -04:00
|
|
|
partID, err := strconv.Atoi(partIDString)
|
|
|
|
if err != nil {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrInvalidPart, r.URL.Path)
|
2016-03-02 14:22:58 -05:00
|
|
|
return
|
2015-05-07 22:55:30 -04:00
|
|
|
}
|
2015-07-02 23:31:22 -04:00
|
|
|
|
2016-05-24 04:52:47 -04:00
|
|
|
// check partID with maximum part ID for multipart objects
|
|
|
|
if isMaxPartID(partID) {
|
|
|
|
writeErrorResponse(w, r, ErrInvalidMaxParts, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-03-02 14:22:58 -05:00
|
|
|
var partMD5 string
|
2016-07-05 04:04:50 -04:00
|
|
|
incomingMD5 := hex.EncodeToString(md5Bytes)
|
2016-10-02 18:51:49 -04:00
|
|
|
sha256sum := ""
|
2016-08-08 23:56:29 -04:00
|
|
|
switch rAuthType {
|
2016-03-12 19:08:15 -05:00
|
|
|
default:
|
|
|
|
// For all unknown auth types return error.
|
|
|
|
writeErrorResponse(w, r, ErrAccessDenied, r.URL.Path)
|
|
|
|
return
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
case authTypeAnonymous:
|
|
|
|
// http://docs.aws.amazon.com/AmazonS3/latest/dev/mpuAndPermissions.html
|
2016-08-10 23:10:48 -04:00
|
|
|
if s3Error := enforceBucketPolicy(bucket, "s3:PutObject", r.URL); s3Error != ErrNone {
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
2016-07-05 04:04:50 -04:00
|
|
|
// No need to verify signature, anonymous request access is already allowed.
|
2016-10-02 18:51:49 -04:00
|
|
|
partMD5, err = objectAPI.PutObjectPart(bucket, object, uploadID, partID, size, r.Body, incomingMD5, sha256sum)
|
2016-08-08 23:56:29 -04:00
|
|
|
case authTypeStreamingSigned:
|
|
|
|
// Initialize stream signature verifier.
|
|
|
|
reader, s3Error := newSignV4ChunkedReader(r)
|
|
|
|
if s3Error != ErrNone {
|
2016-09-19 13:17:46 -04:00
|
|
|
errorIf(errSignatureMismatch, dumpRequest(r))
|
2016-08-08 23:56:29 -04:00
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
2016-10-02 18:51:49 -04:00
|
|
|
partMD5, err = objectAPI.PutObjectPart(bucket, object, uploadID, partID, size, reader, incomingMD5, sha256sum)
|
2016-09-30 17:32:13 -04:00
|
|
|
case authTypeSignedV2, authTypePresignedV2:
|
|
|
|
s3Error := isReqAuthenticatedV2(r)
|
|
|
|
if s3Error != ErrNone {
|
|
|
|
errorIf(errSignatureMismatch, dumpRequest(r))
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
2016-10-02 18:51:49 -04:00
|
|
|
partMD5, err = objectAPI.PutObjectPart(bucket, object, uploadID, partID, size, r.Body, incomingMD5, sha256sum)
|
2016-04-07 06:04:18 -04:00
|
|
|
case authTypePresigned, authTypeSigned:
|
2016-10-02 18:51:49 -04:00
|
|
|
if s3Error := reqSignatureV4Verify(r); s3Error != ErrNone {
|
|
|
|
errorIf(errSignatureMismatch, dumpRequest(r))
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if !skipContentSha256Cksum(r) {
|
|
|
|
sha256sum = r.Header.Get("X-Amz-Content-Sha256")
|
|
|
|
}
|
|
|
|
partMD5, err = objectAPI.PutObjectPart(bucket, object, uploadID, partID, size, r.Body, incomingMD5, sha256sum)
|
2016-02-16 21:50:36 -05:00
|
|
|
}
|
2015-09-19 06:20:07 -04:00
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to create object part.")
|
2016-03-12 19:08:15 -05:00
|
|
|
// Verify if the underlying error is signature mismatch.
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
2015-08-03 19:17:21 -04:00
|
|
|
return
|
|
|
|
}
|
2016-03-02 14:22:58 -05:00
|
|
|
if partMD5 != "" {
|
|
|
|
w.Header().Set("ETag", "\""+partMD5+"\"")
|
2016-02-01 15:19:54 -05:00
|
|
|
}
|
2016-01-08 03:40:06 -05:00
|
|
|
writeSuccessResponse(w, nil)
|
2015-05-07 22:55:30 -04:00
|
|
|
}
|
|
|
|
|
2015-06-30 23:15:48 -04:00
|
|
|
// AbortMultipartUploadHandler - Abort multipart upload
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) AbortMultipartUploadHandler(w http.ResponseWriter, r *http.Request) {
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2015-05-09 19:06:35 -04:00
|
|
|
bucket := vars["bucket"]
|
|
|
|
object := vars["object"]
|
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 16:51:05 -05:00
|
|
|
if s3Error := checkRequestAuthType(r, bucket, "s3:AbortMultipartUpload", serverConfig.GetRegion()); s3Error != ErrNone {
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
2016-03-12 19:08:15 -05:00
|
|
|
return
|
2016-02-15 20:42:39 -05:00
|
|
|
}
|
|
|
|
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
uploadID, _, _, _ := getObjectResources(r.URL.Query())
|
2016-07-31 17:11:14 -04:00
|
|
|
if err := objectAPI.AbortMultipartUpload(bucket, object, uploadID); err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to abort multipart upload.")
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
2015-08-03 19:17:21 -04:00
|
|
|
return
|
|
|
|
}
|
2015-10-16 23:02:37 -04:00
|
|
|
writeSuccessNoContent(w)
|
2015-05-09 19:06:35 -04:00
|
|
|
}
|
|
|
|
|
2015-06-30 23:15:48 -04:00
|
|
|
// ListObjectPartsHandler - List object parts
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) ListObjectPartsHandler(w http.ResponseWriter, r *http.Request) {
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2015-10-16 22:09:35 -04:00
|
|
|
bucket := vars["bucket"]
|
|
|
|
object := vars["object"]
|
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 16:51:05 -05:00
|
|
|
if s3Error := checkRequestAuthType(r, bucket, "s3:ListMultipartUploadParts", serverConfig.GetRegion()); s3Error != ErrNone {
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
2016-03-12 19:08:15 -05:00
|
|
|
return
|
2016-02-15 20:42:39 -05:00
|
|
|
}
|
|
|
|
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
uploadID, partNumberMarker, maxParts, _ := getObjectResources(r.URL.Query())
|
|
|
|
if partNumberMarker < 0 {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrInvalidPartNumberMarker, r.URL.Path)
|
2015-07-16 20:22:45 -04:00
|
|
|
return
|
|
|
|
}
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
if maxParts < 0 {
|
2016-03-10 05:24:52 -05:00
|
|
|
writeErrorResponse(w, r, ErrInvalidMaxParts, r.URL.Path)
|
2015-07-16 20:22:45 -04:00
|
|
|
return
|
|
|
|
}
|
2016-07-31 17:11:14 -04:00
|
|
|
listPartsInfo, err := objectAPI.ListObjectParts(bucket, object, uploadID, partNumberMarker, maxParts)
|
2015-09-19 06:20:07 -04:00
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to list uploaded parts.")
|
2016-05-05 23:24:29 -04:00
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
2015-08-03 19:17:21 -04:00
|
|
|
return
|
|
|
|
}
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
response := generateListPartsResponse(listPartsInfo)
|
2016-03-06 15:16:22 -05:00
|
|
|
encodedSuccessResponse := encodeResponse(response)
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
// Write headers.
|
2016-01-08 03:40:06 -05:00
|
|
|
setCommonHeaders(w)
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
// Write success response.
|
2016-01-08 03:40:06 -05:00
|
|
|
writeSuccessResponse(w, encodedSuccessResponse)
|
2015-05-09 14:41:26 -04:00
|
|
|
}
|
|
|
|
|
2016-09-21 23:08:08 -04:00
|
|
|
// CompleteMultipartUploadHandler - Complete multipart upload.
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) CompleteMultipartUploadHandler(w http.ResponseWriter, r *http.Request) {
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2015-05-07 22:55:30 -04:00
|
|
|
bucket := vars["bucket"]
|
|
|
|
object := vars["object"]
|
2015-06-30 17:42:29 -04:00
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 16:51:05 -05:00
|
|
|
if s3Error := checkRequestAuthType(r, bucket, "s3:PutObject", serverConfig.GetRegion()); s3Error != ErrNone {
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
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-30 19:15:28 -04:00
|
|
|
// Get upload id.
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
uploadID, _, _, _ := getObjectResources(r.URL.Query())
|
2016-03-02 14:22:58 -05:00
|
|
|
|
2016-04-16 15:48:41 -04:00
|
|
|
var md5Sum string
|
2016-04-29 17:24:10 -04:00
|
|
|
completeMultipartBytes, err := ioutil.ReadAll(r.Body)
|
|
|
|
if err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to complete multipart upload.")
|
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-30 19:15:28 -04:00
|
|
|
writeErrorResponse(w, r, ErrInternalError, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
complMultipartUpload := &completeMultipartUpload{}
|
2016-04-29 17:24:10 -04:00
|
|
|
if err = xml.Unmarshal(completeMultipartBytes, complMultipartUpload); err != nil {
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to parse complete multipart upload XML.")
|
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-30 19:15:28 -04:00
|
|
|
writeErrorResponse(w, r, ErrMalformedXML, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
2016-05-25 15:11:26 -04:00
|
|
|
if len(complMultipartUpload.Parts) == 0 {
|
|
|
|
writeErrorResponse(w, r, ErrMalformedXML, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
objectAPI: Fix object API interface, remove unnecessary structs.
ObjectAPI changes.
```
ListObjects(bucket, prefix, marker, delimiter string, maxKeys int) (ListObjectsInfo, *probe.Error)
ListMultipartUploads(bucket, objectPrefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int) (ListMultipartsInfo, *probe.Error)
ListObjectParts(bucket, object, uploadID string, partNumberMarker, maxParts int) (ListPartsInfo, *probe.Error)
CompleteMultipartUpload(bucket string, object string, uploadID string, parts []completePart) (ObjectInfo, *probe.Error)
```
2016-04-03 04:34:20 -04:00
|
|
|
if !sort.IsSorted(completedParts(complMultipartUpload.Parts)) {
|
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-30 19:15:28 -04:00
|
|
|
writeErrorResponse(w, r, ErrInvalidPartOrder, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Complete parts.
|
2016-04-11 04:29:18 -04:00
|
|
|
var completeParts []completePart
|
|
|
|
for _, part := range complMultipartUpload.Parts {
|
|
|
|
part.ETag = strings.TrimPrefix(part.ETag, "\"")
|
|
|
|
part.ETag = strings.TrimSuffix(part.ETag, "\"")
|
|
|
|
completeParts = append(completeParts, part)
|
|
|
|
}
|
2016-08-10 21:47:49 -04:00
|
|
|
|
2016-12-10 19:15:12 -05:00
|
|
|
// Hold write lock on the object.
|
|
|
|
destLock := globalNSMutex.NewNSLock(bucket, object)
|
|
|
|
destLock.Lock()
|
|
|
|
defer destLock.Unlock()
|
|
|
|
|
2016-09-21 23:08:08 -04:00
|
|
|
md5Sum, err = objectAPI.CompleteMultipartUpload(bucket, object, uploadID, completeParts)
|
2015-09-19 06:20:07 -04:00
|
|
|
if err != nil {
|
2016-09-21 23:08:08 -04:00
|
|
|
err = errorCause(err)
|
2016-05-16 17:31:28 -04:00
|
|
|
errorIf(err, "Unable to complete multipart upload.")
|
2016-06-28 17:51:49 -04:00
|
|
|
switch oErr := err.(type) {
|
|
|
|
case PartTooSmall:
|
|
|
|
// Write part too small error.
|
|
|
|
writePartSmallErrorResponse(w, r, oErr)
|
|
|
|
default:
|
|
|
|
// Handle all other generic issues.
|
2016-09-21 23:08:08 -04:00
|
|
|
writeErrorResponse(w, r, toAPIErrorCode(err), r.URL.Path)
|
2016-06-28 17:51:49 -04:00
|
|
|
}
|
2015-08-03 19:17:21 -04:00
|
|
|
return
|
|
|
|
}
|
2016-06-15 23:31:06 -04:00
|
|
|
|
2016-03-12 19:08:15 -05:00
|
|
|
// Get object location.
|
2016-03-01 23:01:40 -05:00
|
|
|
location := getLocation(r)
|
|
|
|
// Generate complete multipart response.
|
2016-04-16 15:48:41 -04:00
|
|
|
response := generateCompleteMultpartUploadResponse(bucket, object, location, md5Sum)
|
2016-09-21 23:08:08 -04:00
|
|
|
encodedSuccessResponse := encodeResponse(response)
|
2016-06-30 21:48:50 -04:00
|
|
|
if err != nil {
|
|
|
|
errorIf(err, "Unable to parse CompleteMultipartUpload response")
|
|
|
|
writeErrorResponseNoHeader(w, r, ErrInternalError, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
2016-07-24 01:51:12 -04:00
|
|
|
|
2016-11-10 10:41:02 -05:00
|
|
|
// Set etag.
|
|
|
|
w.Header().Set("ETag", "\""+md5Sum+"\"")
|
|
|
|
|
2016-07-24 01:51:12 -04:00
|
|
|
// Write success response.
|
2016-06-15 23:31:06 -04:00
|
|
|
w.Write(encodedSuccessResponse)
|
|
|
|
w.(http.Flusher).Flush()
|
2016-07-24 01:51:12 -04:00
|
|
|
|
2016-09-29 01:46:19 -04:00
|
|
|
// Fetch object info for notifications.
|
|
|
|
objInfo, err := objectAPI.GetObjectInfo(bucket, object)
|
|
|
|
if err != nil {
|
|
|
|
errorIf(err, "Unable to fetch object info for \"%s\"", path.Join(bucket, object))
|
|
|
|
return
|
2016-08-05 01:01:58 -04:00
|
|
|
}
|
2016-09-29 01:46:19 -04:00
|
|
|
|
|
|
|
// Notify object created event.
|
|
|
|
eventNotify(eventData{
|
|
|
|
Type: ObjectCreatedCompleteMultipartUpload,
|
|
|
|
Bucket: bucket,
|
|
|
|
ObjInfo: objInfo,
|
|
|
|
ReqParams: map[string]string{
|
|
|
|
"sourceIPAddress": r.RemoteAddr,
|
|
|
|
},
|
|
|
|
})
|
2015-05-07 22:55:30 -04:00
|
|
|
}
|
2015-06-08 14:06:06 -04:00
|
|
|
|
2016-04-12 15:45:15 -04:00
|
|
|
/// Delete objectAPIHandlers
|
2015-06-08 14:06:06 -04:00
|
|
|
|
2016-03-05 19:43:48 -05:00
|
|
|
// DeleteObjectHandler - delete an object
|
2016-04-12 15:45:15 -04:00
|
|
|
func (api objectAPIHandlers) DeleteObjectHandler(w http.ResponseWriter, r *http.Request) {
|
2016-02-15 20:42:39 -05:00
|
|
|
vars := mux.Vars(r)
|
2015-10-16 14:26:01 -04:00
|
|
|
bucket := vars["bucket"]
|
|
|
|
object := vars["object"]
|
|
|
|
|
2016-08-10 21:47:49 -04:00
|
|
|
objectAPI := api.ObjectAPI()
|
|
|
|
if objectAPI == nil {
|
|
|
|
writeErrorResponse(w, r, ErrServerNotInitialized, r.URL.Path)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-21 16:51:05 -05:00
|
|
|
if s3Error := checkRequestAuthType(r, bucket, "s3:DeleteObject", serverConfig.GetRegion()); s3Error != ErrNone {
|
|
|
|
writeErrorResponse(w, r, s3Error, r.URL.Path)
|
accessPolicy: Implement Put, Get, Delete access policy.
This patch implements Get,Put,Delete bucket policies
Supporting - http://docs.aws.amazon.com/AmazonS3/latest/dev/access-policy-language-overview.html
Currently supports following actions.
"*": true,
"s3:*": true,
"s3:GetObject": true,
"s3:ListBucket": true,
"s3:PutObject": true,
"s3:CreateBucket": true,
"s3:GetBucketLocation": true,
"s3:DeleteBucket": true,
"s3:DeleteObject": true,
"s3:AbortMultipartUpload": true,
"s3:ListBucketMultipartUploads": true,
"s3:ListMultipartUploadParts": true,
following conditions for "StringEquals" and "StringNotEquals"
"s3:prefix", "s3:max-keys"
2016-02-03 19:46:56 -05:00
|
|
|
return
|
2016-02-04 15:52:25 -05:00
|
|
|
}
|
2016-11-21 16:51:05 -05:00
|
|
|
|
2016-12-10 19:15:12 -05:00
|
|
|
objectLock := globalNSMutex.NewNSLock(bucket, object)
|
|
|
|
objectLock.Lock()
|
|
|
|
defer objectLock.Unlock()
|
|
|
|
|
2016-05-14 18:47:19 -04:00
|
|
|
/// http://docs.aws.amazon.com/AmazonS3/latest/API/RESTObjectDELETE.html
|
|
|
|
/// Ignore delete object errors, since we are suppposed to reply
|
|
|
|
/// only 204.
|
2016-07-31 17:11:14 -04:00
|
|
|
if err := objectAPI.DeleteObject(bucket, object); err != nil {
|
2016-07-24 01:51:12 -04:00
|
|
|
writeSuccessNoContent(w)
|
|
|
|
return
|
|
|
|
}
|
2015-10-16 23:02:37 -04:00
|
|
|
writeSuccessNoContent(w)
|
2016-07-24 01:51:12 -04:00
|
|
|
|
2016-09-29 01:46:19 -04:00
|
|
|
// Notify object deleted event.
|
|
|
|
eventNotify(eventData{
|
|
|
|
Type: ObjectRemovedDelete,
|
|
|
|
Bucket: bucket,
|
|
|
|
ObjInfo: ObjectInfo{
|
|
|
|
Name: object,
|
|
|
|
},
|
|
|
|
ReqParams: map[string]string{
|
|
|
|
"sourceIPAddress": r.RemoteAddr,
|
|
|
|
},
|
|
|
|
})
|
2015-06-08 14:06:06 -04:00
|
|
|
}
|