写入 HttpResponse 时标头为只读异常

ali*_*if3 6 c# asp.net-mvc .net-core

我有一个看起来像这样的控制器

IHttpWrite httpWrite;

[HttpPost]
public async Task<HttpResponse> Post(Request req)
{
    return await httpWrite.Write(req, Response);
}
Run Code Online (Sandbox Code Playgroud)

这就是所谓的

public async Task<HttpResponse> Write(object data, HttpResponse httpResponse)
{
    var json = JsonConvert.SerializeObject(data);

    httpResponse.OnStarting(() =>
    {
        httpResponse.Clear();
        httpResponse.ContentType = "application/json";
        return Task.CompletedTask;
    });

    await httpResponse.WriteAsync(json);
    return httpResponse;
}
Run Code Online (Sandbox Code Playgroud)

但是调用它会得到这个异常

System.InvalidOperationException: Headers are read-only, response has already started.
   at Microsoft.AspNetCore.Server.Kestrel.Core.Internal.Http.HttpHeaders.ThrowHeadersReadOnlyException()
Run Code Online (Sandbox Code Playgroud)

我知道一旦数据写入响应就无法设置标头,但就我而言,我在向 HttpResponse 写入任何内容之前设置标头。

根据我所做的调试,标头的异常仅在控制器返回 HttpResponse 后才会抛出。这向我表明尝试更改标头是在我的代码之外的某些内容上进行的。

我正在使用 Visual Studio 设置的大部分样板代码。这是默认程序

public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
        WebHost.CreateDefaultBuilder(args)
               .UseStartup<Startup>();
Run Code Online (Sandbox Code Playgroud)

这是调用大多数默认启动

public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
    if (env.IsDevelopment())
    {
        app.UseDeveloperExceptionPage();
    }
    else
    {
        app.UseHsts();
    }

    app.UseHttpsRedirection();
    app.UseStaticFiles();
    app.UseMvc();
}
Run Code Online (Sandbox Code Playgroud)

我相信解决方案要么在于更改配置中的某些内容,以便在返回 HttpResponse 后不自动设置响应标头,要么可能返回不同的对象,以便我不必手动设置标头。由于无法知道在写入 HttpResponse 后尝试设置标头的来源,我不确定如何继续。

Leo*_*rdo 1

为什么不在写响应之前先写标题呢?甚至更好:为什么不将返回声明为更抽象的内容并让框架处理这些内容,而不是手动设置响应?

无论如何,这里有一段关于如何添加自定义标头的代码:

using System;
using System.Linq;
using System.Net;
using Microsoft.AspNetCore.Mvc;

namespace WebApplication.Controllers
{
    [Route("api")]
    public class ValuesController : Controller
    {
        [HttpGet]
        [Route("values/{top}")]
        public IActionResult Get(int top)
        {
            // Generate dummy values
            var list = Enumerable.Range(0, DateTime.Now.Second)
                             .Select(i => $"Value {i}")
                             .ToList();
            list.Reverse();

            var result = new ObjectResult(list.Take(top))
            {
                StatusCode = (int)HttpStatusCode.OK
            };

            Request.HttpContext.Response.Headers.Add("X-Total-Count", list.Count.ToString());

            return result;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)