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).
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;
}
}
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With