Luis Valencia
Luis Valencia

Reputation: 34038

How to return an image (not the url) in binary via a REST API .net

ok I have a webapi which gets/adds/updates/deletes customers, the backend is CosmosDB.

Users are able to upload image for each customer, the file is stored in Azure Blob Storage, but the filename is stored in the CosmosDB property.

[HttpPost]
        public async Task<IHttpActionResult> Adduser([FromBody]User user)
        {
            var telemetry = new TelemetryClient();
            try
            {
                var userStore = CosmosStoreHolder.Instance.CosmosStoreUser;
                if (!ModelState.IsValid)
                {
                    return BadRequest(ModelState);
                }

                //Then we validate the content type
                if (!Request.Content.IsMimeMultipartContent("form-data"))
                {
                    throw new HttpResponseException(HttpStatusCode.UnsupportedMediaType);
                }

                #region File upload
                //Initalize configuration settings
                var accountName = ConfigurationManager.AppSettings["storage:account:name"];
                var accountKey = ConfigurationManager.AppSettings["storage:account:key"];
                var profilepicturecontainername = ConfigurationManager.AppSettings["storage:account:profilepicscontainername"];

                //Instance objects needed to store the files
                var storageAccount = new CloudStorageAccount(new StorageCredentials(accountName, accountKey), true);
                CloudBlobClient blobClient = storageAccount.CreateCloudBlobClient();
                CloudBlobContainer imagesContainer = blobClient.GetContainerReference(profilepicturecontainername);
                var provider = new AzureStorageMultipartFormDataStreamProvider(imagesContainer);

                // Validate extension and image size
                foreach (MultipartFileData file in provider.FileData)
                {

                    var fileName = file.Headers.ContentDisposition.FileName.Trim('\"').Trim();
                    if (fileName.EndsWith(".png"))
                    {
                        var img = Image.FromFile(file.LocalFileName);
                        if (img.Width != 200 && img.Height != 200)
                        {
                            string guid = Guid.NewGuid().ToString();

                            return BadRequest($"Error Lulo. Unsupported extension, only PNG is valid. Or unsuported image dimensions (200px x 200px)");
                        }
                    }
                }


                //Try to upload file
                try
                {
                    await Request.Content.ReadAsMultipartAsync(provider);
                }
                catch (Exception ex)
                {
                    string guid = Guid.NewGuid().ToString();
                    var dt = new Dictionary<string, string>
                    {
                        { "Error Lulo: ", guid }
                    };
                    telemetry.TrackException(ex, dt);
                    return BadRequest($"Error Lulo. An error has occured. Details: {guid} {ex.Message}: ");
                }

                // Retrieve the filename of the file you have uploaded
                var filename = provider.FileData.FirstOrDefault()?.LocalFileName;
                if (string.IsNullOrEmpty(filename))
                {
                    string guid = Guid.NewGuid().ToString();
                    var dt = new Dictionary<string, string>
                    {
                        { "Error Lulo: ", guid }
                    };

                    return BadRequest($"Error Lulo. An error has occured while uploading your file. Please try again.: {guid} ");
                }

                //Rename file
                CloudBlockBlob blobCopy = imagesContainer.GetBlockBlobReference(user.Id + ".png");
                if (!await blobCopy.ExistsAsync())
                {
                    CloudBlockBlob blob = imagesContainer.GetBlockBlobReference(filename);

                    if (await blob.ExistsAsync())
                    {
                        await blobCopy.StartCopyAsync(blob);
                        await blob.DeleteIfExistsAsync();
                    }
                }

                #endregion

                if (string.IsNullOrEmpty(user.CustomerId) && string.IsNullOrEmpty(user.PartnerId))
                {
                    return BadRequest("ClientID or PartnerId must be filled in.");
                }

                var added = await userStore.AddAsync(user);
                return Ok(added);
            }
            catch (Exception ex)
            {
                string guid = Guid.NewGuid().ToString();
                var dt = new Dictionary<string, string>
                {
                    { "Error Lulo: ", guid }
                };

                telemetry.TrackException(ex, dt);
                return BadRequest("Error Lulo: " + guid);
            }             
        }

Now, I need to return to the web api, the image in someway the frontend developers can render it. The Azure Blob Container is not public, so returning the Url will not be enough.

The front end is react.

