如何在 Blazor WASM 客户端中访问 Httpclient 标头

Posted

技术标签:

【中文标题】如何在 Blazor WASM 客户端中访问 Httpclient 标头【英文标题】:How to access Httpclient headers in Blazor WASM Client 【发布时间】:2021-03-31 16:58:15 【问题描述】:

我的客户端代码调用了一个 API,我试图从响应标头中获取返回的 ETag 值。如果我使用 Fiddler,我可以看到响应包含 ETag 标头,如果我使用 Postman 进行 API 调用,我可以看到 ETag 标头,但是无论我采用什么方法尝试检索代码中的标头,我得到的只是一个空。

本质上是 API 调用;

// create request object
var request = new HttpRequestMessage(HttpMethod.Get, url);
// add authorization header
request.Headers.Authorization = new AuthenticationHeaderValue("bearer", await GetBearerToken());
// send request
HttpResponseMessage response = await _client.SendAsync(request);

Fiddler Response Header showing Etag

邮递员的回应是;

Postman response headers

我花了几个小时从网络上搜索和尝试示例,但无论我尝试什么,我都无法获得 ETag 标头。

使用下面的示例代码,我确实获得了前 2 个标头,如返回的 Postman 响应标头中所示,但不是 ETag 标头/值。

String allResponseHeaders = Enumerable
    .Empty<(String name, String value)>()
    .Concat(
    response.Headers
    .SelectMany(kvp => kvp.Value
    .Select(v => (name: kvp.Key, value: v))
    ))
    .Concat(
    response.Content.Headers
    .SelectMany(kvp => kvp.Value
    .Select(v => (name: kvp.Key, value: v))
    ))
   .Aggregate(
   seed: new StringBuilder(),
   func: (sb, pair) => sb.Append(pair.name).Append(": ").Append(pair.value).AppendLine(),
   resultSelector: sb => sb.ToString()
   );

我正在使用 Visual Studio、Blazor 和 aspnetcore 5.0,我希望使用 IndexDB 生成 PWA,并使用 ETag 来减少数据下载。

任何有关如何获取 Etag 标头的帮助将不胜感激...

【问题讨论】:

【参考方案1】:

我和其他人有类似的问题。实际上,我们有 github 问题和关于这个问题的精彩讨论。我只是想与其他可能卡在同一点并访问堆栈溢出的人分享。

问题链接(已关闭):

https://github.com/dotnet/runtime/issues/42179

引用讨论“您的 CORS 策略需要指定公开哪些标头。如果您希望客户端读取所有标头,您可以这样指定”

示例(服务器端):

services.AddCors(options =>

    options.AddPolicy("Open", builder => builder
        .AllowAnyOrigin()
        .AllowAnyHeader()
        .AllowAnyMethod()
        .WithExposedHeaders("*")); // this is the important part!
);

【讨论】:

Dino,非常感谢这次更新,我差点放弃让 eTag 与 Blazor WASM 一起工作。但是,我只是按照您的说明添加了 .WithExposedHeaders("*") 并返回了 eTag 标头。我仍然有一个问题,为什么 Postman 在 API 服务器中没有该设置的情况下获取 eTag 标头,而 Blazor WASM 没有,但这可能是另一天......再次感谢......【参考方案2】:

我创建了非常简单的控制台应用程序并使用它从 API 获取数据,我可以看到响应中返回了 ETag 值,但似乎无法使用 response.Headers.Etag 简单地提取代码中的值.

using System.Net.Http;
using System.Net.Http.Headers;
using System.Threading.Tasks;

namespace SimpleAPIClient


    class Program
    
        private static readonly HttpClient client = new HttpClient();

        static async Task Main(string[] args)
        

            var request = new HttpRequestMessage(HttpMethod.Get, "https://localhost:44338/api/headers/3");
            request.Headers.Authorization = new AuthenticationHeaderValue("bearer", "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJhZG1pbkBjb250cmEtc29mdC5jb20iLCJqdGkiOiI4NjgyMmQ1YS1iNTQ3LTRlMjItYmQ3NS0wNDlkZmE5MTMzZmUiLCJodHRwOi8vc2NoZW1hcy54bWxzb2FwLm9yZy93cy8yMDA1LzA1L2lkZW50aXR5L2NsYWltcy9uYW1laWRlbnRpZmllciI6IjVkODExZmY1LTgyY2QtNDY2NC05ODkzLTZmNGRjMDU0YzFmNSIsImh0dHA6Ly9zY2hlbWFzLm1pY3Jvc29mdC5jb20vd3MvMjAwOC8wNi9pZGVudGl0eS9jbGFpbXMvcm9sZSI6IkFkbWluaXN0cmF0b3IiLCJleHAiOjE2MDgzNTU0MTEsImlzcyI6ImNvbnRyYS1zb2Z0LmNvbSIsImF1ZCI6ImNvbnRyYS1zb2Z0LmNvbSJ9.w_CSstMZkFFn0DcMlcPNNm_Nr0idzHPo6I2hRjEUglQ");
            HttpResponseMessage response = await client.SendAsync(request);

            var etag = response.Headers.ETag;

            int i = 0;
        

    

VS Breakpoint shows ETag data

我认为问题是由于该项目是 Blazor Web Assembly 项目,如果我创建简单的 .net 核心项目,则以下代码可以工作;

client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("bearer", "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiO");
Task<HttpResponseMessage> s = client.GetAsync("https://localhost:44338/api/headers/3");
HttpResponseMessage nresponse2 = await s;
var etag2 = nresponse2.Headers.FirstOrDefault(i => i.Key == "ETag").Value.FirstOrDefault();

但是 Blazor 中的相同代码却没有。

【讨论】:

你能用 .Content.Headers 比如response.Content.Headers.ETag 试试吗? 问题在于 Blazor Web Assembly 项目,如果我创建控制台应用程序,则下面的代码有效;【参考方案3】:

在响应消息对象上,可以直接访问 ETag 标头。

// create request object
var request = new HttpRequestMessage(HttpMethod.Get, url);
// add authorization header
request.Headers.Authorization = new AuthenticationHeaderValue("bearer", await GetBearerToken());
// send request
HttpResponseMessage response = await _client.SendAsync(request);

//read the response headers. The ETag is already a "known" header and is easily accessible via a property
EntityTagHeaderValue etagHeader = response.Headers.ETag;

【讨论】:

感谢您的回复,我尝试了类似的选项,现在我已经尝试了您的代码,但返回的值仍然为空。我刚刚使用 restsharp 和 .net 4.7.2 制作了一个简单的控制台应用程序,并且可以看到所有标题,因此要创建简单的 aspnetcore 控制台应用程序来查看 httpclient 是否有效,将尝试版本 3.1 和 5.0

以上是关于如何在 Blazor WASM 客户端中访问 Httpclient 标头的主要内容,如果未能解决你的问题,请参考以下文章