复制文件上传过程-Asp.net WebApi

时间:2019-06-27 02:48:29

标签: azure asp.net-web-api azure-storage

我创建了一个Web API,允许用户发送文件并将其上传到Azure存储。它的工作方式是,客户端应用程序将连接到API,以将一个或多个文件发送到文件上传控制器,并且控制器将负责其他工作,例如

  • 将文件上传到Azure存储
  • 更新数据库

效果很好,但我认为这不是正确的方法,因为现在我可以看到有两个不同的过程

  1. 将文件从客户端的文件系统上传到我的Web API(服务器)
  2. 从API(服务器)将文件上传到Azure存储

给我的感觉是,我正在复制上载过程,因为同一文件首先从客户端(文件系统)传递到API(服务器),然后传递到Azure(目的地)。我觉得有必要向客户端显示两个进度条,以实现文件上传进度(从客户端到服务器,然后从服务器到Azure)-这对我来说没有任何意义,我觉得我的方法不正确。

我的API最多可以接受250MB,因此您可以想象过载。

你们怎么看?

//// API控制器

if (!Request.Content.IsMimeMultipartContent("form-data"))
   {
    throw new HttpResponseException(HttpStatusCode.UnsupportedMediaType);
   }

var provider = new RestrictiveMultipartMemoryStreamProvider();
var contents = await Request.Content.ReadAsMultipartAsync(provider);
int Total_Files = contents.Contents.Count();
foreach (HttpContent ctnt in contents.Contents) 
 {
   await storageManager.AddBlob(ctnt)
 }

//////流

#region SteamHelper

public class RestrictiveMultipartMemoryStreamProvider : MultipartMemoryStreamProvider
{
    public override Stream GetStream(HttpContent parent, HttpContentHeaders headers)
    {
        var extensions = new[] { "pdf", "doc", "docx", "cab", "zip" };
        var filename = headers.ContentDisposition.FileName.Replace("\"", string.Empty);
        if (filename.IndexOf('.') < 0)
            return Stream.Null;

        var extension = filename.Split('.').Last();
        return extensions.Any(i => i.Equals(extension, StringComparison.InvariantCultureIgnoreCase))
                   ? base.GetStream(parent, headers)
                   : Stream.Null;
    }
}

#endregion SteamHelper

///// AddBlob

public async Task<string> AddBlob(HttpContent _Payload)
        {
            CloudStorageAccount cloudStorageAccount = KeyVault.AzureStorage.GetConnectionString();
            CloudBlobClient cloudBlobClient = cloudStorageAccount.CreateCloudBlobClient();
            CloudBlobContainer cloudBlobContainer = cloudBlobClient.GetContainerReference("SomeContainer");
            cloudBlobContainer.CreateIfNotExists();
            try
            {
                byte[] fileContentBytes = _Payload.ReadAsByteArrayAsync().Result;
                CloudBlockBlob blob = cloudBlobContainer.GetBlockBlobReference("SomeBlob");
                blob.Properties.ContentType = _Payload.Headers.ContentType.MediaType;
                blob.UploadFromByteArray(fileContentBytes, 0, fileContentBytes.Length);
                var B = await blob.CreateSnapshotAsync();
                B.FetchAttributes();
                return "Snapshot ETAG: " + B.Properties.ETag.Replace("\"", "");
            }
            catch (Exception X)
            {
                return ($"Error : " + X.Message);
            }            
        }

1 个答案:

答案 0 :(得分:0)

  

给我的感觉是我将复制过程复制为   相同的文件首先传输到API(服务器),然后传输到Azure   (目标)来自客户端(文件系统)。

我认为你是对的。一种可能的解决方案是让您的API生成Shared Access Signature (SAS) token并在客户端希望上载文件时将SAS令牌/ URI返回给客户端。

使用此SAS URI,客户端可以直接将文件上传到Azure存储,而无需先将其发送到API。客户端成功上传文件后,便可以向API发送消息以更新数据库。

您可以在这里了解有关SAS的更多信息:https://docs.microsoft.com/en-us/azure/storage/common/storage-dotnet-shared-access-signature-part-1

我还写了一篇很久以前的关于使用SAS的博客文章,您可能会发现它有用:https://gauravmantri.com/2013/02/13/revisiting-windows-azure-shared-access-signature/