System.Net.Http.HttpClient caching behavior

27,495

Both clients behave correctly.

must-revalidate only applies to stale responses.

When the must-revalidate directive is present in a response received by a cache, that cache MUST NOT use the entry after it becomes stale to respond to a subsequent request without first revalidating it with the origin server. (I.e., the cache MUST do an end-to-end revalidation every time, if, based solely on the origin server's Expires or max-age value, the cached response is stale.)

Since you do not provide explicit expiration, caches are allowed to use heuristics to determine freshness.

Since you do not provide Last-Modified caches do not need to warn the client that heuristics was used.

If none of Expires, Cache-Control: max-age, or Cache-Control: s- maxage (see section 14.9.3) appears in the response, and the response does not include other restrictions on caching, the cache MAY compute a freshness lifetime using a heuristic. The cache MUST attach Warning 113 to any response whose age is more than 24 hours if such warning has not already been added.

The response age is calculated based on Date header since Age is not present.

If the response is still fresh according to heuristic expiration, caches may use the stored response.

One explanation is that HttpWebRequest uses heuristics and that there was a stored response with status code 200 that was still fresh.

Share:
27,495
NiklasN
Author by

NiklasN

Updated on July 09, 2022

Comments

  • NiklasN
    NiklasN almost 2 years

    I'm using HttpClient 0.6.0 from NuGet.

    I have the following C# code:

    var client = new HttpClient(new WebRequestHandler() {
        CachePolicy =
            new HttpRequestCachePolicy(HttpRequestCacheLevel.CacheIfAvailable)
    });
    client.GetAsync("http://myservice/asdf");
    

    The service (this time CouchDB) returns an ETag value and status code 200 OK. There is returned a Cache-Control header with value must-revalidate

    Update, here are the response headers from couchdb (taken from the visual studio debugger):

    Server: CouchDB/1.1.1 (Erlang OTP/R14B04)
    Etag: "1-27964df653cea4316d0acbab10fd9c04"
    Date: Fri, 09 Dec 2011 11:56:07 GMT
    Cache-Control: must-revalidate
    

    Next time I do the exact same request, HttpClient does a conditional request and gets back 304 Not Modified. Which is right.

    However, if I am using low-level HttpWebRequest class with the same CachePolicy, the request isn't even made the second time. This is the way I would want HttpClient also behave.

    Is it the must-revalidate header value or why is HttpClient behaving differently? I would like to do only one request and then have the rest from cache without the conditional request..

    (Also, as a side-note, when debugging, the Response status code is shown as 200 OK, even though the service returns 304 Not Modified)

  • Pavel Gatilov
    Pavel Gatilov over 11 years
    This is not correct, the spec doesn't say that. The correct answer has been given by Hans Malherbe
  • NiklasN
    NiklasN over 11 years
    Thank you for the correct answer. I accepted it. This made things much more clear to me!