Как переопределить вывод необработанных исключений по умолчанию в 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
}