14

我在共享主机上部署的 ASP.NET MVC 应用程序上遇到自定义错误问题。我创建了一个 ErrorController 并将以下代码添加到 Global.asax 以捕获未处理的异常,记录它们,然后将控制权转移到 ErrorController 以显示自定义错误。此代码取自此处

protected void Application_Error(object sender, EventArgs e)
{
    Exception ex = Server.GetLastError();
    Response.Clear();

    HttpException httpEx = ex as HttpException;
    RouteData routeData = new RouteData();
    routeData.Values.Add("controller", "Error");

    if (httpEx == null)
    {
        routeData.Values.Add("action", "Index");
    }
    else
    {
        switch (httpEx.GetHttpCode())
        {
            case 404:
                routeData.Values.Add("action", "HttpError404");
                break;
            case 500:
                routeData.Values.Add("action", "HttpError500");
                break;
            case 503:
                routeData.Values.Add("action", "HttpError503");
                break;
            default:
                routeData.Values.Add("action", "Index");
                break;
        }
    }

    ExceptionLogger.LogException(ex); // <- This is working. Errors get logged

    routeData.Values.Add("error", ex);
    Server.ClearError();
    IController controller = new ErrorController();
    // The next line doesn't seem to be working
    controller.Execute(new RequestContext(new HttpContextWrapper(Context), routeData));
}

Application_Error 肯定会触发,因为日志记录工作正常,但我没有显示我的自定义错误页面,而是获得了 Go Daddy 通用错误页面。从博客文章的标题中,我注意到它使用了 MVC 框架的 Release Candidate 2。1.0 中是否发生了一些变化,导致最后一行代码不起作用?像往常一样,它在我的机器上运行良好

任何建议将不胜感激。

编辑:忘了提到我已经尝试了 Web.config 中 customErrors 模式的所有 3 种可能性(Off、On 和 RemoteOnly)。无论此设置如何,结果都相同。

编辑 2:我也尝试过在控制器类上使用和不使用 [HandleError] 装饰。

更新:我已经弄清楚并修复了 404。Go Daddy 的主机控制中心的设置面板中有一个部分可以控制 404 行为,并且默认显示它们的通用页面,显然这会覆盖任何 Web.config 设置。所以我的自定义 404 页面现在按预期显示。但是,500s 和 503s 仍然无法正常工作。如果 Sql Server 抛出异常,我在 HomeController 中有代码来获取内容的静态文本版本,如下所示:

public ActionResult Index()
{
    CcmDataClassesDataContext dc = new CcmDataClassesDataContext();

    // This might generate an exception which will be handled in the OnException override
    HomeContent hc = dc.HomeContents.GetCurrentContent();

    ViewData["bodyId"] = "home";
    return View(hc);
}

protected override void OnException(ExceptionContext filterContext)
{
    // Only concerned here with SqlExceptions so an HTTP 503 message can
    // be displayed in the Home View. All others will bubble up to the
    // Global.asax.cs and be handled/logged there.
    System.Data.SqlClient.SqlException sqlEx =
        filterContext.Exception as System.Data.SqlClient.SqlException;
    if (sqlEx != null)
    {
        try
        {
            ExceptionLogger.LogException(sqlEx);
        }
        catch
        {
            // couldn't log exception, continue without crashing
        }

        ViewData["bodyId"] = "home";
        filterContext.ExceptionHandled = true;
        HomeContent hc = ContentHelper.GetStaticContent();
        if (hc == null)
        {
            // Couldn't get static content. Display friendly message on Home View.
            Response.StatusCode = 503;
            this.View("ContentError").ExecuteResult(this.ControllerContext);
        }
        else
        {
            // Pass the static content to the regular Home View
            this.View("Index", hc).ExecuteResult(this.ControllerContext);
        }
    }
}

这是尝试获取静态内容的代码:

