¿Cómo se llama Error.cshtml en ASP.NET MVC?

78

He leído una docena de preguntas similares en StackOverflow, pero parece que no puedo entender esto. Con respecto al nodo de errores personalizados en web.config y HandleErrorAttribute, ¿cómo se llama a Error.cshtml? En última instancia, la respuesta a esta pregunta puede ser la respuesta a una de esas varias preguntas que ya existen sobre el manejo de errores de ASP.NET MVC. Pero, el hecho es que no sé cuál.

LJM
fuente

Respuestas:

91

Dentro de su Global.asax tiene el siguiente método:

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

Esto registra el HandleErrorAttribute como filtro de acción global. Esto significa que este controlador se aplica automáticamente a todas las acciones del controlador. Ahora echemos un vistazo a cómo se implementa este atributo observando el código fuente:

[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;
    }
}

El código fuente contiene comentarios y es más que autoexplicativo. Lo primero que comprueba es si ha habilitado errores personalizados en su web.config (es decir <customErrors mode="On">). Si no lo ha hecho, no hace nada => YSOD. Si ha habilitado los errores personalizados, la vista Error muestra un modelo que contiene el seguimiento de la pila de excepciones y otra información útil.

Darin Dimitrov
fuente
Todavía soy bastante nuevo en ASP.NET MVC, pero pensé que cada Vista tenía que corresponder a la acción de algún controlador. ¿Cuál es el controlador y la acción aquí y cómo sabe ir a Compartido para la vista? Parece que este mecanismo está aquí para interrumpir e intercambiar en la vista Error.
LJM
3
Sí, cada vista debe corresponder a una acción del controlador. La acción del controlador en este caso es la que se está ejecutando y que arroja una excepción. Podría ser cualquier acción del controlador. Esta excepción es interceptada por el filtro de acción global (que en este caso es un filtro de excepción) y muestra la vista Error. Dado que la excepción se lanza dentro de la acción del controlador, esta acción nunca regresa, simplemente deja de ejecutarse en estas etapas y maneja la ejecución al controlador de errores que a su vez muestra la vista.
Darin Dimitrov
12
¿Ha cambiado esto en MVC 5.1? Me parece que HandleErrorAttribute está registrado de forma predeterminada (sin que tengamos que agregarlo a la lista de filtros en RegisterGlobalFilters) y ya no necesitamos habilitar errores personalizados.
Johnny Oshika
1
¿Ha cambiado la respuesta en MVC 5? No obtengo la página de error. Todo lo que veo es el YSOD. Tengo el <customError mode = "On">. ¿Qué más falta?
Dan