From fe796ba53872f6de3bd786fff9067b583d613a86 Mon Sep 17 00:00:00 2001 From: Marina Biryukova Date: Thu, 19 Oct 2023 17:22:26 +0300 Subject: [PATCH] [#217] Consider Copy-Source-SSE-* headers during copy Signed-off-by: Marina Biryukova --- CHANGELOG.md | 1 + api/errors/errors.go | 7 ++ api/handler/copy.go | 45 ++++++-- api/handler/copy_test.go | 165 +++++++++++++++++++++++++++ api/handler/delete_test.go | 5 +- api/handler/handlers_test.go | 14 ++- api/handler/locking_test.go | 5 +- api/handler/multipart_upload.go | 25 ++-- api/handler/multipart_upload_test.go | 48 ++++++++ api/handler/object_list_test.go | 11 +- api/handler/put.go | 37 +++++- api/headers.go | 4 + api/layer/encryption/encryption.go | 10 +- api/layer/layer.go | 18 ++- api/layer/multipart_upload.go | 15 ++- 15 files changed, 355 insertions(+), 55 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 7da941290..a78169332 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -36,6 +36,7 @@ This document outlines major changes between releases. - Add new `frostfs.client_cut` config param (#192) - Add new `frostfs.buffer_max_size_for_put` config param and sync TZ hash for PUT operations (#197) - Add `X-Amz-Version-Id` header after complete multipart upload (#227) +- Add handling of `X-Amz-Copy-Source-Server-Side-Encryption-Customer-*` headers during copy (#217) ### Changed - Update prometheus to v1.15.0 (#94) diff --git a/api/errors/errors.go b/api/errors/errors.go index d9c2bfaa6..68d2eacb7 100644 --- a/api/errors/errors.go +++ b/api/errors/errors.go @@ -149,6 +149,7 @@ const ( ErrInvalidEncryptionAlgorithm ErrInvalidSSECustomerKey ErrMissingSSECustomerKey + ErrMissingSSECustomerAlgorithm ErrMissingSSECustomerKeyMD5 ErrSSECustomerKeyMD5Mismatch ErrInvalidSSECustomerParameters @@ -1062,6 +1063,12 @@ var errorCodes = errorCodeMap{ Description: "Requests specifying Server Side Encryption with Customer provided keys must provide an appropriate secret key.", HTTPStatusCode: http.StatusBadRequest, }, + ErrMissingSSECustomerAlgorithm: { + ErrCode: ErrMissingSSECustomerAlgorithm, + Code: "InvalidArgument", + Description: "Requests specifying Server Side Encryption with Customer provided keys must provide a valid encryption algorithm.", + HTTPStatusCode: http.StatusBadRequest, + }, ErrMissingSSECustomerKeyMD5: { ErrCode: ErrMissingSSECustomerKeyMD5, Code: "InvalidArgument", diff --git a/api/handler/copy.go b/api/handler/copy.go index cefb52511..611d5f523 100644 --- a/api/handler/copy.go +++ b/api/handler/copy.go @@ -107,23 +107,36 @@ func (h *handler) CopyObjectHandler(w http.ResponseWriter, r *http.Request) { } srcObjInfo := extendedSrcObjInfo.ObjectInfo - encryptionParams, err := formEncryptionParams(r) + srcEncryptionParams, err := formCopySourceEncryptionParams(r) + if err != nil { + h.logAndSendError(w, "invalid sse headers", reqInfo, err) + return + } + dstEncryptionParams, err := formEncryptionParams(r) if err != nil { h.logAndSendError(w, "invalid sse headers", reqInfo, err) return } - if err = encryptionParams.MatchObjectEncryption(layer.FormEncryptionInfo(srcObjInfo.Headers)); err != nil { + if err = srcEncryptionParams.MatchObjectEncryption(layer.FormEncryptionInfo(srcObjInfo.Headers)); err != nil { + if errors.IsS3Error(err, errors.ErrInvalidEncryptionParameters) || errors.IsS3Error(err, errors.ErrSSEEncryptedObject) || + errors.IsS3Error(err, errors.ErrInvalidSSECustomerParameters) { + h.logAndSendError(w, "encryption doesn't match object", reqInfo, err, zap.Error(err)) + return + } h.logAndSendError(w, "encryption doesn't match object", reqInfo, errors.GetAPIError(errors.ErrBadRequest), zap.Error(err)) return } + var dstSize uint64 if srcSize, err := layer.GetObjectSize(srcObjInfo); err != nil { h.logAndSendError(w, "failed to get source object size", reqInfo, err) return } else if srcSize > layer.UploadMaxSize { //https://docs.aws.amazon.com/AmazonS3/latest/API/API_CopyObject.html h.logAndSendError(w, "too bid object to copy with single copy operation, use multipart upload copy instead", reqInfo, errors.GetAPIError(errors.ErrInvalidRequestLargeCopy)) return + } else { + dstSize = srcSize } args, err := parseCopyObjectArgs(r.Header) @@ -174,20 +187,21 @@ func (h *handler) CopyObjectHandler(w http.ResponseWriter, r *http.Request) { srcObjInfo.Headers[api.ContentType] = srcObjInfo.ContentType } metadata = makeCopyMap(srcObjInfo.Headers) - delete(metadata, layer.MultipartObjectSize) // object payload will be real one rather than list of compound parts + filterMetadataMap(metadata) } else if contentType := r.Header.Get(api.ContentType); len(contentType) > 0 { metadata[api.ContentType] = contentType } params := &layer.CopyObjectParams{ - SrcVersioned: srcObjPrm.Versioned(), - SrcObject: srcObjInfo, - ScrBktInfo: srcObjPrm.BktInfo, - DstBktInfo: dstBktInfo, - DstObject: reqInfo.ObjectName, - SrcSize: srcObjInfo.Size, - Header: metadata, - Encryption: encryptionParams, + SrcVersioned: srcObjPrm.Versioned(), + SrcObject: srcObjInfo, + ScrBktInfo: srcObjPrm.BktInfo, + DstBktInfo: dstBktInfo, + DstObject: reqInfo.ObjectName, + DstSize: dstSize, + Header: metadata, + SrcEncryption: srcEncryptionParams, + DstEncryption: dstEncryptionParams, } params.CopiesNumbers, err = h.pickCopiesNumbers(metadata, dstBktInfo.LocationConstraint) @@ -262,7 +276,7 @@ func (h *handler) CopyObjectHandler(w http.ResponseWriter, r *http.Request) { h.reqLogger(ctx).Error(logs.CouldntSendNotification, zap.Error(err)) } - if encryptionParams.Enabled() { + if dstEncryptionParams.Enabled() { addSSECHeaders(w.Header(), r.Header) } } @@ -275,6 +289,13 @@ func makeCopyMap(headers map[string]string) map[string]string { return res } +func filterMetadataMap(metadata map[string]string) { + delete(metadata, layer.MultipartObjectSize) // object payload will be real one rather than list of compound parts + for key := range layer.EncryptionMetadata { + delete(metadata, key) + } +} + func isCopyingToItselfForbidden(reqInfo *middleware.ReqInfo, srcBucket string, srcObject string, settings *data.BucketSettings, args *copyObjectArgs) bool { if reqInfo.BucketName != srcBucket || reqInfo.ObjectName != srcObject { return false diff --git a/api/handler/copy_test.go b/api/handler/copy_test.go index 9a97a5538..9ee078309 100644 --- a/api/handler/copy_test.go +++ b/api/handler/copy_test.go @@ -1,13 +1,19 @@ package handler import ( + "crypto/md5" + "crypto/tls" + "encoding/base64" "encoding/xml" "net/http" "net/url" + "strconv" "testing" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api" + "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/errors" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer" + "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer/encryption" "github.com/stretchr/testify/require" ) @@ -98,6 +104,165 @@ func TestCopyMultipart(t *testing.T) { equalDataSlices(t, data, copiedData) } +func TestCopyEncryptedToUnencrypted(t *testing.T) { + tc := prepareHandlerContext(t) + + bktName, srcObjName := "bucket-for-copy", "object-for-copy" + key1 := []byte("firstencriptionkeyofsourceobject") + key1Md5 := md5.Sum(key1) + key2 := []byte("anotherencriptionkeysourceobject") + key2Md5 := md5.Sum(key2) + bktInfo := createTestBucket(tc, bktName) + + srcEnc, err := encryption.NewParams(key1) + require.NoError(t, err) + srcObjInfo := createTestObject(tc, bktInfo, srcObjName, *srcEnc) + require.True(t, containEncryptionMetadataHeaders(srcObjInfo.Headers)) + + dstObjName := "copy-object" + + // empty copy-source-sse headers + w, r := prepareTestRequest(tc, bktName, dstObjName, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, bktName+"/"+srcObjName) + tc.Handler().CopyObjectHandler(w, r) + + assertStatus(t, w, http.StatusBadRequest) + assertS3Error(t, w, errors.GetAPIError(errors.ErrSSEEncryptedObject)) + + // empty copy-source-sse-custom-key + w, r = prepareTestRequest(tc, bktName, dstObjName, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, bktName+"/"+srcObjName) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerAlgorithm, layer.AESEncryptionAlgorithm) + tc.Handler().CopyObjectHandler(w, r) + + assertStatus(t, w, http.StatusBadRequest) + assertS3Error(t, w, errors.GetAPIError(errors.ErrMissingSSECustomerKey)) + + // empty copy-source-sse-custom-algorithm + w, r = prepareTestRequest(tc, bktName, dstObjName, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, bktName+"/"+srcObjName) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKey, base64.StdEncoding.EncodeToString(key1)) + tc.Handler().CopyObjectHandler(w, r) + + assertStatus(t, w, http.StatusBadRequest) + assertS3Error(t, w, errors.GetAPIError(errors.ErrMissingSSECustomerAlgorithm)) + + // invalid copy-source-sse-custom-key + w, r = prepareTestRequest(tc, bktName, dstObjName, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, bktName+"/"+srcObjName) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerAlgorithm, layer.AESEncryptionAlgorithm) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKey, base64.StdEncoding.EncodeToString(key2)) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKeyMD5, base64.StdEncoding.EncodeToString(key2Md5[:])) + tc.Handler().CopyObjectHandler(w, r) + + assertStatus(t, w, http.StatusBadRequest) + assertS3Error(t, w, errors.GetAPIError(errors.ErrInvalidSSECustomerParameters)) + + // success copy + w, r = prepareTestRequest(tc, bktName, dstObjName, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, bktName+"/"+srcObjName) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerAlgorithm, layer.AESEncryptionAlgorithm) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKey, base64.StdEncoding.EncodeToString(key1)) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKeyMD5, base64.StdEncoding.EncodeToString(key1Md5[:])) + tc.Handler().CopyObjectHandler(w, r) + + assertStatus(t, w, http.StatusOK) + dstObjInfo, err := tc.Layer().GetObjectInfo(tc.Context(), &layer.HeadObjectParams{BktInfo: bktInfo, Object: dstObjName}) + require.NoError(t, err) + require.Equal(t, srcObjInfo.Headers[layer.AttributeDecryptedSize], strconv.Itoa(int(dstObjInfo.Size))) + require.False(t, containEncryptionMetadataHeaders(dstObjInfo.Headers)) +} + +func TestCopyUnencryptedToEncrypted(t *testing.T) { + tc := prepareHandlerContext(t) + + bktName, srcObjName := "bucket-for-copy", "object-for-copy" + key := []byte("firstencriptionkeyofsourceobject") + keyMd5 := md5.Sum(key) + bktInfo := createTestBucket(tc, bktName) + + srcObjInfo := createTestObject(tc, bktInfo, srcObjName, encryption.Params{}) + require.False(t, containEncryptionMetadataHeaders(srcObjInfo.Headers)) + + dstObjName := "copy-object" + + // invalid copy-source-sse headers + w, r := prepareTestRequest(tc, bktName, dstObjName, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, bktName+"/"+srcObjName) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerAlgorithm, layer.AESEncryptionAlgorithm) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKey, base64.StdEncoding.EncodeToString(key)) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKeyMD5, base64.StdEncoding.EncodeToString(keyMd5[:])) + tc.Handler().CopyObjectHandler(w, r) + + assertStatus(t, w, http.StatusBadRequest) + assertS3Error(t, w, errors.GetAPIError(errors.ErrInvalidEncryptionParameters)) + + // success copy + w, r = prepareTestRequest(tc, bktName, dstObjName, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, bktName+"/"+srcObjName) + r.Header.Set(api.AmzServerSideEncryptionCustomerAlgorithm, layer.AESEncryptionAlgorithm) + r.Header.Set(api.AmzServerSideEncryptionCustomerKey, base64.StdEncoding.EncodeToString(key)) + r.Header.Set(api.AmzServerSideEncryptionCustomerKeyMD5, base64.StdEncoding.EncodeToString(keyMd5[:])) + tc.Handler().CopyObjectHandler(w, r) + + assertStatus(t, w, http.StatusOK) + dstObjInfo, err := tc.Layer().GetObjectInfo(tc.Context(), &layer.HeadObjectParams{BktInfo: bktInfo, Object: dstObjName}) + require.NoError(t, err) + require.True(t, containEncryptionMetadataHeaders(dstObjInfo.Headers)) + require.Equal(t, strconv.Itoa(int(srcObjInfo.Size)), dstObjInfo.Headers[layer.AttributeDecryptedSize]) +} + +func TestCopyEncryptedToEncryptedWithAnotherKey(t *testing.T) { + tc := prepareHandlerContext(t) + + bktName, srcObjName := "bucket-for-copy", "object-for-copy" + key1 := []byte("firstencriptionkeyofsourceobject") + key1Md5 := md5.Sum(key1) + key2 := []byte("anotherencriptionkeysourceobject") + key2Md5 := md5.Sum(key2) + bktInfo := createTestBucket(tc, bktName) + + srcEnc, err := encryption.NewParams(key1) + require.NoError(t, err) + srcObjInfo := createTestObject(tc, bktInfo, srcObjName, *srcEnc) + require.True(t, containEncryptionMetadataHeaders(srcObjInfo.Headers)) + + dstObjName := "copy-object" + + w, r := prepareTestRequest(tc, bktName, dstObjName, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, bktName+"/"+srcObjName) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerAlgorithm, layer.AESEncryptionAlgorithm) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKey, base64.StdEncoding.EncodeToString(key1)) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKeyMD5, base64.StdEncoding.EncodeToString(key1Md5[:])) + r.Header.Set(api.AmzServerSideEncryptionCustomerAlgorithm, layer.AESEncryptionAlgorithm) + r.Header.Set(api.AmzServerSideEncryptionCustomerKey, base64.StdEncoding.EncodeToString(key2)) + r.Header.Set(api.AmzServerSideEncryptionCustomerKeyMD5, base64.StdEncoding.EncodeToString(key2Md5[:])) + tc.Handler().CopyObjectHandler(w, r) + + assertStatus(t, w, http.StatusOK) + dstObjInfo, err := tc.Layer().GetObjectInfo(tc.Context(), &layer.HeadObjectParams{BktInfo: bktInfo, Object: dstObjName}) + require.NoError(t, err) + require.True(t, containEncryptionMetadataHeaders(dstObjInfo.Headers)) + require.Equal(t, srcObjInfo.Headers[layer.AttributeDecryptedSize], dstObjInfo.Headers[layer.AttributeDecryptedSize]) +} + +func containEncryptionMetadataHeaders(headers map[string]string) bool { + for k := range headers { + if _, ok := layer.EncryptionMetadata[k]; ok { + return true + } + } + return false +} + func copyObject(hc *handlerContext, bktName, fromObject, toObject string, copyMeta CopyMeta, statusCode int) { w, r := prepareTestRequest(hc, bktName, toObject, nil) r.Header.Set(api.AmzCopySource, bktName+"/"+fromObject) diff --git a/api/handler/delete_test.go b/api/handler/delete_test.go index e837cd021..0a4b22fee 100644 --- a/api/handler/delete_test.go +++ b/api/handler/delete_test.go @@ -11,6 +11,7 @@ import ( "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/data" apiErrors "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/errors" + "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer/encryption" apistatus "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/client/status" oid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object/id" "github.com/aws/aws-sdk-go/aws" @@ -427,7 +428,7 @@ func TestDeleteObjectCheckMarkerReturn(t *testing.T) { func createBucketAndObject(tc *handlerContext, bktName, objName string) (*data.BucketInfo, *data.ObjectInfo) { bktInfo := createTestBucket(tc, bktName) - objInfo := createTestObject(tc, bktInfo, objName) + objInfo := createTestObject(tc, bktInfo, objName, encryption.Params{}) return bktInfo, objInfo } @@ -438,7 +439,7 @@ func createVersionedBucketAndObject(t *testing.T, tc *handlerContext, bktName, o require.NoError(t, err) putBucketVersioning(t, tc, bktName, true) - objInfo := createTestObject(tc, bktInfo, objName) + objInfo := createTestObject(tc, bktInfo, objName, encryption.Params{}) return bktInfo, objInfo } diff --git a/api/handler/handlers_test.go b/api/handler/handlers_test.go index fff0ae0e6..667409f9d 100644 --- a/api/handler/handlers_test.go +++ b/api/handler/handlers_test.go @@ -16,6 +16,7 @@ import ( "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/cache" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/data" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer" + "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer/encryption" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/middleware" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/resolver" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/pkg/service/tree" @@ -249,7 +250,7 @@ func createTestBucketWithLock(hc *handlerContext, bktName string, conf *data.Obj return bktInfo } -func createTestObject(hc *handlerContext, bktInfo *data.BucketInfo, objName string) *data.ObjectInfo { +func createTestObject(hc *handlerContext, bktInfo *data.BucketInfo, objName string, encryption encryption.Params) *data.ObjectInfo { content := make([]byte, 1024) _, err := rand.Read(content) require.NoError(hc.t, err) @@ -259,11 +260,12 @@ func createTestObject(hc *handlerContext, bktInfo *data.BucketInfo, objName stri } extObjInfo, err := hc.Layer().PutObject(hc.Context(), &layer.PutObjectParams{ - BktInfo: bktInfo, - Object: objName, - Size: uint64(len(content)), - Reader: bytes.NewReader(content), - Header: header, + BktInfo: bktInfo, + Object: objName, + Size: uint64(len(content)), + Reader: bytes.NewReader(content), + Header: header, + Encryption: encryption, }) require.NoError(hc.t, err) diff --git a/api/handler/locking_test.go b/api/handler/locking_test.go index 7e415a4e4..d4bb13eb4 100644 --- a/api/handler/locking_test.go +++ b/api/handler/locking_test.go @@ -13,6 +13,7 @@ import ( "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/data" apiErrors "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/errors" + "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer/encryption" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/middleware" "github.com/stretchr/testify/require" ) @@ -426,7 +427,7 @@ func TestObjectLegalHold(t *testing.T) { bktInfo := createTestBucketWithLock(hc, bktName, nil) objName := "obj-for-legal-hold" - createTestObject(hc, bktInfo, objName) + createTestObject(hc, bktInfo, objName, encryption.Params{}) getObjectLegalHold(hc, bktName, objName, legalHoldOff) @@ -470,7 +471,7 @@ func TestObjectRetention(t *testing.T) { bktInfo := createTestBucketWithLock(hc, bktName, nil) objName := "obj-for-retention" - createTestObject(hc, bktInfo, objName) + createTestObject(hc, bktInfo, objName, encryption.Params{}) getObjectRetention(hc, bktName, objName, nil, apiErrors.ErrNoSuchKey) diff --git a/api/handler/multipart_upload.go b/api/handler/multipart_upload.go index 56742d948..4f7fd0c7a 100644 --- a/api/handler/multipart_upload.go +++ b/api/handler/multipart_upload.go @@ -345,6 +345,17 @@ func (h *handler) UploadPartCopy(w http.ResponseWriter, r *http.Request) { return } + srcEncryptionParams, err := formCopySourceEncryptionParams(r) + if err != nil { + h.logAndSendError(w, "invalid sse headers", reqInfo, err) + return + } + + if err = srcEncryptionParams.MatchObjectEncryption(layer.FormEncryptionInfo(srcInfo.Headers)); err != nil { + h.logAndSendError(w, "encryption doesn't match object", reqInfo, fmt.Errorf("%w: %s", errors.GetAPIError(errors.ErrBadRequest), err), additional...) + return + } + p := &layer.UploadCopyParams{ Versioned: headPrm.Versioned(), Info: &layer.UploadInfoParams{ @@ -352,10 +363,11 @@ func (h *handler) UploadPartCopy(w http.ResponseWriter, r *http.Request) { Bkt: bktInfo, Key: reqInfo.ObjectName, }, - SrcObjInfo: srcInfo, - SrcBktInfo: srcBktInfo, - PartNumber: partNumber, - Range: srcRange, + SrcObjInfo: srcInfo, + SrcBktInfo: srcBktInfo, + SrcEncryption: srcEncryptionParams, + PartNumber: partNumber, + Range: srcRange, } p.Info.Encryption, err = formEncryptionParams(r) @@ -364,11 +376,6 @@ func (h *handler) UploadPartCopy(w http.ResponseWriter, r *http.Request) { return } - if err = p.Info.Encryption.MatchObjectEncryption(layer.FormEncryptionInfo(srcInfo.Headers)); err != nil { - h.logAndSendError(w, "encryption doesn't match object", reqInfo, fmt.Errorf("%w: %s", errors.GetAPIError(errors.ErrBadRequest), err), additional...) - return - } - info, err := h.obj.UploadPartCopy(ctx, p) if err != nil { h.logAndSendError(w, "could not upload part copy", reqInfo, err, additional...) diff --git a/api/handler/multipart_upload_test.go b/api/handler/multipart_upload_test.go index dbf06b209..0fcb6dadf 100644 --- a/api/handler/multipart_upload_test.go +++ b/api/handler/multipart_upload_test.go @@ -2,6 +2,8 @@ package handler import ( "crypto/md5" + "crypto/tls" + "encoding/base64" "encoding/hex" "encoding/xml" "fmt" @@ -13,6 +15,7 @@ import ( "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api" s3Errors "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/errors" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer" + "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer/encryption" "github.com/stretchr/testify/require" ) @@ -183,6 +186,51 @@ func TestMultipartUploadSize(t *testing.T) { uploadPartCopy(hc, bktName, objName2, uploadInfo.UploadID, 1, sourceCopy, 0, 0) uploadPartCopy(hc, bktName, objName2, uploadInfo.UploadID, 2, sourceCopy, 0, partSize) }) + + t.Run("check correct size when copy part from encrypted source", func(t *testing.T) { + newBucket, newObjName := "new-bucket", "new-object-multipart" + bktInfo := createTestBucket(hc, newBucket) + + srcObjName := "source-object" + key := []byte("firstencriptionkeyofsourceobject") + keyMd5 := md5.Sum(key) + srcEnc, err := encryption.NewParams(key) + require.NoError(t, err) + srcObjInfo := createTestObject(hc, bktInfo, srcObjName, *srcEnc) + + multipartInfo := createMultipartUpload(hc, newBucket, newObjName, headers) + + sourceCopy := newBucket + "/" + srcObjName + + query := make(url.Values) + query.Set(uploadIDQuery, multipartInfo.UploadID) + query.Set(partNumberQuery, "1") + + // empty copy-source-sse headers + w, r := prepareTestRequestWithQuery(hc, newBucket, newObjName, query, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, sourceCopy) + hc.Handler().UploadPartCopy(w, r) + + assertStatus(t, w, http.StatusBadRequest) + + // success copy + w, r = prepareTestRequestWithQuery(hc, newBucket, newObjName, query, nil) + r.TLS = &tls.ConnectionState{} + r.Header.Set(api.AmzCopySource, sourceCopy) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerAlgorithm, layer.AESEncryptionAlgorithm) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKey, base64.StdEncoding.EncodeToString(key)) + r.Header.Set(api.AmzCopySourceServerSideEncryptionCustomerKeyMD5, base64.StdEncoding.EncodeToString(keyMd5[:])) + hc.Handler().UploadPartCopy(w, r) + + uploadPartCopyResponse := &UploadPartCopyResponse{} + readResponse(hc.t, w, http.StatusOK, uploadPartCopyResponse) + + completeMultipartUpload(hc, newBucket, newObjName, multipartInfo.UploadID, []string{uploadPartCopyResponse.ETag}) + attr := getObjectAttributes(hc, newBucket, newObjName, objectParts) + require.Equal(t, 1, attr.ObjectParts.PartsCount) + require.Equal(t, srcObjInfo.Headers[layer.AttributeDecryptedSize], strconv.Itoa(attr.ObjectParts.Parts[0].Size)) + }) } func TestListParts(t *testing.T) { diff --git a/api/handler/object_list_test.go b/api/handler/object_list_test.go index c899511a1..e8e1caed8 100644 --- a/api/handler/object_list_test.go +++ b/api/handler/object_list_test.go @@ -8,6 +8,7 @@ import ( "testing" "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/data" + "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer/encryption" "github.com/stretchr/testify/require" ) @@ -95,7 +96,7 @@ func TestS3CompatibilityBucketListV2BothContinuationTokenStartAfter(t *testing.T bktInfo, _ := createBucketAndObject(tc, bktName, objects[0]) for _, objName := range objects[1:] { - createTestObject(tc, bktInfo, objName) + createTestObject(tc, bktInfo, objName, encryption.Params{}) } listV2Response1 := listObjectsV2(tc, bktName, "", "", "bar", "", 1) @@ -120,7 +121,7 @@ func TestS3BucketListV2EncodingBasic(t *testing.T) { objects := []string{"foo+1/bar", "foo/bar/xyzzy", "quux ab/thud", "asdf+b"} for _, objName := range objects { - createTestObject(hc, bktInfo, objName) + createTestObject(hc, bktInfo, objName, encryption.Params{}) } query := make(url.Values) @@ -150,7 +151,7 @@ func TestS3BucketListDelimiterBasic(t *testing.T) { bktInfo, _ := createBucketAndObject(tc, bktName, objects[0]) for _, objName := range objects[1:] { - createTestObject(tc, bktInfo, objName) + createTestObject(tc, bktInfo, objName, encryption.Params{}) } listV1Response := listObjectsV1(tc, bktName, "", "/", "", -1) @@ -169,7 +170,7 @@ func TestS3BucketListV2DelimiterPercentage(t *testing.T) { bktInfo, _ := createBucketAndObject(tc, bktName, objects[0]) for _, objName := range objects[1:] { - createTestObject(tc, bktInfo, objName) + createTestObject(tc, bktInfo, objName, encryption.Params{}) } listV2Response := listObjectsV2(tc, bktName, "", "%", "", "", -1) @@ -189,7 +190,7 @@ func TestS3BucketListV2DelimiterPrefix(t *testing.T) { bktInfo, _ := createBucketAndObject(tc, bktName, objects[0]) for _, objName := range objects[1:] { - createTestObject(tc, bktInfo, objName) + createTestObject(tc, bktInfo, objName, encryption.Params{}) } var empty []string diff --git a/api/handler/put.go b/api/handler/put.go index df7441217..c69054cb5 100644 --- a/api/handler/put.go +++ b/api/handler/put.go @@ -6,7 +6,6 @@ import ( "encoding/base64" "encoding/json" "encoding/xml" - errorsStd "errors" "fmt" "io" "net" @@ -376,16 +375,38 @@ func (h *handler) getBodyReader(r *http.Request) (io.ReadCloser, error) { } func formEncryptionParams(r *http.Request) (enc encryption.Params, err error) { - sseCustomerAlgorithm := r.Header.Get(api.AmzServerSideEncryptionCustomerAlgorithm) - sseCustomerKey := r.Header.Get(api.AmzServerSideEncryptionCustomerKey) - sseCustomerKeyMD5 := r.Header.Get(api.AmzServerSideEncryptionCustomerKeyMD5) + return formEncryptionParamsBase(r, false) +} + +func formCopySourceEncryptionParams(r *http.Request) (enc encryption.Params, err error) { + return formEncryptionParamsBase(r, true) +} + +func formEncryptionParamsBase(r *http.Request, isCopySource bool) (enc encryption.Params, err error) { + var sseCustomerAlgorithm, sseCustomerKey, sseCustomerKeyMD5 string + if isCopySource { + sseCustomerAlgorithm = r.Header.Get(api.AmzCopySourceServerSideEncryptionCustomerAlgorithm) + sseCustomerKey = r.Header.Get(api.AmzCopySourceServerSideEncryptionCustomerKey) + sseCustomerKeyMD5 = r.Header.Get(api.AmzCopySourceServerSideEncryptionCustomerKeyMD5) + } else { + sseCustomerAlgorithm = r.Header.Get(api.AmzServerSideEncryptionCustomerAlgorithm) + sseCustomerKey = r.Header.Get(api.AmzServerSideEncryptionCustomerKey) + sseCustomerKeyMD5 = r.Header.Get(api.AmzServerSideEncryptionCustomerKeyMD5) + } if len(sseCustomerAlgorithm) == 0 && len(sseCustomerKey) == 0 && len(sseCustomerKeyMD5) == 0 { return } if r.TLS == nil { - return enc, errorsStd.New("encryption available only when TLS is enabled") + return enc, errors.GetAPIError(errors.ErrInsecureSSECustomerRequest) + } + + if len(sseCustomerKey) > 0 && len(sseCustomerAlgorithm) == 0 { + return enc, errors.GetAPIError(errors.ErrMissingSSECustomerAlgorithm) + } + if len(sseCustomerAlgorithm) > 0 && len(sseCustomerKey) == 0 { + return enc, errors.GetAPIError(errors.ErrMissingSSECustomerKey) } if sseCustomerAlgorithm != layer.AESEncryptionAlgorithm { @@ -394,10 +415,16 @@ func formEncryptionParams(r *http.Request) (enc encryption.Params, err error) { key, err := base64.StdEncoding.DecodeString(sseCustomerKey) if err != nil { + if isCopySource { + return enc, errors.GetAPIError(errors.ErrInvalidSSECustomerParameters) + } return enc, errors.GetAPIError(errors.ErrInvalidSSECustomerKey) } if len(key) != layer.AESKeySize { + if isCopySource { + return enc, errors.GetAPIError(errors.ErrInvalidSSECustomerParameters) + } return enc, errors.GetAPIError(errors.ErrInvalidSSECustomerKey) } diff --git a/api/headers.go b/api/headers.go index 0a51e641b..a91149cfa 100644 --- a/api/headers.go +++ b/api/headers.go @@ -67,6 +67,10 @@ const ( AmzServerSideEncryptionCustomerKey = "x-amz-server-side-encryption-customer-key" AmzServerSideEncryptionCustomerKeyMD5 = "x-amz-server-side-encryption-customer-key-MD5" + AmzCopySourceServerSideEncryptionCustomerAlgorithm = "x-amz-copy-source-server-side-encryption-customer-algorithm" + AmzCopySourceServerSideEncryptionCustomerKey = "x-amz-copy-source-server-side-encryption-customer-key" + AmzCopySourceServerSideEncryptionCustomerKeyMD5 = "x-amz-copy-source-server-side-encryption-customer-key-MD5" + OwnerID = "X-Owner-Id" ContainerID = "X-Container-Id" ContainerName = "X-Container-Name" diff --git a/api/layer/encryption/encryption.go b/api/layer/encryption/encryption.go index 1b71b730f..380c289ce 100644 --- a/api/layer/encryption/encryption.go +++ b/api/layer/encryption/encryption.go @@ -10,6 +10,7 @@ import ( "fmt" "io" + "git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/errors" "github.com/minio/sio" ) @@ -100,8 +101,11 @@ func (p Params) HMAC() ([]byte, []byte, error) { // MatchObjectEncryption checks if encryption params are valid for provided object. func (p Params) MatchObjectEncryption(encInfo ObjectEncryption) error { - if p.Enabled() != encInfo.Enabled { - return errorsStd.New("invalid encryption view") + if p.Enabled() && !encInfo.Enabled { + return errors.GetAPIError(errors.ErrInvalidEncryptionParameters) + } + if !p.Enabled() && encInfo.Enabled { + return errors.GetAPIError(errors.ErrSSEEncryptedObject) } if !encInfo.Enabled { @@ -122,7 +126,7 @@ func (p Params) MatchObjectEncryption(encInfo ObjectEncryption) error { mac.Write(hmacSalt) expectedHmacKey := mac.Sum(nil) if !bytes.Equal(expectedHmacKey, hmacKey) { - return errorsStd.New("mismatched hmac key") + return errors.GetAPIError(errors.ErrInvalidSSECustomerParameters) } return nil diff --git a/api/layer/layer.go b/api/layer/layer.go index 63a4d93ed..126523a79 100644 --- a/api/layer/layer.go +++ b/api/layer/layer.go @@ -160,11 +160,12 @@ type ( ScrBktInfo *data.BucketInfo DstBktInfo *data.BucketInfo DstObject string - SrcSize uint64 + DstSize uint64 Header map[string]string Range *RangeParams Lock *data.ObjectLock - Encryption encryption.Params + SrcEncryption encryption.Params + DstEncryption encryption.Params CopiesNumbers []uint32 } // CreateBucketParams stores bucket create request parameters. @@ -291,6 +292,13 @@ const ( AttributeFrostfsCopiesNumber = "frostfs-copies-number" // such format to match X-Amz-Meta-Frostfs-Copies-Number header ) +var EncryptionMetadata = map[string]struct{}{ + AttributeEncryptionAlgorithm: {}, + AttributeDecryptedSize: {}, + AttributeHMACSalt: {}, + AttributeHMACKey: {}, +} + func (t *VersionedObject) String() string { return t.Name + ":" + t.VersionID } @@ -583,7 +591,7 @@ func (n *layer) CopyObject(ctx context.Context, p *CopyObjectParams) (*data.Exte Versioned: p.SrcVersioned, Range: p.Range, BucketInfo: p.ScrBktInfo, - Encryption: p.Encryption, + Encryption: p.SrcEncryption, }) if err != nil { return nil, fmt.Errorf("get object to copy: %w", err) @@ -592,10 +600,10 @@ func (n *layer) CopyObject(ctx context.Context, p *CopyObjectParams) (*data.Exte return n.PutObject(ctx, &PutObjectParams{ BktInfo: p.DstBktInfo, Object: p.DstObject, - Size: p.SrcSize, + Size: p.DstSize, Reader: objPayload, Header: p.Header, - Encryption: p.Encryption, + Encryption: p.DstEncryption, CopiesNumbers: p.CopiesNumbers, }) } diff --git a/api/layer/multipart_upload.go b/api/layer/multipart_upload.go index 0ebdc59f0..35e4b8320 100644 --- a/api/layer/multipart_upload.go +++ b/api/layer/multipart_upload.go @@ -74,12 +74,13 @@ type ( } UploadCopyParams struct { - Versioned bool - Info *UploadInfoParams - SrcObjInfo *data.ObjectInfo - SrcBktInfo *data.BucketInfo - PartNumber int - Range *RangeParams + Versioned bool + Info *UploadInfoParams + SrcObjInfo *data.ObjectInfo + SrcBktInfo *data.BucketInfo + SrcEncryption encryption.Params + PartNumber int + Range *RangeParams } CompleteMultipartParams struct { @@ -316,6 +317,7 @@ func (n *layer) UploadPartCopy(ctx context.Context, p *UploadCopyParams) (*data. if objSize, err := GetObjectSize(p.SrcObjInfo); err == nil { srcObjectSize = objSize + size = objSize } if p.Range != nil { @@ -333,6 +335,7 @@ func (n *layer) UploadPartCopy(ctx context.Context, p *UploadCopyParams) (*data. Versioned: p.Versioned, Range: p.Range, BucketInfo: p.SrcBktInfo, + Encryption: p.SrcEncryption, }) if err != nil { return nil, fmt.Errorf("get object to upload copy: %w", err)