Страницы ошибок не найдены, выбрасывание ошибки ELMAH с помощью пользовательских страниц ошибок

Я внес некоторые изменения в Global.asax, чтобы показать пользовательские страницы ошибок (403, 404 и 500). Здесь код:

    public class MvcApplication : System.Web.HttpApplication
    {
        protected void Application_Start()
        {
            AreaRegistration.RegisterAllAreas();

            WebApiConfig.Register(GlobalConfiguration.Configuration);
            //FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
            RouteConfig.RegisterRoutes(RouteTable.Routes);
            BundleConfig.RegisterBundles(BundleTable.Bundles);
        }

        protected void Application_Error(object sender, EventArgs e)
        {
            if (Context.IsCustomErrorEnabled)
            {
                ShowCustomErrorPage(Server.GetLastError());
            }
        }

        private void ShowCustomErrorPage(Exception exception)
        {
            HttpException httpException = exception as HttpException;
            if (httpException == null)
            {
                httpException = new HttpException(500, "Internal Server Error", exception);
            }

            Response.Clear();
            RouteData routeData = new RouteData();
            routeData.Values.Add("controller", "Error");
            routeData.Values.Add("fromAppErrorEvent", true);

            switch (httpException.GetHttpCode())
            {
                case 403:
                    routeData.Values.Add("action", "AccessDenied");
                    break;

                case 404:
                    routeData.Values.Add("action", "NotFound");
                    break;

                case 500:
                    routeData.Values.Add("action", "ServerError");
                    break;

                default:
                    routeData.Values.Add("action", "DefaultError");
                    routeData.Values.Add("httpStatusCode", httpException.GetHttpCode());
                    break;
            }

            Server.ClearError();

            IController controller = new ErrorController();
            controller.Execute(new RequestContext(new HttpContextWrapper(Context), routeData));
        }
    }

Я также добавил следующее в свой Web.Config:

<customErrors mode="On">
    <!-- There is custom handling of errors in Global.asax -->
</customErrors>

Собственные страницы ошибок отображаются правильно, и ELMAH корректно регистрирует ошибку, которая была (целенаправленно) выбрана. Но ELMAH также обнаруживает и регистрирует дополнительную ошибку:

System.InvalidOperationException: The view 'Error' or its master was not found or no view engine supports the searched locations. The following locations were searched: ~/Views/account/Error.aspx ~/Views/account/Error.ascx ~/Views/Shared/Error.aspx ~/Views/Shared/Error.ascx ~/Views/account/Error.cshtml ~/Views/account/Error.vbhtml ~/Views/Shared/Error.cshtml ~/Views/Shared/Error.vbhtml

Мои первые инстинкты привели меня к отключению глобального HandleErrorAttribute в конфигурации фильтра. И аналогичные SO-вопросы, такие как проблема MVC с пользовательскими страницами ошибок, привели меня к тому, что мои подозрения были правильными. Но даже после отключения глобального HandleErrorAttribute я все еще получаю сообщение об ошибке, что невозможно увидеть ошибку Error! Что дает? Моя единственная другая догадка заключается в том, что мой базовый контроллер получен из System.Web.Mvc.Controller. Я попытался изучить источник, чтобы увидеть, применяется ли HandleErrorAttribute к System.Web.Mvc.Controller, но ничего не может получить...

UPDATE: Я попробовал переопределить мой базовый контроллер, чтобы отмечать исключения, которые обрабатываются следующим образом:

protected override void OnException(ExceptionContext filterContext)
{
    filterContext.ExceptionHandled = true;
    base.OnException(filterContext);
}

но это не решило проблему.

UPDATE2: Я поместил файл Error.aspx в общие представления, чтобы посмотреть, что произойдет. Когда он там, ELMAH регистрирует принудительное исключение, а затем общее представление подано вверх - оно никогда не достигает Application_Error().... не слишком уверенно, что с ним делать.

Ответ 1

Наконец, он работал над моим удовлетворением...

В пакете Elmah.Mvc применяется скрытый обработчик ошибок. Я отключил это, добавив следующую строку в web.config <appSettings> (значение было установлено как "false" по умолчанию из установки nuget)

<add key="elmah.mvc.disableHandleErrorFilter" value="true" />

Итак, теперь мои ошибки распространяются до Application_Error и регистрируются Elmah, минуя фильтр Elmah, и отображают правильную страницу с ошибкой (а не ту, что находится в/shared/error.cshtml)

Ответ 2

Если вы работаете в интегрированном режиме IIS 7, вам нужно добавить Response.TrySkipIisCustomErrors = true; в Application_Error. В противном случае IIS будет перенаправлять клиента на страницу пользовательских ошибок, несмотря на все, что вы делаете в коде.

Дополнительную информацию смотрите здесь: http://www.west-wind.com/weblog/posts/2009/Apr/29/IIS-7-Error-Pages-taking-over-500-Errors

Изменить: вот тело моего Application_Error:

if (HttpContext.Current != null)
{
    Server.ClearError();
    Response.TrySkipIisCustomErrors = true;
    RouteData data = new RouteData();
    data.Values.Add("controller", "Error");
    data.Values.Add("action", "Error");
    IController controller = new MyApp.Controllers.ErrorController();
    controller.Execute(new RequestContext(new HttpContextWrapper(Context), data));
}