我创建了一个Web API,允许用户发送文件并将其上传到Azure存储。它的工作方式是,客户端应用程序将连接到API,以将一个或多个文件发送到文件上传控制器,并且控制器将负责其他工作,例如
效果很好,但我认为这不是正确的方法,因为现在我可以看到有两个不同的过程
给我的感觉是,我正在复制上载过程,因为同一文件首先从客户端(文件系统)传递到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);
}
}
答案 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/。