Подтвердить что ты не робот

Фильтр сжатия MVC 3, вызывающий искаженный вывод

Итак, у меня есть настраиваемый атрибут CompressAttribute, который настроен как глобальный фильтр в global.asax. Он использует отражение для проверки типа возврата текущего метода действий, и если он "ViewResult", он сжимает выходные данные с помощью GZip или Deflate. Он отлично работает, если только страница не выдает 500 Server Error. Если возникла ошибка, вместо отображения страницы ошибки .NET я получаю кучу этого:

`Я% &/м {JJT

По-видимому, он пытается кодировать страницу "Ошибка сервера 500", которая вызывает проблемы. Какой лучший способ справиться с этим?

Здесь код фильтра:

public override void OnActionExecuting(ActionExecutingContext filterContext)
        {
            MethodInfo actionMethodInfo = Common.GetActionMethodInfo(filterContext);
            if (GetReturnType(actionMethodInfo).ToLower() != "viewresult") return;

            HttpRequestBase request = filterContext.HttpContext.Request;

            string acceptEncoding = request.Headers["Accept-Encoding"];

            if (string.IsNullOrEmpty(acceptEncoding)) return;

            acceptEncoding = acceptEncoding.ToUpperInvariant();

            HttpResponseBase response = filterContext.HttpContext.Response;

            if (acceptEncoding.Contains("GZIP"))
            {
                response.AppendHeader("Content-encoding", "gzip");
                response.Filter = new WebCompressionStream(response.Filter, CompressionType.GZip);
            }
            else if (acceptEncoding.Contains("DEFLATE"))
            {
                response.AppendHeader("Content-encoding", "deflate");
                response.Filter = new WebCompressionStream(response.Filter, CompressionType.Deflate);
            }
        }
4b9b3361

Ответ 1

Итак, я смог решить эту проблему, очистив свойство Response.Filter в событии Application_Error:

public void Application_Error(object sender, EventArgs e)
{
    Response.Filter.Dispose();
}

Интересно, есть ли более правильный способ сделать это, хотя...

Ответ 2

Вы также можете решить эту проблему, добавив OnResultExecuting вместо OnActionExecuting. Это дает несколько преимуществ

  • Вы можете обнаружить результат действия, не прибегая к отражению.
  • OnResultExecuting не будет выполняться в исключительных случаях (MVC будет вызывать OnException, но не OnResultExecuting)

Что-то вроде этого:

public sealed class MyAttribute  : ActionFilterAttribute
{
    /// <summary>
    /// Called by MVC just before the result (typically a view) is executing.
    /// </summary>
    /// <param name="filterContext"></param>
    public override void OnResultExecuting(ResultExecutingContext filterContext)
    {
        var result = filterContext.Result;
        if (result is ViewResultBase)
        {
            var response = filterContext.HttpContext.Response;

            // Check your request parameters and attach filter.
        }
    }

Ответ 3

У меня была такая же проблема в asp.net mvc 1.0, просматривающем страницу, в которой содержался RenderAction (из сборки фьючерсов). По-видимому, проблема заключалась в том, что ответ дважды кодировался. Мне пришлось создать фильтр действий для этих дочерних действий, чтобы флаг был установлен в коллекции DataTokens RouteData. Затем мне пришлось модифицировать фильтр сжатия, чтобы он возвращался в случае установки флага. Мне также пришлось иметь дело с порядком выполнения фильтров. Возможно, это может помочь, проверьте, вызван ли фильтр сжатия более одного раза, когда страница с ошибкой поднята.

Ответ 4

Принятый ответ не будет работать, если что-либо уже было записано на выход.

Вместо удаления фильтра вы можете убедиться, что заголовки сохраняются на месте:

 protected void Application_PreSendRequestHeaders()
{
    // ensure that if GZip/Deflate Encoding is applied that headers are set
    // also works when error occurs if filters are still active
    HttpResponse response = HttpContext.Current.Response;
    if (response.Filter is GZipStream && response.Headers["Content-encoding"] != "gzip")
        response.AppendHeader("Content-encoding", "gzip");
    else if (response.Filter is DeflateStream && response.Headers["Content-encoding"] != "deflate")
        response.AppendHeader("Content-encoding", "deflate");
}