c# - 如何通过 REST API .net 以二进制形式返回图像(不是 url)

标签 c# .net asp.net-web-api asp.net-web-api2 azure-blob-storage

好的,我有一个获取/添加/更新/删除客户的 webapi,后端是 CosmosDB。

用户可以为每个客户上传图像,文件存储在 Azure Blob 存储中,但文件名存储在 CosmosDB 属性中。

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

现在,我需要返回到 web api,前端开发人员可以以某种方式呈现图像。 Azure Blob 容器不是公共(public)的,因此返回 Url 是不够的。

前端是react。

这是我的get方法(只返回图片Url)

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

为了以防万一,下面的用户 POCO:

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

最佳答案

如果您没有图像的固定公共(public) url,则需要将其编码为 base64 并将其作为内联图像嵌入。

如果您可以从 Web API 后端访问 POCO 中的 url,则可以使用以下代码检索和转换图像:

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

请注意,HttpClient 应该是静态的,以使其能够重用连接。这是 Microsoft 推荐的,可以提高性能。 更多相关信息:

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

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

编辑: 这是实际嵌入图像的 HTML:

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

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

关于c# - 如何通过 REST API .net 以二进制形式返回图像(不是 url),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58872188/

相关文章:

.net - 使用 ClickOnce 应用程序部署 NLog

c# - 使用 Web API 的模型绑定(bind)接口(interface)属性

c# - 我如何在 css 中更改文本框的宽度?

c# - 为什么我需要 2 Console.ReadLine();暂停控制台?

c# - LINQ 按匹配字符位置排序

C# Autolevel 片段?

c# - 验证是否存在可用网络连接的最简单方法是什么?

.net - Visual Studio 资源生成 - 自定义工具命名空间

xamarin - xamarin.forms 中的数据发布错误

c# - 无法从特定角色中删除用户