当 AJAX 调用中发生错误时,ELMAH 返回 Error.aspx

发布于 2024-09-27 03:08:05 字数 2467 浏览 5 评论 0原文

我正在使用 ELMAH 来记录错误。我用我自己的 HandleError 属性扩展了它,允许 ELMAH 处理所有错误。据我了解,ELMAH 在记录错误后会将 Error.aspx 返回到浏览器。除了 Ajax 调用之外,这一切都很好。如果 Ajax 调用期间发生某些情况,我的客户期望以 JSON 形式返回异常。对于AjaxRequests,如何让ELMAH返回异常而不是返回Error.aspx?

以下是我扩展 HandleError 的方式:

public class ProspectorHandleErrorAttribute : HandleErrorAttribute
{
    public override void OnException(ExceptionContext context)
    {
   base.OnException(context);

   if (context.HttpContext.Request.IsAjaxRequest())
   {
   var innerExceptionMessage = (context.Exception.InnerException != null) ? context.Exception.InnerException.Message : String.Empty;

context.HttpContext.Response.StatusCode = (int)System.Net.HttpStatusCode.InternalServerError;
context.Result = new JsonResult
{
 JsonRequestBehavior = JsonRequestBehavior.AllowGet,
 Data = new
 {
  context.Exception.Message,
  context.Exception.StackTrace,
  innerExceptionMessage
 }
};

context.ExceptionHandled = true;

LogException(context.Exception);
  }
  else
  {
var e = context.Exception;
if (!context.ExceptionHandled   // if unhandled, will be logged anyhow
 || RaiseErrorSignal(e)      // prefer signaling, if possible
 || IsFiltered(context))     // filtered?
 return;

LogException(e);
   }
    }

    private static bool RaiseErrorSignal(Exception e)
    {
        var context = HttpContext.Current;
        if (context == null)
            return false;
        var signal = ErrorSignal.FromContext(context);
        if (signal == null)
            return false;
        signal.Raise(e, context);
        return true;
    }

    private static bool IsFiltered(ExceptionContext context) 
    {
        var config = context.HttpContext.GetSection("elmah/errorFilter") 
                     as ErrorFilterConfiguration;

        if (config == null) 
            return false;

        var testContext = new ErrorFilterModule.AssertionHelperContext(
                                  context.Exception, HttpContext.Current);

        return config.Assertion.Test(testContext);
    }

    private static void LogException(Exception e)
    {
        var context = HttpContext.Current;
        ErrorLog.GetDefault(context).Log(new Error(e, context));
    }
  }
}

使用上述代码时,我在客户端上收到以下错误:

执行处理程序“System.Web.Mvc.HttpHandlerUtil+ServerExecuteHttpHandlerWrapper”的子请求时出错。

这告诉我 ELMAH 正在尝试返回 Error.aspx,这是一个强类型视图,而我没有向该视图提供正确的数据。

如何让 ELMAH 返回异常而不是调用 Error.aspx?我正在使用 jQuery 向服务器发送 AJAX 请求。

谢谢

汤姆

I'm using ELMAH to log errors. I'm extending HandleError attribute with my own that allows ELMAH to handle all errors. It's my understanding that ELMAH will return Error.aspx to the browser after the error has been logged. That's well and good except for Ajax calls. My client is expecting an Exception returned as JSON if something happens during an Ajax call. For AjaxRequests, how do you get ELMAH to return an exception instead of returning Error.aspx?

Here's how I'm extending HandleError:

public class ProspectorHandleErrorAttribute : HandleErrorAttribute
{
    public override void OnException(ExceptionContext context)
    {
   base.OnException(context);

   if (context.HttpContext.Request.IsAjaxRequest())
   {
   var innerExceptionMessage = (context.Exception.InnerException != null) ? context.Exception.InnerException.Message : String.Empty;

context.HttpContext.Response.StatusCode = (int)System.Net.HttpStatusCode.InternalServerError;
context.Result = new JsonResult
{
 JsonRequestBehavior = JsonRequestBehavior.AllowGet,
 Data = new
 {
  context.Exception.Message,
  context.Exception.StackTrace,
  innerExceptionMessage
 }
};

context.ExceptionHandled = true;

LogException(context.Exception);
  }
  else
  {
var e = context.Exception;
if (!context.ExceptionHandled   // if unhandled, will be logged anyhow
 || RaiseErrorSignal(e)      // prefer signaling, if possible
 || IsFiltered(context))     // filtered?
 return;

LogException(e);
   }
    }

    private static bool RaiseErrorSignal(Exception e)
    {
        var context = HttpContext.Current;
        if (context == null)
            return false;
        var signal = ErrorSignal.FromContext(context);
        if (signal == null)
            return false;
        signal.Raise(e, context);
        return true;
    }

    private static bool IsFiltered(ExceptionContext context) 
    {
        var config = context.HttpContext.GetSection("elmah/errorFilter") 
                     as ErrorFilterConfiguration;

        if (config == null) 
            return false;

        var testContext = new ErrorFilterModule.AssertionHelperContext(
                                  context.Exception, HttpContext.Current);

        return config.Assertion.Test(testContext);
    }

    private static void LogException(Exception e)
    {
        var context = HttpContext.Current;
        ErrorLog.GetDefault(context).Log(new Error(e, context));
    }
  }
}

When using the above code, I get the following error on the client:

Error executing child request for handler 'System.Web.Mvc.HttpHandlerUtil+ServerExecuteHttpHandlerWrapper'.

What this is telling me is that ELMAH is trying to return Error.aspx, which is a strongly typed view and I'm not providing the view the correct data.

How do I get ELMAH to return an exception instead of calling Error.aspx? I'm using jQuery for sending AJAX requests to server.

Thanks

Tom

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文