@ -4,9 +4,6 @@ import (
"encoding/hex"
"encoding/hex"
"encoding/xml"
"encoding/xml"
"fmt"
"fmt"
"github.com/google/uuid"
"github.com/seaweedfs/seaweedfs/weed/s3api/s3err"
"golang.org/x/exp/slices"
"math"
"math"
"path/filepath"
"path/filepath"
"sort"
"sort"
@ -14,6 +11,10 @@ import (
"strings"
"strings"
"time"
"time"
"github.com/google/uuid"
"github.com/seaweedfs/seaweedfs/weed/s3api/s3err"
"golang.org/x/exp/slices"
"github.com/aws/aws-sdk-go/aws"
"github.com/aws/aws-sdk-go/aws"
"github.com/aws/aws-sdk-go/service/s3"
"github.com/aws/aws-sdk-go/service/s3"
@ -90,40 +91,55 @@ func (s3a *S3ApiServer) completeMultipartUpload(input *s3.CompleteMultipartUploa
return nil , s3err . ErrNoSuchUpload
return nil , s3err . ErrNoSuchUpload
}
}
// check whether completedParts is more than received parts
{
partNumbers := make ( map [ int ] struct { } , len ( entries ) )
partEntries := make ( map [ int ] [ ] * filer_pb . Entry , len ( entries ) )
for _ , entry := range entries {
for _ , entry := range entries {
glog . V ( 4 ) . Infof ( "completeMultipartUpload part entries %s" , entry . Name )
if strings . HasSuffix ( entry . Name , ".part" ) && ! entry . IsDirectory {
if strings . HasSuffix ( entry . Name , ".part" ) && ! entry . IsDirectory {
partNumberString := entry . Name [ : len ( entry . Name ) - len ( ".part" ) ]
partNumber , err := strconv . Atoi ( partNumberString )
if err == nil {
partNumbers [ partNumber ] = struct { } { }
}
}
var partNumberString string
index := strings . Index ( entry . Name , "_" )
if index != - 1 {
partNumberString = entry . Name [ : index ]
} else {
partNumberString = entry . Name [ : len ( entry . Name ) - len ( ".part" ) ]
}
}
for _ , part := range completedParts {
if _ , found := partNumbers [ part . PartNumber ] ; ! found {
return nil , s3err . ErrInvalidPart
partNumber , err := strconv . Atoi ( partNumberString )
if err != nil {
glog . Errorf ( "completeMultipartUpload failed to pasre partNumber %s:%s" , partNumberString , err )
continue
}
}
//there maybe multi same part, because of client retry
partEntries [ partNumber ] = append ( partEntries [ partNumber ] , entry )
}
}
}
}
mime := pentry . Attributes . Mime
mime := pentry . Attributes . Mime
var finalParts [ ] * filer_pb . FileChunk
var finalParts [ ] * filer_pb . FileChunk
var offset int64
var offset int64
var deleteEntries [ ] * filer_pb . Entry
for _ , part := range completedParts {
entries := partEntries [ part . PartNumber ]
// check whether completedParts is more than received parts
if len ( entries ) == 0 {
glog . Errorf ( "part %d has no entry" , part . PartNumber )
return nil , s3err . ErrInvalidPart
}
found := false
for _ , entry := range entries {
for _ , entry := range entries {
if strings . HasSuffix ( entry . Name , ".part" ) && ! entry . IsDirectory {
partETag , found := findByPartNumber ( entry . Name , completedParts )
if ! found {
if found {
deleteEntries = append ( deleteEntries , entry )
continue
continue
}
}
partETag := strings . Trim ( part . ETag , ` " ` )
entryETag := hex . EncodeToString ( entry . Attributes . GetMd5 ( ) )
entryETag := hex . EncodeToString ( entry . Attributes . GetMd5 ( ) )
glog . Warningf ( "complete etag %s, partEtag %s" , partETag , entryETag )
if partETag != "" && len ( partETag ) == 32 && entryETag != "" && entryETag != partETag {
if partETag != "" && len ( partETag ) == 32 && entryETag != "" && entryETag != partETag {
glog . Errorf ( "completeMultipartUpload %s ETag mismatch chunk: %s part: %s" , entry . Name , entryETag , partETag )
return nil , s3err . ErrInvalidPart
err = fmt . Errorf ( "completeMultipartUpload %s ETag mismatch chunk: %s part: %s" , entry . Name , entryETag , partETag )
deleteEntries = append ( deleteEntries , entry )
continue
}
}
for _ , chunk := range entry . GetChunks ( ) {
for _ , chunk := range entry . GetChunks ( ) {
p := & filer_pb . FileChunk {
p := & filer_pb . FileChunk {
@ -137,7 +153,14 @@ func (s3a *S3ApiServer) completeMultipartUpload(input *s3.CompleteMultipartUploa
finalParts = append ( finalParts , p )
finalParts = append ( finalParts , p )
offset += int64 ( chunk . Size )
offset += int64 ( chunk . Size )
}
}
found = true
err = nil
}
if err != nil {
glog . Errorf ( "%s" , err )
return nil , s3err . ErrInvalidPart
}
}
}
}
entryName := filepath . Base ( * input . Key )
entryName := filepath . Base ( * input . Key )
@ -186,6 +209,13 @@ func (s3a *S3ApiServer) completeMultipartUpload(input *s3.CompleteMultipartUploa
} ,
} ,
}
}
for _ , deleteEntry := range deleteEntries {
//delete unused part data
glog . Infof ( "completeMultipartUpload cleanup %s upload %s unused %s" , * input . Bucket , * input . UploadId , deleteEntry . Name )
if err = s3a . rm ( uploadDirectory , deleteEntry . Name , true , true ) ; err != nil {
glog . Warningf ( "completeMultipartUpload cleanup %s upload %s unused %s : %v" , * input . Bucket , * input . UploadId , deleteEntry . Name , err )
}
}
if err = s3a . rm ( s3a . genUploadsFolder ( * input . Bucket ) , * input . UploadId , false , true ) ; err != nil {
if err = s3a . rm ( s3a . genUploadsFolder ( * input . Bucket ) , * input . UploadId , false , true ) ; err != nil {
glog . V ( 1 ) . Infof ( "completeMultipartUpload cleanup %s upload %s: %v" , * input . Bucket , * input . UploadId , err )
glog . V ( 1 ) . Infof ( "completeMultipartUpload cleanup %s upload %s: %v" , * input . Bucket , * input . UploadId , err )
}
}