Cancel PutObjectPart on upload abort (#7940)

Calling ListMultipartUploads fails if an upload is aborted while a
part is being uploaded because the directory for the upload exists
(since fsRenameFile ends up calling os.MkdirAll) but the meta JSON file
doesn't. To fix this we make sure an upload hasn't been aborted during
PutObjectPart by checking the existence of the directory for the upload
while moving the temporary part file into it.
master
Daryl Finlay 5 years ago committed by Harshavardhana
parent 87e6533cf3
commit 9389a55e5d
  1. 22
      cmd/fs-v1-helpers.go
  2. 8
      cmd/fs-v1-multipart.go
  3. 6
      cmd/os-reliable.go

@ -280,7 +280,7 @@ func fsOpenFile(ctx context.Context, readPath string, offset int64) (io.ReadClos
} }
// Stat to get the size of the file at path. // Stat to get the size of the file at path.
st, err := os.Stat(readPath) st, err := fr.Stat()
if err != nil { if err != nil {
err = osErrToFSFileErr(err) err = osErrToFSFileErr(err)
if err != errFileNotFound { if err != errFileNotFound {
@ -386,6 +386,26 @@ func fsFAllocate(fd int, offset int64, len int64) (err error) {
return nil return nil
} }
// Renames source path to destination path, fails if the destination path
// parents are not already created.
func fsSimpleRenameFile(ctx context.Context, sourcePath, destPath string) error {
if err := checkPathLength(sourcePath); err != nil {
logger.LogIf(ctx, err)
return err
}
if err := checkPathLength(destPath); err != nil {
logger.LogIf(ctx, err)
return err
}
if err := os.Rename(sourcePath, destPath); err != nil {
logger.LogIf(ctx, err)
return osErrToFSFileErr(err)
}
return nil
}
// Renames source path to destination path, creates all the // Renames source path to destination path, creates all the
// missing parents if they don't exist. // missing parents if they don't exist.
func fsRenameFile(ctx context.Context, sourcePath, destPath string) error { func fsRenameFile(ctx context.Context, sourcePath, destPath string) error {

@ -152,7 +152,7 @@ func (fs *FSObjects) ListMultipartUploads(ctx context.Context, bucket, object, k
return result, toObjectErr(err) return result, toObjectErr(err)
} }
// S3 spec says uploaIDs should be sorted based on initiated time. ModTime of fs.json // S3 spec says uploadIDs should be sorted based on initiated time. ModTime of fs.json
// is the creation time of the uploadID, hence we will use that. // is the creation time of the uploadID, hence we will use that.
var uploads []MultipartInfo var uploads []MultipartInfo
for _, uploadID := range uploadIDs { for _, uploadID := range uploadIDs {
@ -326,7 +326,11 @@ func (fs *FSObjects) PutObjectPart(ctx context.Context, bucket, object, uploadID
partPath := pathJoin(uploadIDDir, fs.encodePartFile(partID, etag, data.ActualSize())) partPath := pathJoin(uploadIDDir, fs.encodePartFile(partID, etag, data.ActualSize()))
if err = fsRenameFile(ctx, tmpPartPath, partPath); err != nil { // Make sure not to create parent directories if they don't exist - the upload might have been aborted.
if err = fsSimpleRenameFile(ctx, tmpPartPath, partPath); err != nil {
if err == errFileNotFound || err == errFileAccessDenied {
return pi, InvalidUploadID{UploadID: uploadID}
}
return pi, toObjectErr(err, minioMetaMultipartBucket, partPath) return pi, toObjectErr(err, minioMetaMultipartBucket, partPath)
} }

@ -156,11 +156,11 @@ func renameAll(srcFilePath, dstFilePath string) (err error) {
// Reliably retries os.RenameAll if for some reason os.RenameAll returns // Reliably retries os.RenameAll if for some reason os.RenameAll returns
// syscall.ENOENT (parent does not exist). // syscall.ENOENT (parent does not exist).
func reliableRename(srcFilePath, dstFilePath string) (err error) { func reliableRename(srcFilePath, dstFilePath string) (err error) {
if err = reliableMkdirAll(path.Dir(dstFilePath), 0777); err != nil {
return err
}
i := 0 i := 0
for { for {
if err = reliableMkdirAll(path.Dir(dstFilePath), 0777); err != nil {
return err
}
// After a successful parent directory create attempt a renameAll. // After a successful parent directory create attempt a renameAll.
if err = os.Rename(srcFilePath, dstFilePath); err != nil { if err = os.Rename(srcFilePath, dstFilePath); err != nil {
// Retry only for the first retryable error. // Retry only for the first retryable error.

Loading…
Cancel
Save