public static HomeContent GetStaticContent()
{
    HomeContent hc;

    try
    {
        string path = Configuration.CcmConfigSection.Config.Content.PathToStaticContent;
        string fileText = File.ReadAllText(path);
        string regex = @"^[^#]([^\r\n]*)";
        MatchCollection matches = Regex.Matches(fileText, regex, RegexOptions.Multiline);
        hc = new HomeContent
            {
                ID = Convert.ToInt32(matches[0].Value),
                Title = matches[1].Value,
                DateAdded = DateTime.Parse(matches[2].Value),
                Body = matches[3].Value,
                IsCurrent = true
            };
    }
    catch (Exception ex)
    {
        try
        {
            ExceptionLogger.LogException(ex);
        }
        catch
        {
            // couldn't log exception, continue without crashing
        }
        hc = null;
    }

    return hc;
}

我已经验证,如果我更改连接字符串以生成 SqlException,代码会正确记录错误,然后抓取并显示静态内容。但是,如果我还更改了 Web.config 中静态文本文件的路径以测试 503 版本的主视图,那么我得到的是一个除了“服务不可用”之外什么都没有的页面。就是这样。没有具有网站外观的自定义 503 消息。

有没有人对可能有帮助的代码改进有任何建议?向 HttpResponse 添加不同的标头会有所帮助吗?还是 Go Daddy 强硬地劫持了 503?

4

2 回答 2

29

我找到了解决方案,而且非常简单。原来问题实际上出在 IIS7 中。在 Visual Studio 中调试此问题时,我看到了 HttpResponse 对象的一个​​我以前没有注意到的属性:

public bool TrySkipIisCustomErrors { get; set; }

这将我引向离我最近的搜索引擎,该搜索引擎找到了 Rick Strahl 的一篇很棒的博文,以及angeringpets.com上的另一篇博文,以及SO 上的这个问题。这些链接比我能更好地解释血淋淋的细节,但是 Rick 帖子中的这句话很好地捕捉到了它:

真正的混乱发生在这里,因为错误被 ASP.NET 捕获,但最终仍然由 IIS 处理,它查看 500 状态代码并返回库存 IIS 错误页面。

这种行为似乎也特定于集成模式下的 IIS7。来自msdn

在 IIS 7.0 中以经典模式运行时,TrySkipIisCustomErrors 属性默认值为 true。在集成模式下运行时,TrySkipIisCustomErrors 属性默认值为 false。

所以基本上我最终要做的就是Response.TrySkipIisCustomErrors = true;在任何设置Response.StatusCode为 500 或 503 的代码之后添加,现在一切都按设计运行。

于 2009-11-12T02:13:18.783 回答
0

我在 GoDaddy 上托管了一个 ASP.NET MVC 站点,并且还遇到了处理自定义错误页面的问题。通过反复试验,我发现 GoDaddy 在 HTTP 级别拦截错误。

例如,任何返回 HTTP 状态代码 404 的页面都会导致 GoDaddy 的自定义错误页面被接管。最终,我更改了自定义错误页面以返回 200 状态,并且与 404 相关的问题消失了。我的 HTML 是一样的,只是需要更改 HTTP 状态。

诚然,我从未尝试过对 503 状态响应做同样的事情,但同样的缓解措施可能会奏效。如果您从返回 503 状态更改为返回 200 状态,问题会消失吗?

请注意,如果您执行此解决方法,您将希望阻止搜索引擎索引您的错误页面,一旦返回 200 状态将与常规页面无法区分(从搜索引擎的角度来看)。所以一定要添加一个 META ROBOTS 标签来防止你的错误页面被索引,例如

<META NAME="ROBOTS" CONTENT="NOINDEX">

这种方法的缺点可能是您的页面可能会从 Google 中删除,这绝对不是一件好事!

更新:因此,此外,您还可以检测用户代理是否为爬虫,如果是爬虫,则返回 503,如果不是爬虫,则返回 200。有关如何检测的信息,请参阅此博客文章爬虫。是的,我知道将不同的内容返回给爬虫和用户是一个 SEO 禁忌,但我已经在几个网站上这样做了,到目前为止没有任何不良影响,所以我不确定这是一个多大的问题。

两种方法(META ROBOTS 和机器人检测)可能是你最好的选择,以防任何古怪的爬虫从机器人检测器中溜走。

于 2009-11-10T23:48:02.880 回答