5

我避免使用默认的 ASP.NET 重定向错误的方法(正如许多人所做的那样)。干净的 AJAX 代码和 SEO 是其中的原因。

但是,我是用下面的方法来做的,看来我可能会HttpContext.Current.Items在转帐中输掉?

<httpErrors errorMode="Custom" existingResponse="Replace">
    <remove statusCode="401" />
    <remove statusCode="403" />
    <remove statusCode="404" />
    <remove statusCode="500" />
    <error statusCode="401" responseMode="ExecuteURL" path="/Account/SignIn" />
    <error statusCode="403" responseMode="ExecuteURL" path="/Site/Forbidden" />
    <error statusCode="404" responseMode="ExecuteURL" path="/Site/NotFound" />
    <error statusCode="500" responseMode="ExecuteURL" path="/Site/Error" />
</httpErrors>

我以为它只是在幕后表演Server.Transfer(),据我所知保留Items。(参见:HttpContext.Current.Items 的范围http://weblog.west-wind.com/posts/2010/Jan/20/HttpContextItems-and-ServerTransferExecute

但我也在Items“ExecuteURL”之前捕获了一些东西,并在传输(或任何它是)之后检索/输出它,它似乎消失了。我已经看到它进入Items集合,我看到Count提升到 5,然后当检索到值时,集合中只有 2 个项目。

到底是怎么回事?


如果您想更多地了解我在做什么并推荐一个替代实现,我愿意接受。我正在使用它以一种不受竞争条件的方式将 ELMAH 错误 ID 推送到 ViewModel 中。(即,我要替换的常见解决方法是仅显示最近的错误。)这是我的代码:

全球.asax

protected void ErrorLog_Logged(object sender, ErrorLoggedEventArgs args) {
    ElmahSupplement.CurrentId = args.Entry.Id;
}

void ErrorLog_Filtering(object sender, ExceptionFilterEventArgs e) {
    if (ElmahSupplement.IsNotFound(e.Exception)) {
        ElmahSupplement.LogNotFound((e.Context as HttpContext).Request);
        e.Dismiss();
    }
}

站点控制器.cs

public virtual ActionResult Error() {
    Response.StatusCode = 500;
    return View(MVC.Site.Views.Error, ElmahSupplement.CurrentId);
}

ElmahSupplement.cs

public class ElmahSupplement {
    // TODO: This is a rather fragile way to access this info
    private static readonly Guid contextId = new Guid("A41A67AA-8966-4205-B6C1-14128A653F21");

    public static string CurrentId {
        get { 
            return
                // Elmah 1.2 will fail to log when enumerating form values that raise RequestValidationException (angle brackets)
                // https://code.google.com/p/elmah/issues/detail?id=217
                // So this id could technically be empty here
                (HttpContext.Current.Items[contextId] as string);
        }
        set {
            HttpContext.Current.Items[contextId] = value;
        }
    }

    public static void LogNotFound(HttpRequest request) {
        var context = RepositoryProxy.Context;
        context.NotFoundErrors.Add(new NotFoundError {
            RecordedOn = DateTime.UtcNow,
            Url = request.Url.ToString(),
            ClientAddress = request.UserHostAddress,
            Referrer = request.UrlReferrer == null ? "" : request.UrlReferrer.ToString()
        });
        context.SaveChanges();
    }

    public static bool IsNotFound(Exception e) {
        HttpException he = e as HttpException;
        return he != null && he.GetHttpCode() == 404;
    }
}
4

2 回答 2

1

如此处所述,ExecuteURL 生成两个请求:第一个引发异常,第二个生成错误响应。

由于 Context.Items 在请求之间被清除,您的代码总是会看到生成的第二个请求,因此项目之间的差异。

尝试帖子中的建议:使用 system.web > customErrors 和 redirectMode="ResponseRewrite" 代替。

于 2014-04-04T17:30:02.180 回答
0

我跟踪并确定了以下内容。有些是松散推断的。

CustomErrorModule(在 IIS 模块堆栈中)接收 SEND_RESPONSE 通知。

HttpStatus 为 500,因此它克隆上下文,设置新 URL(根据匹配的自定义错误规则),并在此上下文上执行请求(请参阅ExecuteRequest)。

HttpContext.Items每个文档的目的是:

获取可用于在 HTTP 请求期间在 IHttpModule 接口和 IHttpHandler 接口之间组织和共享数据的键/值集合。

批判地看待这个函数定义,当然只有“HTTP请求”。但是,Items字典本身很可能是一个以 HttpContext 为键的字典中的项,它是这个正在执行的子请求中的唯一(克隆)引用。跟踪显示为此运行的完整管道(所有模块,例如重复身份验证)ExecuteURL,因此当然需要此隔离上下文。

对于非托管代码,GetParentContext是微不足道的。但是,从托管代码中,此层次结构不可用。所以,我没有办法找回原来的Items.

作为替代解决方案,利用 Global.asax 变量可能是有用的,因为我的测试显示子请求共享一个ApplicationInstance,但我不确定客户端对此的访问一定是顺序的。

另一种可能更好的方法是避免重新运行整个管道;永远不要退出 MVC 处理程序(例如Controller.OnExceptionTransferToAction)。但是,这会阻止为错误页面配置实现单点真相,因为错误也可能在 MVC 的意识之外引发。

于 2014-04-08T20:06:32.980 回答