Output caching for an ApiController (MVC4 Web API)
Asked Answered
E

5

51

I'm trying to cache the output of an ApiController method in Web API.

Here's the controller code:

public class TestController : ApiController
{
    [OutputCache(Duration = 10, VaryByParam = "none", Location = OutputCacheLocation.Any)]
    public string Get()
    {
        return System.DateTime.Now.ToString();
    }
}

N.B. I'd also tried the OutputCache attribute on the controller itself, as well as several combinations of its parameters.

The route is registered in Global.asax:

namespace WebApiTest
{
    public class Global : HttpApplication
    {
        protected void Application_Start(object sender, EventArgs e)
        {
            RouteTable.Routes.MapHttpRoute("default", routeTemplate: "{controller}");
        }
    }
}

I get a successful response, but it's not cached anywhere:

HTTP/1.1 200 OK
Cache-Control: no-cache
Pragma: no-cache
Content-Type: application/xml; charset=utf-8
Expires: -1
Server: Microsoft-IIS/7.5
X-AspNet-Version: 4.0.30319
X-Powered-By: ASP.NET
Date: Wed, 18 Jul 2012 17:56:17 GMT
Content-Length: 96

<string xmlns="http://schemas.microsoft.com/2003/10/Serialization/">18/07/2012 18:56:17</string>

I was not able to find documentation for output caching in Web API.

Is this a limitation of the Web API in MVC4 or am I doing something wrong?

Engelhart answered 18/7, 2012 at 18:5 Comment(0)
D
46

WebAPI does not have any built in support for the [OutputCache] attribute. Take a look at this article to see how you could implement this feature yourself.

Dashing answered 18/7, 2012 at 18:18 Comment(3)
Tohid, I could not get that to work. Apart from anything else the HttpActionExecutedContext doesn't have the Result property.Hampstead
what about this? strathweb.com/2012/05/output-caching-in-asp-net-web-apiAnstus
Flatwhite is a great output caching library with support for web api: github.com/vanthoainguyen/FlatwhiteMargiemargin
C
35

The answer of Aliostad states that Web API turns off caching, and the code of HttpControllerHandler shows that it does WHEN response.Headers.CacheControl is null.

To make your example ApiController Action return a cacheable result, you can:

using System.Net.Http;

public class TestController : ApiController
{
    public HttpResponseMessage Get()
    {
        var response = Request.CreateResponse(HttpStatusCode.OK);
        response.Content = new StringContent(System.DateTime.Now.ToString());
        response.Headers.CacheControl = new CacheControlHeaderValue();
        response.Headers.CacheControl.MaxAge = new TimeSpan(0, 10, 0);  // 10 min. or 600 sec.
        response.Headers.CacheControl.Public = true;
        return response;
    }
}

and you will get a HTTP response header like this:

Cache-Control: public, max-age=600
Content-Encoding: gzip
Content-Type: text/plain; charset=utf-8
Date: Wed, 13 Mar 2013 21:06:10 GMT
...
Canikin answered 13/3, 2013 at 21:27 Comment(3)
This question is about "output caching" which is a specific type of caching where the response is cached by the web server itself. AFAIK this is not related to Cache-Control headers..Mckinzie
In my view, what you do on the server side, how you cache (memory, disk, database, other) and for how long is up to you. If you mark the item as cacheable and public, the browser and proxy servers (among others) on the way between the client and your server are allowed to serve themselves the cached content, instead of getting it from your server again.Canikin
@Mckinzie The MVC output caching attribute parameters get translated to different Cache-Control headers. See this for an example: https://mcmap.net/q/303735/-outputcache-setting-inside-my-asp-net-mvc-web-application-multiple-syntax-to-prevent-caching So while this answer doesn't specifically use the OutputCacheAttribute, it does achieve the desired result.Fionnula
G
17

For the last few months, I have been working on HTTP caching for ASP.NET Web API. I have contributed to WebApiContrib for server-side and relevant information can be found on my blog.

Recently I have started to expand the work and add the client-side as well in the CacheCow library. First NuGet packages have been released now (thanks to Tugberk) More to come. I will write a blog post soon on this. So watch the space.


But in order to answer your question, ASP.NET Web API by default turns off the caching. If you want the response to be cached, you need to add the CacheControl header to the response in your controller (and in fact better be in a delegating handler similar to CachingHandler in CacheCow).

This snippet is from HttpControllerHandler in ASP.NET Web Stack source code:

        CacheControlHeaderValue cacheControl = response.Headers.CacheControl;

        // TODO 335085: Consider this when coming up with our caching story
        if (cacheControl == null)
        {
            // DevDiv2 #332323. ASP.NET by default always emits a cache-control: private header.
            // However, we don't want requests to be cached by default.
            // If nobody set an explicit CacheControl then explicitly set to no-cache to override the
            // default behavior. This will cause the following response headers to be emitted:
            //     Cache-Control: no-cache
            //     Pragma: no-cache
            //     Expires: -1
            httpContextBase.Response.Cache.SetCacheability(HttpCacheability.NoCache);
        }
Gunslinger answered 19/7, 2012 at 8:9 Comment(4)
This link doesn't work: byterot.blogspot.co.uk/2012/07/…Gaytan
@Gaytan have a look here byterot.blogspot.co.uk/2012/07/…Gunslinger
@Gunslinger do you have a link to documentation supporting the statement "ASP.NET Web API by default turns off the caching"? I know this is true - just need official confirmation.Mora
@NoelAbrahams studying the ASP.NET Web API codebase my friend :) Is not documented as far I am aware.Gunslinger
K
8

I am very late, but still thought to post this great article on Caching in WebApi

https://codewala.net/2015/05/25/outputcache-doesnt-work-with-web-api-why-a-solution/

public class CacheWebApiAttribute : ActionFilterAttribute
{
    public int Duration { get; set; }

    public override void OnActionExecuted(HttpActionExecutedContext filterContext)
    {
        filterContext.Response.Headers.CacheControl = new CacheControlHeaderValue()
        {
            MaxAge = TimeSpan.FromMinutes(Duration),
            MustRevalidate = true,
            Private = true
        };
    }
}

In the above code, we have overridden OnActionExecuted method and set the required header in the response. Now I have decorated the Web API call as

[CacheWebApi(Duration = 20)]
        public IEnumerable<string> Get()
        {
            return new string[] { DateTime.Now.ToLongTimeString(), DateTime.UtcNow.ToLongTimeString() };
        }
Koons answered 19/4, 2018 at 13:49 Comment(0)
H
-23

You could use this on a regular MVC Controller:

[OutputCache(Duration = 10, VaryByParam = "none", Location = OutputCacheLocation.Any)]
public string Get()
{
    HttpContext.Current.Response.Cache.SetOmitVaryStar(true);
    return System.DateTime.Now.ToString();
}

but OutputCache attribute is in System.Web.Mvc namespace and not available in an ApiController.

Hermes answered 26/9, 2012 at 5:24 Comment(1)
[OutputCache] you're using is from System.Web.Mvc, not System.Web.Http (WebApi)Bread

© 2022 - 2024 — McMap. All rights reserved.