From f293df647cba26d1d1dde39a721f42aeeb838937 Mon Sep 17 00:00:00 2001 From: Harshavardhana Date: Tue, 21 Jun 2022 14:11:12 -0700 Subject: [PATCH] s3/zip: extract metadata properly for Zipped objects (#15123) s3/zip: extra metadata properly for Zipped objects fixes #15121 --- cmd/object-api-datatypes.go | 21 ++++++++++++ cmd/s3-zip-handlers.go | 65 +++++++++++++++++-------------------- 2 files changed, 51 insertions(+), 35 deletions(-) diff --git a/cmd/object-api-datatypes.go b/cmd/object-api-datatypes.go index 6bf4bfd45..54a4fa73a 100644 --- a/cmd/object-api-datatypes.go +++ b/cmd/object-api-datatypes.go @@ -18,6 +18,7 @@ package cmd import ( + "encoding/base64" "io" "math" "time" @@ -178,6 +179,26 @@ type ObjectInfo struct { SuccessorModTime time.Time } +// ArchiveInfo returns any saved zip archive meta information +func (o ObjectInfo) ArchiveInfo() []byte { + if len(o.UserDefined) == 0 { + return nil + } + z, ok := o.UserDefined[archiveInfoMetadataKey] + if !ok { + return nil + } + if len(z) > 0 && z[0] >= 32 { + // FS/gateway mode does base64 encoding on roundtrip. + // zipindex has version as first byte, which is below any base64 value. + zipInfo, _ := base64.StdEncoding.DecodeString(z) + if len(zipInfo) != 0 { + return zipInfo + } + } + return []byte(z) +} + // Clone - Returns a cloned copy of current objectInfo func (o ObjectInfo) Clone() (cinfo ObjectInfo) { cinfo = ObjectInfo{ diff --git a/cmd/s3-zip-handlers.go b/cmd/s3-zip-handlers.go index c82656039..8b5dbfc9c 100644 --- a/cmd/s3-zip-handlers.go +++ b/cmd/s3-zip-handlers.go @@ -140,21 +140,10 @@ func (api objectAPIHandlers) getObjectInArchiveFileHandler(ctx context.Context, return } - var zipInfo []byte - - if z, ok := zipObjInfo.UserDefined[archiveInfoMetadataKey]; ok { - if globalIsErasure { - zipInfo = []byte(z) - } else { - zipInfo, err = base64.StdEncoding.DecodeString(z) - logger.LogIf(ctx, err) - // Will attempt to re-read... - } - } + zipInfo := zipObjInfo.ArchiveInfo() if len(zipInfo) == 0 { zipInfo, err = updateObjectMetadataWithZipInfo(ctx, objectAPI, bucket, zipPath, opts) } - if err != nil { writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL) return @@ -247,15 +236,11 @@ func listObjectsV2InArchive(ctx context.Context, objectAPI ObjectLayer, bucket, return ListObjectsV2Info{}, nil } - var zipInfo []byte - - if z, ok := zipObjInfo.UserDefined[archiveInfoMetadataKey]; ok { - zipInfo = []byte(z) - } else { + zipInfo := zipObjInfo.ArchiveInfo() + if len(zipInfo) == 0 { // Always update the latest version zipInfo, err = updateObjectMetadataWithZipInfo(ctx, objectAPI, bucket, zipPath, ObjectOptions{}) } - if err != nil { return ListObjectsV2Info{}, err } @@ -332,11 +317,10 @@ func getFilesListFromZIPObject(ctx context.Context, objectAPI ObjectLayer, bucke return nil, ObjectInfo{}, err } b, err := ioutil.ReadAll(gr) + gr.Close() if err != nil { - gr.Close() return nil, ObjectInfo{}, err } - gr.Close() if size > len(b) { size = len(b) } @@ -439,11 +423,8 @@ func (api objectAPIHandlers) headObjectInArchiveFileHandler(ctx context.Context, return } - var zipInfo []byte - - if z, ok := zipObjInfo.UserDefined[archiveInfoMetadataKey]; ok { - zipInfo = []byte(z) - } else { + zipInfo := zipObjInfo.ArchiveInfo() + if len(zipInfo) == 0 { zipInfo, err = updateObjectMetadataWithZipInfo(ctx, objectAPI, bucket, zipPath, opts) } if err != nil { @@ -498,20 +479,34 @@ func updateObjectMetadataWithZipInfo(ctx context.Context, objectAPI ObjectLayer, } srcInfo.UserDefined[archiveTypeMetadataKey] = archiveType - if globalIsErasure { - srcInfo.UserDefined[archiveInfoMetadataKey] = string(zipInfo) + var zipInfoStr string + if globalIsGateway { + zipInfoStr = base64.StdEncoding.EncodeToString(zipInfo) } else { - srcInfo.UserDefined[archiveInfoMetadataKey] = base64.StdEncoding.EncodeToString(zipInfo) + zipInfoStr = string(zipInfo) } - srcInfo.metadataOnly = true - // Always update the same version id & modtime + if globalIsGateway { + srcInfo.UserDefined[archiveInfoMetadataKey] = zipInfoStr - // Passing opts twice as source & destination options will update the metadata - // of the same object version to avoid creating a new version. - _, err = objectAPI.CopyObject(ctx, bucket, object, bucket, object, srcInfo, opts, opts) - if err != nil { - return nil, err + // Use CopyObject API only for Gateway mode. + if _, err = objectAPI.CopyObject(ctx, bucket, object, bucket, object, srcInfo, opts, opts); err != nil { + return nil, err + } + } else { + popts := ObjectOptions{ + MTime: srcInfo.ModTime, + VersionID: srcInfo.VersionID, + EvalMetadataFn: func(oi ObjectInfo) error { + oi.UserDefined[archiveTypeMetadataKey] = archiveType + oi.UserDefined[archiveInfoMetadataKey] = zipInfoStr + return nil + }, + } + // For all other modes use in-place update to update metadata on a specific version. + if _, err = objectAPI.PutObjectMetadata(ctx, bucket, object, popts); err != nil { + return nil, err + } } return zipInfo, nil