Подтвердить что ты не робот

Как загрузить изображение и данные POST на конечную точку ApiController Azure Mobile Services?

Я пытаюсь загрузить изображение и данные формы POST (хотя в идеале я бы хотел, чтобы оно было json) для конечной точки в моем приложении Azure Mobile Services.

>

У меня есть метод 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]string metadata, но тогда он отлично работает.

Когда [FromBody]string metadata включен (как указано выше), я получаю ошибку:

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

Однако я хотел бы POST дополнительный metadata (который может быть длинным, поэтому я не хочу помещать его в Uri).

Как я могу сохранить логику загрузки файла, а также POST дополнительные данные строки на мой контроллер?

Я использую Azure Mobile Services, поэтому этот код находится внутри System.Web.Http.ApiController (если это имеет значение).

4b9b3361

Ответ 1

То, что я делал ранее, - это заставить клиента опубликовать json-модель, которая содержит как метаданные, так и фактические файлы. Каждый файл представляет собой строку с кодировкой base64 фактического содержимого файла. Следующий код должен иметь возможность обрабатывать данные uri + base64.

В моем приложении frontend используется API файлов javascript для получения ссылки на объект FileReader, который может возвращать строку base64 (data uri) с помощью метода readAsDataURL. Это в основном делает что-то вроде этого:

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"
}]
}

Контроллер выглядит следующим образом:

[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();
}

Тогда у меня есть следующие модели для поддержки контроллера:

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;
        }

    }
}