This is my get method (which returns the picture Url, only)

 [HttpGet]
        public async Task<IHttpActionResult> GetUser(string email)
        {
            var telemetry = new TelemetryClient();
            try
            {
                var userStore = CosmosStoreHolder.Instance.CosmosStoreUser;
                var roleStore = CosmosStoreHolder.Instance.CosmosStoreRole;

                var user = await userStore.Query().FirstOrDefaultAsync(x => x.EmailAddress == email);
                if (user == null)
                {
                    return Unauthorized();
                }
                var role = await roleStore.Query().FirstOrDefaultAsync(x => x.Id == user.RoleId);
                user.RoleName = role.RoleName;

                return Ok(user);
            }
            catch (Exception ex)
            {
                string guid = Guid.NewGuid().ToString();
                var dt = new Dictionary<string, string>
                {
                    { "Error Lulo: ", guid }
                };

                telemetry.TrackException(ex, dt);
                return BadRequest("Error Lulo: " + guid);
            }
        }

And just in case, the User POCO here below:

public class User : ISharedCosmosEntity
    {
        [JsonProperty("Id")]
        public string Id { get; set; }
        public string EmailAddress { get; set; }
        public string FirstName { get; set; }
        public string LastName { get; set; }
        public bool Enabled { get; set; }

        public string ProfilePictureUrl { get; set; }
        public string RoleName { get; set; }
        public string CustomerName { get; set; }
        public string PartnerName { get; set; }

        public string CustomerId{ get; set; }
        public string PartnerId { get; set; }
        public string RoleId { get; set; }

        [CosmosPartitionKey]
        public string CosmosEntityName { get; set; }
    }

Upvotes: 3

Views: 2865

Answers (5)

Cujoey
Cujoey

Reputation: 147

This is how I did this in one of my previous project. I have a method that takes the Url path of the picture and the name and extension. The method convert the image to a base64 string.

public string ConvertToBase64(string path)
{
   byte[] b = System.IO.File.ReadAllBytes(path);
   var base64img = "data:image/jpg;base64," + Convert.ToBase64String(b);
   Return base64img;
}

And use it like this.

var imgUrl = AzureImgUrlPath + "imageName.jpg";
var imageBase64String = ConvertToBase64(imgUrl);

Upvotes: 1

Marcel Kirsche
Marcel Kirsche

Reputation: 455

If you do not have a constant public url to the image, you need to encode it to base64 and embed it as inline image.

If you can access the url inside your POCO from your Web API backend, then you can retrieve and convert the image with the following code:

        private static HttpClient _httpClient = new HttpClient();

        public async Task<string> GetInlineImageSrcAsync(string url)
        {
            var bytes = await _httpClient.GetByteArrayAsync(url);
            var base64 = Convert.ToBase64String(bytes);
            var mimeType = "image/png";
            // If mime types differ, try this
            // var mimeType = $"image/{ParseExtensionFromUrl(url)}"
            var inlineImageSrc = $"data:{mimeType};base64,{base64}";
            return inlineImageSrc;
        }

        public string ParseExtensionFromUrl(string url)
        {
            return url.Substring(url.LastIndexOf(".") + 1);
        }

Note that HttpClient should be static to enable it to reuse connections. This is recommended by Microsoft and boosts performance. More on this:

https://medium.com/@nuno.caneco/c-httpclient-should-not-be-disposed-or-should-it-45d2a8f568bc

https://aspnetmonsters.com/2016/08/2016-08-27-httpclientwrong/

EDIT: This is the HTML to actually embed the image:

<img src="<THE STRING YOU RETURNED>" />

<!-- For Example: -->
<img src="">

Upvotes: 2

Andreas Paulsson
Andreas Paulsson

Reputation: 7813

The best way to return a file in Web API is to load it to memory and then return it via the File method:

[HttpGet]
public IActionResult Get()
{            
    Byte[] b = ...;   // Load blob from storage to byte array, usually via a MemoryStream. 
    return File(b, "image/jpeg");
}

The File method is really a helper that return a FileContentReult or FileStreamResult, you can read motre about it here:

Difference between FileContentResult and FileStreamResult

If you use one of the File method overloads that takes a stream as argument, you can even avoid buffering the entire blob in memory.

Upvotes: 1

Dtrl-01
Dtrl-01

Reputation: 9

I'm assuming you're looking to embed the picture in a element? Why don't you use Base64? Then you can return something like:

data:image/[png/jpg depending on image type];base64,[Your Base64 image value here]

Upvotes: 1

MswatiLomnyama
MswatiLomnyama

Reputation: 1386

You might consider converting the image to base64 string using the approach on: https://www.c-sharpcorner.com/blogs/convert-an-image-to-base64-string-and-base64-string-to-image

Upvotes: 0

Related Questions