From b2d9710debeb6b9ba5a733d07c02194206d8ea5e Mon Sep 17 00:00:00 2001 From: poornas Date: Sun, 13 Jan 2019 08:17:33 -0800 Subject: [PATCH] Fix PutObjectAsync handling of unknown size streams (#271) Handle zero byte streams passed to PutObjectAsync with size -1 (unknown size stream) by uploading a single part with 0 bytes before completing the multipart upload. --- Minio.Functional.Tests/FunctionalTest.cs | 41 +++++++++++++++++++++++- Minio/ApiEndpoints/ObjectOperations.cs | 2 +- 2 files changed, 41 insertions(+), 2 deletions(-) diff --git a/Minio.Functional.Tests/FunctionalTest.cs b/Minio.Functional.Tests/FunctionalTest.cs index 39312818a..c905b0beb 100644 --- a/Minio.Functional.Tests/FunctionalTest.cs +++ b/Minio.Functional.Tests/FunctionalTest.cs @@ -193,6 +193,7 @@ public static void Main(string[] args) PutObject_Test5(minioClient).Wait(); PutObject_Test6(minioClient).Wait(); PutObject_Test7(minioClient).Wait(); + PutObject_Test8(minioClient).Wait(); // Test StatObject function StatObject_Test1(minioClient).Wait(); @@ -662,7 +663,7 @@ private async static Task PutObject_Test7(MinioClient minio) {"objectName",objectName}, {"contentType", contentType}, {"data","1MB"}, - {"size","1MB"}, + {"size","-1"}, }; try { @@ -689,6 +690,44 @@ await minio.PutObjectAsync(bucketName, new MintLogger("PutObject_Test7",putObjectSignature1,"Tests whether PutObject with unknown stream-size passes",TestStatus.FAIL,(DateTime.Now - startTime),"",ex.Message, ex.ToString(), args).Log(); } } + private async static Task PutObject_Test8(MinioClient minio) + { + DateTime startTime = DateTime.Now; + string bucketName = GetRandomName(15); + string objectName = GetRandomName(10); + string contentType = "application/octet-stream"; + Dictionary args = new Dictionary + { + { "bucketName", bucketName}, + {"objectName",objectName}, + {"contentType", contentType}, + {"data","0B"}, + {"size","-1"}, + }; + try + { + // Putobject call where unknown stream sent 0 bytes. + await Setup_Test(minio, bucketName); + using (System.IO.MemoryStream filestream = rsg.GenerateStreamFromSeed(0)) + { + long size = -1; + long file_write_size = filestream.Length; + + await minio.PutObjectAsync(bucketName, + objectName, + filestream, + size, + contentType); + await minio.RemoveObjectAsync(bucketName, objectName); + await TearDown(minio, bucketName); + } + new MintLogger("PutObject_Test8",putObjectSignature1,"Tests PutObject where unknown stream sends 0 bytes",TestStatus.PASS,(DateTime.Now - startTime), args:args).Log(); + } + catch (Exception ex) + { + new MintLogger("PutObject_Test8",putObjectSignature1,"Tests PutObject where unknown stream sends 0 bytes",TestStatus.FAIL,(DateTime.Now - startTime),"",ex.Message, ex.ToString(), args).Log(); + } + } private async static Task PutGetStatEncryptedObject_Test1(MinioClient minio) { DateTime startTime = DateTime.Now; diff --git a/Minio/ApiEndpoints/ObjectOperations.cs b/Minio/ApiEndpoints/ObjectOperations.cs index 1d5527efc..624b3b540 100644 --- a/Minio/ApiEndpoints/ObjectOperations.cs +++ b/Minio/ApiEndpoints/ObjectOperations.cs @@ -280,7 +280,7 @@ await GetObjectAsync(bucketName, objectName, (stream) => for (partNumber = 1; partNumber <= partCount; partNumber++) { byte[] dataToCopy = ReadFull(data, (int)partSize); - if (dataToCopy == null) + if (dataToCopy == null && numPartsUploaded > 0) { break; }