ASP.NET MVC "Кеширование пончиков" и TempData - PullRequest
5 голосов
/ 26 марта 2009

Есть ли решение для доступа к свойству TempData в методе HttpResponseBase.WriteSubstitution ()

Это не работает:

<%= Response.WriteSubstitution(x => Html.Encode(TempData["message"].ToString())) %>

Но это работает:

<%= Response.WriteSubstitution(x => DateTime.Now.ToString()) %>

Проблема заключается в обработке запросов для однократно кэшированных страниц. Согласно http://msdn.microsoft.com/en-us/library/system.web.httpresponse.writesubstitution.aspx:

При первом запросе к странице WriteSubstitution вызывает делегат HttpResponseSubstitutionCallback для создания выходных данных. Затем он добавляет буфер замены к ответу, который сохраняет делегат для вызова в будущих запросах. Наконец, он ухудшает кешируемость на стороне клиента с общедоступной до серверной, обеспечивая возможность повторного вызова делегата в будущем при вызове страницы, не кэшируя на клиенте.

Другими словами, делегат не имеет доступа к свойству Session (SessionStateTempDataProvider хранит TempData в сеансе), потому что нет «нормального» жизненного цикла запроса. Насколько я понимаю, он обрабатывается при событии HttpApplication.ResolveRequestCache / HttpApplication.PostResolveRequestCache, но текущее состояние получается при событии HttpApplication.AcquireRequestState (http://msdn.microsoft.com/en-us/library/ms178473.aspx)

Может быть, мне нужен какой-то "продвинутый пользовательский TempDataProvider" :) Есть идеи?

1 Ответ

5 голосов
/ 29 марта 2009

Я нашел решение:

Основная идея заключается в сохранении копии TempData в кэше и получении ее при каждом запросе. Решение представляет собой комбинацию пользовательского TempDataProvider и простого модуля http. Плюс есть пара помощников и статические классы.

Вот код:

CustomTempDataProvider:

using System;
using System.Collections.Generic;
using System.Web;
using System.Web.Caching;
using System.Web.Mvc;

public class CustomTempDataProvider : SessionStateTempDataProvider, IHttpModule
{
    public void Init(HttpApplication application)
    {
        application.BeginRequest += new EventHandler(application_BeginRequest);
    }

    void application_BeginRequest(object sender, EventArgs e)
    {
        var httpContext = HttpContext.Current;
        var tempData = httpContext.Cache[TempDataKey] ?? new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
        httpContext.Items.Add("TempData", tempData);
        httpContext.Cache.Remove(TempDataKey);
    }

    public override void SaveTempData(ControllerContext controllerContext,
        IDictionary<string, object> values)
    {
        HttpContext.Current.Cache.Insert(TempDataKey, values, null, DateTime.Now.AddMinutes(5), Cache.NoSlidingExpiration, CacheItemPriority.NotRemovable, null);
        base.SaveTempData(controllerContext, values);
    }

    public static string TempDataKey
    {
        get
        {
            string sessionID = "0";
            var httpContext = HttpContext.Current;
            if(httpContext.Session != null)
            {
                sessionID = httpContext.Session.SessionID;
            }
            else if (httpContext.Request.Cookies["ASP.NET_SessionId"] != null)
            {
                sessionID = httpContext.Request.Cookies["ASP.NET_SessionId"].Value;
            }
            return "TempData-For-Session-" + sessionID;
        }
    }

    public void Dispose()
    {
    }
}

Зарегистрируйте его в web.config:

<?xml version="1.0" encoding="UTF-8"?>
<configuration>
    <system.web>
        <httpModules>
            <add name="CustomTempDataProvider" type="CustomTempDataProvider" />
        </httpModules>
    </system.web>
    <system.webServer>
        <modules>
            <remove name="CustomTempDataProvider" />
            <add name="CustomTempDataProvider" type="CustomTempDataProvider" />
        </modules>
    </system.webServer>
</configuration>

CustomControllerFactory:

using System.Web.Routing;
using System.Web.Mvc;

public class CustomControllerFactory : DefaultControllerFactory
{
    public override IController CreateController(
        RequestContext requestContext, string controllerName)
    {
        var controller = (Controller)base.CreateController(requestContext, controllerName);
        controller.TempDataProvider = new CustomTempDataProvider();
        return controller;
    }
}

зарегистрируйте его в Global.asax:

protected void Application_Start()
{
    RegisterRoutes(RouteTable.Routes);

    ControllerBuilder.Current.SetControllerFactory(typeof(CustomControllerFactory));
}

Статический класс для доступа к TempData:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;

public static class CustomTempData
{
    public static object Get(string key)
    {
        var tempData = HttpContext.Current.Items["TempData"] as IDictionary<string, object>;
        var item = tempData.FirstOrDefault(x => x.Key == key).Value ?? String.Empty;
        return item;
    }
}

Помощник для замены после кэширования:

using System;
using System.Web;
using System.Web.Mvc;

public static class Html
{
    public delegate object MvcResponseSubstitutionCallback(HttpContextBase context);

    public static object MvcResponseSubstitute(this HtmlHelper html, MvcResponseSubstitutionCallback callback)
    {
        html.ViewContext.HttpContext.Response.WriteSubstitution(
            context =>
                HttpUtility.HtmlEncode(
                    (callback(new HttpContextWrapper(context)) ?? String.Empty).ToString()
                )
            );

        return null;
    }
}

Теперь это успешно работает:

<h3><%= Html.MvcResponseSubstitute(context => CustomTempData.Get("message")) %></h3>

Если вы понимаете русский язык, прочитайте это

Надеюсь, это поможет

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...