c# - 如何将图像和 POST 数据上传到 Azure 移动服务 ApiController 终结点?

标签 c# asp.net-mvc azure post

我正在尝试将图像 POST 表单数据(尽管理想情况下我希望它是 json)上传到端点在我的 Azure 移动服务应用程序中。

我有 ApiController 方法:

[HttpPost]
[Route("api/upload/{databaseId}/{searchingEnabled}/{trackingEnabled}")]
public async Task<IHttpActionResult> Upload(string databaseId, string searchingEnabled, string trackingEnabled, [FromBody]string metadata) {

    if (!Request.Content.IsMimeMultipartContent()) {
        return BadRequest("No image is uploaded.");
    }
    else {
        var provider = new MultipartMemoryStreamProvider();
        await Request.Content.ReadAsMultipartAsync(provider);
        foreach (var file in provider.Contents) {
             // Process each image uploaded
        }
    }
}

这仅在我删除[FromBody]字符串元数据时才起作用,但随后效果很好。

当包含 [FromBody]string 元数据 时(如上所述),我收到错误:

The request entity's media type 'multipart/form-data' is not supported for this resource.

但是,我想POST额外的元数据(可能很长,所以我不想把它放在乌里)。

如何保留文件上传逻辑,以及POST附加字符串数据到我的 Controller ?

我正在使用 Azure 移动服务,因此此代码位于 System.Web.Http.ApiController 内(如果这很重要)。

最佳答案

我之前所做的就是让客户端发布一个包含元数据和实际文件的 json 模型。每个文件都是实际文件内容的 Base64 编码字符串。下面的代码应该能够处理数据 uri + base64。

我的前端应用程序正在使用 javascript File API 获取对 FileReader 对象的引用,该对象可以使用 readAsDataURL 方法返回 base64(数据 uri)字符串。它基本上做了这样的事情:

var attachment = {};

function loadAttachmentFromFileInput(element) {
    var file = element.files[0];

    var reader = new FileReader();

    reader.onload = function(e) {
        var result = reader.result;

        attachment = {
            data: result,
            filename: file.name,
            type: file.type,
            size: file.size
            }
    }

    reader.readAsDataURL(file);    
}

然后我将构建一个 POST 模型,该模型会生成以下 json 内容:

{
  "messageId": 1,
  "foo": "bar",
  "bar": "foo",
  "attachments": [{
    "filename": "stackoverflow.jpg",
    "data": "",
    "type": "image/jpeg"
}]
}

Controller 看起来像这样:

[Route("api/messages/{messageId:guid}")]
public async Task<IHttpActionResult> Post(Guid messageId, CreateMessageAttachments model)
{
    // Access to all properties in your post model
    Trace.WriteLine(model.Foo);
    Trace.WriteLine(model.Bar);

    foreach (var attachment in model.Attachments)
    {
        // Do what you need to with the bytes from the uploaded attachments
        var bytes = attachment.GetByteArray();
    }

    return Ok();
}

那么我有以下型号来支持 Controller :

public class CreateMessageAttachments
{
    public Guid MessageId { get; set; }
    public string Foo { get; set; }
    public string Bar { get; set; }
    public IList<CreateAttachment> Attachments { get; set; }
}

public class CreateAttachment
{
    public string Data { get; set; }
    public string Filename { get; set; }
    public string Type { get; set; }

    public string GetBase64()
    {
        if (string.IsNullOrWhiteSpace(Data))
            return null;

        var index = Data.LastIndexOf("base64");

        if (index == -1)
            return Data;

        return Data.Substring(index + 7);
    }

    public byte[] GetByteArray()
    {
        try
        {
            var base64 = GetBase64();

            if (string.IsNullOrWhiteSpace(base64))
                return null;

            return Convert.FromBase64String(base64);
        }
        catch
        {
            return null;
        }

    }
}

关于c# - 如何将图像和 POST 数据上传到 Azure 移动服务 ApiController 终结点?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32015898/

相关文章:

c# - 如何根据点获取字符串的子代

c# - Windows 7 中的 ServiceController 权限

django + celery + redis + postgres 挂了

wcf - Azure 可以与 Amazon 互操作吗?

C# (OOP) 嵌套业务对象

c# - 在c#中从json字符串中查找一个值

c# - MVC&EF : remove time from datetime

c# - 从方法中重定向到 Action

jquery - 无法在 mvc 4 中使用 jquery ajax 更新部分 View

azure - 使用 azure oauth 客户端凭据流时如何添加可选声明以访问 token