create subcomposed objects if total parts are > 32 (#4593)

This commit is contained in:
Bala FA 2017-06-28 10:57:05 +05:30 committed by Dee Koder
parent 1af331c05c
commit 53e7fdc847

View File

@ -25,6 +25,7 @@ import (
"fmt" "fmt"
"hash" "hash"
"io" "io"
"math"
"regexp" "regexp"
"strings" "strings"
"time" "time"
@ -56,6 +57,13 @@ const (
// token prefixed with GCS returned marker to differentiate // token prefixed with GCS returned marker to differentiate
// from user supplied marker. // from user supplied marker.
gcsTokenPrefix = "##minio" gcsTokenPrefix = "##minio"
// maxComponents - maximum component object count to create a composite object.
// Refer https://cloud.google.com/storage/docs/composite-objects
maxComponents = 32
// maxPartCount - maximum multipart parts GCS supports which is 32 x 32 = 1024.
maxPartCount = 1024
) )
// Stored in gcs.json - Contents of this file is not used anywhere. It can be // Stored in gcs.json - Contents of this file is not used anywhere. It can be
@ -899,14 +907,39 @@ func (l *gcsGateway) CompleteMultipartUpload(bucket string, key string, uploadID
parts[i] = l.client.Bucket(bucket).Object(gcsMultipartDataName(uploadID, uploadedPart.ETag)) parts[i] = l.client.Bucket(bucket).Object(gcsMultipartDataName(uploadID, uploadedPart.ETag))
} }
if len(parts) > 32 { if len(parts) > maxPartCount {
// we need to split up the compose of more than 32 parts
// into subcomposes. This means that the first 32 parts will
// compose to a composed-object-0, next parts to composed-object-1,
// the final compose will compose composed-object* to 1.
return ObjectInfo{}, traceError(NotSupported{}) return ObjectInfo{}, traceError(NotSupported{})
} }
composeCount := int(math.Ceil(float64(len(parts)) / float64(maxComponents)))
if composeCount > 1 {
// Create composes of every 32 parts.
composeParts := make([]*storage.ObjectHandle, composeCount)
for i := 0; i < composeCount; i++ {
// Create 'composed-object-N' using next 32 parts.
composeName := fmt.Sprintf("composed-object-%d", i)
composeParts[i] = l.client.Bucket(bucket).Object(gcsMultipartDataName(uploadID, composeName))
start := i * maxComponents
end := start + maxComponents
if end > len(parts) {
end = len(parts)
}
composer := composeParts[i].ComposerFrom(parts[start:end]...)
composer.ContentType = partZeroAttrs.ContentType
composer.Metadata = partZeroAttrs.Metadata
_, err = composer.Run(l.ctx)
if err != nil {
return ObjectInfo{}, gcsToObjectError(traceError(err), bucket, key)
}
}
// As composes are successfully created, final object needs to be created using composes.
parts = composeParts
}
dst := l.client.Bucket(bucket).Object(key) dst := l.client.Bucket(bucket).Object(key)
composer := dst.ComposerFrom(parts...) composer := dst.ComposerFrom(parts...)