I'm trying to implement a MongoDB / Memory combined Output Cache Provider to use with MVC4. Here is my initial implementation:
public class CustomOutputCacheProvider : OutputCacheProvider
{
public override object Get(string key)
{
FileLogger.Log(key);
return null;
}
public override object Add(string key, object entry, DateTime utcExpiry)
{
return entry;
}
public override void Set(string key, object entry, DateTime utcExpiry)
{
}
public override void Remove(string key)
{
}
}
And my web config entry:
<caching>
<outputCache defaultProvider="CustomOutputCacheProvider">
<providers>
<add name="CustomOutputCacheProvider" type="MyApp.Base.Mvc.CustomOutputCacheProvider" />
</providers>
</outputCache>
</caching>
And the usage within HomeController:
[OutputCache(Duration = 15)]
public ActionResult Index()
{
return Content("Home Page");
}
My problem is, when I check the logfile for the keys that are requested, I see not only the request to home controller, but all other paths as well:
a2/ <-- should only log this entry
a2/test
a2/images/test/50115c53/1f37e409/4c7ab27d/50115c531f37e4094c7ab27d.jpg
a2/scripts/jquery-1.7.2.min.js
I've figured that I shouldn't set the CustomOutputCacheProvider as the defaultProvider in Web.Config, what I couldn't figure out is how to specify the cache provider that I want to use for a specific controller action.
With Asp.Net Web Pages you can accomplish it by using <%@ OutputCache Duration="60" VaryByParam="None" providerName="DiskCache" %>
at the top of the page, but for MVC the only solution I could find is to override HttpApplication.GetOutputCacheProviderName Method in Global.asax.
Is there a more elegant way to accomplish this by using the [OutputCache] attribute?
In ASP.NET MVC, there is an OutputCache filter attribute that you can apply and this is the same concept as output caching in web forms. The output cache enables you to cache the content returned by a controller action. Output caching basically allows you to store the output of a particular controller in the memory.
By default, when you use the [OutputCache] attribute, content is cached in three locations: the web server, any proxy servers, and the web browser. You can control exactly where content is cached by modifying the Location property of the [OutputCache] attribute. By default, the Location property has the value Any.
The output cache is located on the Web server where the request was processed. This value corresponds to the Server enumeration value. The output cache can be stored only at the origin server or at the requesting client. Proxy servers are not allowed to cache the response.
The output cache enables you to cache the content returned by a controller action. That way, the same content does not need to be generated each and every time the same controller action is invoked. Imagine, for example, that your ASP.NET MVC application displays a list of database records in a view named Index.
Is there a more elegant way to set the OutputCacheProvider using the [OutputCache] attribute?
I think the answer is no, (well not with the current mvc4 release) since there is no relationship between implementing a custom OutputCacheProvider
and decorating an action with the OutputCache
attribute.
As you discovered by implementing the custom provider and logging in the Get method you see every request made to the web server. If you were to remove the OutputCache
attribute from all your actions you will still see every request in out log file. I thought the answer for this ASP.NET MVC hits outputcache for every action was pretty useful to confirm that.
Since it looks like you only want to implement one output-cache provider then I think your only option is to not set the default provider and continue to override the GetOutputCacheProviderName
implementation (as you have already mentioned). Perhaps something like this to exclude all Content, Images and Scripts
public override string GetOutputCacheProviderName(HttpContext context)
{
string absolutePath = context.Request.Url.AbsolutePath;
if (absolutePath.StartsWith("/Content/", StringComparison.CurrentCultureIgnoreCase)
|| absolutePath.StartsWith("/Scripts/", StringComparison.CurrentCultureIgnoreCase)
|| absolutePath.StartsWith("/Images/", StringComparison.CurrentCultureIgnoreCase))
return base.GetOutputCacheProviderName(context);
return "CustomOutputCacheProvider";
}
If you need to implement more than one output-cache provider then I guess you'll have to implement a helper to give you the correct provider name. But here's an example where I've resolved the routing data for you; where as the prev example looked directly at the url.
public override string GetOutputCacheProviderName(HttpContext context)
{
RouteCollection rc = new RouteCollection();
MvcApplication.RegisterRoutes(rc);
RouteData rd = rc.GetRouteData(new HttpContextWrapper(HttpContext.Current));
if (rd == null)
return base.GetOutputCacheProviderName(context);
var controller = rd.Values["controller"].ToString();
var action = rd.Values["action"].ToString();
if (controller.Equals("Content", StringComparison.CurrentCultureIgnoreCase)
|| controller.Equals("Scripts", StringComparison.CurrentCultureIgnoreCase)
|| controller.Equals("Images", StringComparison.CurrentCultureIgnoreCase))
return base.GetOutputCacheProviderName(context);
if (controller.Equals("Account", StringComparison.CurrentCultureIgnoreCase))
return "AccountOutputCacheProvider";
if (controller.Equals("Something", StringComparison.CurrentCultureIgnoreCase))
return controller + "OutputCacheProvider";
return "CustomOutputCacheProvider";
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With