Как переопределить вывод необработанных исключений по умолчанию в Owin?

Я написал простой сервер, используя автономный хостинг OWA и 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>

Но я хочу другой вывод - так как я могу переопределить это?

Ответ 1

Вы делаете это, создавая OWIN MiddleWare и подключая его к конвейеру:

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, исключения не будут распространяться. Этот обработчик предназначен для любого исключения, которое происходит с помощью неуправляемого базового сервиса.

Ответ 2

Когда вы ожидаете метод async, исключение не будет передано в контекст вызывающего, но будет доступно путем проверки задачи, возвращенной вызывающему.

Итак, с этой модификацией решения, предоставленного @yuval-itzchakov, вы сможете уловить основные исключения:

var subtask = Next.Invoke(context);

await subtask;

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