Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How can I upload an image and POST data to an Azure Mobile Services ApiController endpoint?

I am trying to upload an image and POST form data (although ideally I'd like it to be json) to an endpoint in my Azure Mobile Services application.

I have the ApiController method:

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

This only works when I remove the [FromBody]string metadata, but then it works great.

When [FromBody]string metadata is included (as above), I get the error:

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

However, I would like to POST additional metadata (which can be long, so I don't want to put it in the Uri).

How can I keep the file upload logic, and also POST additional string data to my controller?

I am using Azure Mobile Services, so this code is inside an System.Web.Http.ApiController (if that matters).

like image 521
Brett Avatar asked Aug 14 '15 17:08

Brett


1 Answers

What I've done previously is to make the client post a json model which contains both the metadata and the actual files. Each file is then a base64 encoded string of the actual file content. The below code should be able to handle data uri + base64.

My frontend app is using the javascript File API to get a reference to a FileReader object which can return a base64 (data uri) string by using the readAsDataURL method. It's basically doing something like this:

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

I'm then building a POST model which results in the following json content:

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

Controller looks like this:

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

Then I have the following models to support the 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;
        }

    }
}
like image 106
Martin Avatar answered Oct 19 '22 21:10

Martin