dotnetzip создает пустой поврежденный файл - PullRequest
0 голосов
/ 29 ноября 2018

Я пытаюсь создать функцию, запускаемую по протоколу HTTP, которая будет извлекать большой двоичный объект из хранилища Azure, архивировать его, а затем повторно загружать почтовый индекс в хранилище больших двоичных объектов.Где-то по пути это идет не так.Он успешно заархивировал и повторно загрузил, но файл почтового индекса имеет содержание 0 КБ и не может быть открыт.Любая помощь будет оценена.Найдите код ниже.

 public static async Task<HttpResponseMessage> Run([HttpTrigger(AuthorizationLevel.Function, "get", "post", Route = null)]HttpRequestMessage req, TraceWriter log)
    {
        log.Info("C# HTTP trigger function processed a request.");
        try
        {
          string fileName = req.GetQueryNameValuePairs()
                .FirstOrDefault(q => string.Compare(q.Key, "fileName", true) == 0)
                .Value;

            log.Info($"fileName: {fileName}");

            dynamic request = await req.Content.ReadAsAsync<object>();
            var container = (string)request.body;

            string connectionString = Environment.GetEnvironmentVariable("blobstorageconnectionstring");
            CloudStorageAccount storageAccount = CloudStorageAccount.Parse(connectionString);
            CloudBlobClient client = storageAccount.CreateCloudBlobClient();

            MemoryStream outputStream = new MemoryStream();

            using (ZipFile zip = new ZipFile())
            {
                zip.AddEntry($"{fileName}.xlsx", DownloadFromBlobStorage(fileName, connectionString, container));
                zip.CompressionMethod = CompressionMethod.Deflate;
                zip.Save(outputStream);
            }


            UploadToBlobStorage(fileName + "ZIP", outputStream, connectionString, container);

            return req.CreateResponse(HttpStatusCode.OK, "OK");
        }
        catch (Exception ex)
        {
            log.Error("error", ex: ex);

            return req.CreateResponse(HttpStatusCode.BadRequest);
        }

    }
    private static void UploadToBlobStorage(string name, MemoryStream dataStream, string storageConnectionString, string blobContainerName)
    {

        CloudStorageAccount account = CloudStorageAccount.Parse(storageConnectionString);
        CloudBlobClient client = account.CreateCloudBlobClient();

        CloudBlobContainer container = client.GetContainerReference(blobContainerName);
        container.CreateIfNotExists();

        CloudBlockBlob blob = container.GetBlockBlobReference(name);
        blob.Properties.ContentType = "application/x-zip-compressed";

        dataStream.Position = 0;
        blob.UploadFromStream(dataStream);
    }
    private static MemoryStream DownloadFromBlobStorage(string fileName, string storageConnectionString, string blobContainerName)
    {

        CloudStorageAccount account = CloudStorageAccount.Parse(storageConnectionString);
        CloudBlobClient client = account.CreateCloudBlobClient();

        CloudBlobContainer container = client.GetContainerReference(blobContainerName);
        container.CreateIfNotExists();

        CloudBlockBlob blob = container.GetBlockBlobReference(fileName);

        MemoryStream memStream = new MemoryStream();

        blob.DownloadToStream(memStream);

        return memStream;
    }

1 Ответ

0 голосов
/ 29 ноября 2018

Похоже, вам следует поместить объект MemoryStream, возвращенный из DownloadFromBlobStorage(), в ноль (вы уже делаете это с outputStream / dataStream на UploadToBlobStorage()), прежде чем передавать его в zip.AddEntry():

   MemoryStream inputStream = DownloadFromBlobStorage(fileName, connectionString, container);
   inputStream.Position = 0;
   zip.AddEntry($"{fileName}.xlsx", inputStream);

Посмотрите на этот ответ:

Создание Zip-файла из потока и его загрузка - Переполнение стека

После еще одного исследования я обнаружил, что обычно вы всегда должны позиционировать поток перед передачей его для функции:

Кто отвечает за позиционирование потока?- стек разработки программного обеспечения

...