如何通过REST API .net以二进制形式返回图像(而非URL)

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

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

export interface Food {
  value: string;
  viewValue: string;
}

/**
 * @title Basic select
 */
@Component({
  selector: "select-overview-example",templateUrl: "select-overview-example.html",styleUrls: ["select-overview-example.css"]
})
export class SelectOverviewExample {
  foods: Food[] = [
    { value: "steak-0",viewValue: "Steak" },{ value: "pizza-1",viewValue: "Pizza" },{ value: "tacos-2",viewValue: "Tacos" }
  ];
  selected= -1;
  allSelected:false;


  /*Track checkbox events*/
  onCheckboxChange(event) {
    console.log(event);
    if(event === 0 && !this.allSelected){
      this.allSelected = true;
    }else if(event === 0 && this.allSelected){
      this.allSelected = false;
    }
    console.log(this.allSelected);
    /** For other values **/
  }

}

现在,我需要返回Web api,前端开发人员可以某种方式渲染该图像。 Azure Blob容器不是公共的,因此返回Url是不够的。

前端有反应。

这是我的get方法(仅返回图片网址)

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

[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,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);
            }             
        }
d276418269 回答:如何通过REST API .net以二进制形式返回图像(而非URL)

我假设您要将图片嵌入元素中? 您为什么不使用Base64?然后您可以返回类似的内容:

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

,

在Web API中返回文件的最佳方法是将其加载到内存中,然后通过File方法返回它:

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

File方法实际上是一个返回FileContentReult或FileStreamResult的助手,您可以在此处阅读有关它的详细信息:

Difference between FileContentResult and FileStreamResult

如果您使用File方法的重载之一(将流作为参数),甚至可以避免在内存中缓冲整个blob。

,

如果您没有恒定的图片公共网址,则需要将其编码为base64并将其嵌入为内嵌图片。

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

        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="">
,

这就是我在上一个项目中做到的方式。我有一个采用图片的网址路径以及名称和扩展名的方法。该方法将图像转换为base64字符串。

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

并像这样使用它。

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

您可以考虑使用以下方法将图像转换为base64字符串:https://www.c-sharpcorner.com/blogs/convert-an-image-to-base64-string-and-base64-string-to-image

本文链接:https://www.f2er.com/3097210.html

大家都在问