【问题标题】:How to override default unhandled exception output in Owin?如何覆盖 Owin 中默认的未处理异常输出?
【发布时间】:2026-02-06 01:25:01
【问题描述】:

我使用 Owin 自托管和 WebApi 编写了简单的服务器:

namespace OwinSelfHostingTest
{
    using System.Threading;
    using System.Web.Http;
    using Microsoft.Owin.Hosting;
    using Owin;

    public class Startup
    {
        public void Configuration(IAppBuilder builder)
        {
            var config = new HttpConfiguration();

            config.Routes.MapHttpRoute(
                "Default",
                "{controller}/{id}",
                new { id = RouteParameter.Optional }
                );

            builder.UseWebApi(config);
        }
    }

    public class Server
    {
        private ManualResetEvent resetEvent = new ManualResetEvent(false);
        private Thread thread;

        private const string ADDRESS = "http://localhost:9000/";

        public void Start()
        {
            this.thread = new Thread(() =>
                {
                    using (var host = WebApp.Start<Startup>(ADDRESS))
                    {
                        resetEvent.WaitOne(Timeout.Infinite, true);
                    }
                });
            thread.Start();
        }

        public void Stop()
        {
            resetEvent.Set();
        }
    }

}

当控制器出现异常时,Owin 会返回如下 XML 响应:

<Error>
    <Message>An error has occurred.</Message>
    <ExceptionMessage>Attempted to divide by zero.</ExceptionMessage>
    <ExceptionType>System.DivideByZeroException</ExceptionType>
    <StackTrace> 
        ...
    </StackTrace>
</Error>

但我想要不同的输出 - 那么我该如何覆盖它呢?

【问题讨论】:

  • 你在寻找什么样的输出?

标签: c# exception asp.net-web-api owin


【解决方案1】:

您可以通过创建 OWIN 中间件并将其连接到管道中来实现:

public class CustomExceptionMiddleware : OwinMiddleware
{
   public CustomExceptionMiddleware(OwinMiddleware next) : base(next)
   {}

   public override async Task Invoke(IOwinContext context)
   {
      try
      {
          await Next.Invoke(context);
      }
      catch(Exception ex)
      {
          // Custom stuff here
      }
   }
 }

并在启动时挂钩:

public class Startup
{
    public void Configuration(IAppBuilder builder)
    {
        var config = new HttpConfiguration();

        config.Routes.MapHttpRoute(
            "Default",
            "{controller}/{id}",
            new { id = RouteParameter.Optional }
            );

        builder.Use<CustomExceptionMiddleware>().UseWebApi(config);
    }
}

这样任何未处理的异常都会被您的中间件捕获,并允许您自定义输出结果。

需要注意的重要一点:如果托管的事物有它自己的异常处理逻辑,就像 WebAPI 那样,异常不会传播。此处理程序适用于由托管的底层服务处理的任何异常。

【讨论】:

  • 顺便说一句,这不起作用。在调用 Next.Invoke() 时,永远不会触发 catch 块。在我的搜索中偶然发现这一点会很高兴知道如何做到这一点。被吞下的异常不会出现上下文或字典。
  • @DrSchizo 好吧,如果您的异常是由 WebAPI 处理的,那么您当然不会在这里看到它,因为它已经被处理并包含在上述框架中。
【解决方案2】:

当您等待异步方法时,异常不会被抛出到调用者的上下文中,但可以通过检查返回给调用者的任务来获得。

因此,通过对@yuval-itzchakov 提供的解决方案的这种修改,您将能够捕获底层异常:

var subtask = Next.Invoke(context);

await subtask;

if (subtask.Exception != null)
{
    // log subtask.Exception here
}

【讨论】:

  • 如果TaskTask&lt;T&gt; 抛出异常,它肯定会在await 点传播。为什么你认为它不会?
  • 任务中的异常不是这样工作的。