I was thinking about asking on Software Recommendations, but then I've found out that it may be a too strange request and it needs some clarification first.
My points are:
- Each response contains an
etag
- which is a hash of the content
- and which is globally unique (with sufficient probability)
- The content is (mostly) dynamic and may change anytime (
expires
andmax-age
headers are useless here). - The content is partly user-dependent, as given by the permissions (which itself change sometimes).
Basically, the proxy should contain a cache mapping the etag
to the response content. The etag
gets obtained from the server and in the most common case, the server does not deal with the response content at all.
It should go like follows: The proxy always sends a request to the server and then either
- 1 the server returns only the
etag
and the proxy makes a lookup based on it and- 1.1 on cache hit,
- it reads the response data from cache
- and sends a response to the client
- 1.2 on cache miss,
- it asks the server again and then
- the server returns the response with content and
etag
, - the proxy stores it in its cache
- and sends a response to the client
- 1.1 on cache hit,
- 2 or the server returns the response with content and
etag
,- the proxy stores the data in its cache
- and sends a response to the client
For simplicity, I left out the handling of the if-none-match
header, which is rather obvious.
My reason for this is that the most common case 1.1 can be implemented very efficiently in the server (using its cache mapping requests to etags
; the content isn't cached in the server), so that most requests can be handled without the server dealing with the response content. This should be better than first getting the content from a side cache and then serving it.
In case 1.2, there are two requests to the server, which sounds bad, but is no worse than the server asking a side cache and getting a miss.
Q1: I wonder, how to map the first request to HTTP. In case 1, it's like a HEAD request. In case 2, it's like GET. The decision between the two is up to the server: If it can serve the etag
without computing the content, then it's case 1, otherwise, it's case 2.
Q2: Is there a reverse proxy doing something like this? I've read about nginx, HAProxy and Varnish and it doesn't seem to be the case. This leads me to Q3: Is this a bad idea? Why?
Q4: If not, then which existing proxy is easiest to adapt?
An Example
A GET request like /catalog/123/item/456
from user U1
was served with some content C1
and etag: 777777
. The proxy stored C1
under the key 777777
.
Now the same request comes from user U2
. The proxy forwards it, the server returns just etag: 777777
and the proxy is lucky, finds C1
in its cache (case 1.1) and sends it to U2
. In this example, neither the clients not the proxy knew the expected result.
The interesting part is how could the server know the etag
without computing the answer. For example, it can have a rule stating that requests of this form return the same result for all users, assuming that the given user is allowed to see it. So when the request from U1
came, it computed C1
and stored the etag
under the key /catalog/123/item/456
. When the same request came from U2
, it just verified that U2
is permitted to see the result.