Як викликається Error.cshtml в ASP.NET MVC?


78

Я прочитав десяток подібних запитань щодо StackOverflow, але, здається, не можу зрозуміти цього. Що стосується вузла користувацьких помилок у web.config та HandleErrorAttribute, як взагалі викликається Error.cshtml? Зрештою, відповідь на це запитання може бути відповіддю на одне з тих кількох питань, що вже існують стосовно обробки помилок ASP.NET MVC. Але справа в тому, що я не знаю, який саме.

Відповіді:


91

Усередині вашого Global.asax у вас є такий спосіб:

public static void RegisterGlobalFilters(GlobalFilterCollection filters)
{
    filters.Add(new HandleErrorAttribute());
}

Це реєструє HandleErrorAttribute як глобальний фільтр дій. Це означає, що цей обробник автоматично застосовується до всіх дій контролера. Тепер давайте подивимось, як реалізований цей атрибут, переглянувши вихідний код:

[SuppressMessage("Microsoft.Performance", "CA1813:AvoidUnsealedAttributes", Justification = "This attribute is AllowMultiple = true and users might want to override behavior.")]
[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, Inherited = true, AllowMultiple = true)]
public class HandleErrorAttribute : FilterAttribute, IExceptionFilter {

    private const string _defaultView = "Error";

    private readonly object _typeId = new object();

    private Type _exceptionType = typeof(Exception);
    private string _master;
    private string _view;

    public Type ExceptionType {
        get {
            return _exceptionType;
        }
        set {
            if (value == null) {
                throw new ArgumentNullException("value");
            }
            if (!typeof(Exception).IsAssignableFrom(value)) {
                throw new ArgumentException(String.Format(CultureInfo.CurrentCulture,
                    MvcResources.ExceptionViewAttribute_NonExceptionType, value.FullName));
            }

            _exceptionType = value;
        }
    }

    public string Master {
        get {
            return _master ?? String.Empty;
        }
        set {
            _master = value;
        }
    }

    public override object TypeId {
        get {
            return _typeId;
        }
    }

    public string View {
        get {
            return (!String.IsNullOrEmpty(_view)) ? _view : _defaultView;
        }
        set {
            _view = value;
        }
    }

    public virtual void OnException(ExceptionContext filterContext) {
        if (filterContext == null) {
            throw new ArgumentNullException("filterContext");
        }
        if (filterContext.IsChildAction) {
            return;
        }

        // If custom errors are disabled, we need to let the normal ASP.NET exception handler
        // execute so that the user can see useful debugging information.
        if (filterContext.ExceptionHandled || !filterContext.HttpContext.IsCustomErrorEnabled) {
            return;
        }

        Exception exception = filterContext.Exception;

        // If this is not an HTTP 500 (for example, if somebody throws an HTTP 404 from an action method),
        // ignore it.
        if (new HttpException(null, exception).GetHttpCode() != 500) {
            return;
        }

        if (!ExceptionType.IsInstanceOfType(exception)) {
            return;
        }

        string controllerName = (string)filterContext.RouteData.Values["controller"];
        string actionName = (string)filterContext.RouteData.Values["action"];
        HandleErrorInfo model = new HandleErrorInfo(filterContext.Exception, controllerName, actionName);
        filterContext.Result = new ViewResult {
            ViewName = View,
            MasterName = Master,
            ViewData = new ViewDataDictionary<HandleErrorInfo>(model),
            TempData = filterContext.Controller.TempData
        };
        filterContext.ExceptionHandled = true;
        filterContext.HttpContext.Response.Clear();
        filterContext.HttpContext.Response.StatusCode = 500;

        // Certain versions of IIS will sometimes use their own error page when
        // they detect a server error. Setting this property indicates that we
        // want it to try to render ASP.NET MVC's error page instead.
        filterContext.HttpContext.Response.TrySkipIisCustomErrors = true;
    }
}

Вихідний код містить коментарі і є більш ніж зрозумілим. Перше, що він перевіряє, чи увімкнули ви власні помилки у своєму web.config (тобто <customErrors mode="On">). Якщо у вас цього немає, це нічого не робить => YSOD. Якщо ви ввімкнули власні помилки, він відображає подання Помилка, передаючи йому модель, що містить стек стежка винятків та іншу корисну інформацію.


Я все ще досить новачок в ASP.NET MVC, але я думав, що кожен подання повинен відповідати дії якогось контролера. У чому тут контролер та дія, і як він знає, щоб перейти до спільного доступу для перегляду? Схоже, цей механізм тут для переривання та обміну в режимі помилок.
LJM

3
Так, кожен вигляд повинен відповідати дії контролера. Дія контролера в цьому випадку - це та, яка виконується і створює виняток. Це може бути будь-яка дія контролера. Цей виняток перехоплюється глобальним фільтром дій (який у цьому випадку є фільтром винятків) і відображає подання Помилка. Оскільки виняток потрапляє всередину дії контролера, ця дія ніколи не повертається, вона просто зупиняє виконання на цих етапах і обробляє виконання обробнику помилок, який, у свою чергу, робить подання.
Дарін Димитров

12
Чи змінилося це в MVC 5.1? Мені здається, що HandleErrorAttribute зареєстровано за замовчуванням (без того, щоб ми додавали його до списку фільтрів у RegisterGlobalFilters), і нам більше не потрібно вмикати власні помилки.
Johnny Oshika,

1
Чи змінилася відповідь у MVC 5? Я не отримую сторінку помилки. Все, що я бачу, - це YSOD. У мене є <customError mode = "On">. Чого ще не вистачає?
Ден
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.