Skip to content

Commit

Permalink
Treat responses to collapsed requests as fresh (#1927)
Browse files Browse the repository at this point in the history
Squid transactions involving collapsed requests receive their responses
as Store cache hits. Cache hits undergo mandatory freshness validation
checks and, if those checks detect a stale response, may trigger
revalidation (e.g., an If-Modified-Since request to the origin server).
This logic results in a still-being-delivered-to-Squid response
triggering its own revalidation if that response is deemed stale on
arrival by collapsed request (e.g., has a Cache-Control: max-age=0
header field).

HTTP RFC 9111 Section 4.7 briefly mentions collapsed requests but is
ambiguous with regard to validation of responses to collapsed requests.
IETF HTTP Working Group chair believes that specs are unclear, and that
these responses should not be treated as cache hits (in primary cases):
https://lists.w3.org/Archives/Public/ietf-http-wg/2024JanMar/0095.html

This change follows the above advice and prevents arguably excessive
freshness checks for responses to collapsed requests. This change is
collapsed-forwarding specific: It does not prevent freshness checks for
responses that were, at the time of a hit request, either fully cached
or still receiving response body bytes.

After this change, clientReplyContext can no longer collapse twice, once
after initial Store lookup and then again during refresh, because the
first collapse now precludes refresh.
  • Loading branch information
rousskov authored and squid-anubis committed Nov 5, 2024
1 parent fff4502 commit d58d286
Show file tree
Hide file tree
Showing 11 changed files with 41 additions and 7 deletions.
8 changes: 8 additions & 0 deletions doc/release-notes/release-7.sgml.in
Original file line number Diff line number Diff line change
Expand Up @@ -194,6 +194,14 @@ This section gives an account of those changes in three categories:
<tag>external_acl_type</tag>
<p>Removed <em>%IDENT</em> format code with Ident protocol support.

<tag>collapsed_forwarding</tag>
<p>Squid no longer revalidates responses to collapsed requests, treating
all such responses as fresh. This change follows IETF HTTP Working Group
advice (in an HTTP gray area) and prevents arguably excessive freshness
checks for responses to collapsed requests. This change does not prevent
freshness checks for responses that were, at the time of a hit request,
either fully cached or still receiving response body bytes.

</descrip>

<sect1>Removed directives<label id="removeddirectives">
Expand Down
4 changes: 4 additions & 0 deletions src/StoreClient.h
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ class StoreEntry;
class ACLFilledChecklist;
class LogTags;

// TODO: Merge store_client into StoreClient.
/// a storeGetPublic*() caller
class StoreClient: public Acl::ChecklistFiller
{
Expand All @@ -58,6 +59,9 @@ class StoreClient: public Acl::ChecklistFiller
bool mayInitiateCollapsing() const { return onCollapsingPath(); }
/// whether Squid configuration allows collapsing for this transaction
bool onCollapsingPath() const;

/// whether startCollapsingOn() was called and returned true
mutable bool didCollapse = false;
};

#if USE_DELAY_POOLS
Expand Down
9 changes: 8 additions & 1 deletion src/cf.data.pre
Original file line number Diff line number Diff line change
Expand Up @@ -6438,7 +6438,9 @@ LOC: Config.accessList.sendHit
DOC_START
Responses denied by this directive will not be served from the cache
(but may still be cached, see store_miss). This directive has no
effect on the responses it allows and on the cached objects.
effect on the responses it allows and on the cached objects. This
directive is applied to both regular from-cache responses and responses
reused by collapsed requests (see collapsed_forwarding).

Please see the "cache" directive for a summary of differences among
store_miss, send_hit, and cache directives.
Expand Down Expand Up @@ -7225,6 +7227,11 @@ DOC_START
requests hitting a stale cached object. Revalidation collapsing
is currently disabled for Squid instances containing SMP-aware
disk or memory caches and for Vary-controlled cached objects.

A response reused by the collapsed request is deemed fresh in that
request processing context -- Squid does not apply refresh_pattern and
internal freshness validation checks to collapsed transactions. Squid
does apply send_hit rules.
DOC_END

NAME: collapsed_forwarding_access
Expand Down
2 changes: 1 addition & 1 deletion src/client_side_reply.cc
Original file line number Diff line number Diff line change
Expand Up @@ -627,7 +627,7 @@ clientReplyContext::cacheHit(const StoreIOBuffer result)
http->updateLoggingTags(LOG_TCP_MISS);
processMiss();
return;
} else if (!r->flags.internal && refreshCheckHTTP(e, r)) {
} else if (!r->flags.internal && !didCollapse && refreshCheckHTTP(e, r)) {
debugs(88, 5, "clientCacheHit: in refreshCheck() block");
/*
* We hold a stale copy; it needs to be validated
Expand Down
4 changes: 2 additions & 2 deletions src/htcp.cc
Original file line number Diff line number Diff line change
Expand Up @@ -981,10 +981,10 @@ htcpSpecifier::checkHit()
debugs(31, 3, "NO; public object not found");
} else if (!e->validToSend()) {
debugs(31, 3, "NO; entry not valid to send" );
} else if (refreshCheckHTCP(e, checkHitRequest.getRaw())) {
debugs(31, 3, "NO; cached response is stale");
} else if (e->hittingRequiresCollapsing() && !startCollapsingOn(*e, false)) {
debugs(31, 3, "NO; prohibited CF hit: " << *e);
} else if (!didCollapse && refreshCheckHTCP(e, checkHitRequest.getRaw())) {
debugs(31, 3, "NO; cached response is stale");
} else {
debugs(31, 3, "YES!?");
hit = e;
Expand Down
4 changes: 2 additions & 2 deletions src/icp_v2.cc
Original file line number Diff line number Diff line change
Expand Up @@ -170,10 +170,10 @@ ICPState::confirmAndPrepHit(const StoreEntry &e) const
if (!e.validToSend())
return false;

if (!Config.onoff.icp_hit_stale && refreshCheckICP(&e, request))
if (e.hittingRequiresCollapsing() && !startCollapsingOn(e, false))
return false;

if (e.hittingRequiresCollapsing() && !startCollapsingOn(e, false))
if (!Config.onoff.icp_hit_stale && !didCollapse && refreshCheckICP(&e, request))
return false;

return true;
Expand Down
9 changes: 8 additions & 1 deletion src/refresh.cc
Original file line number Diff line number Diff line change
Expand Up @@ -523,6 +523,10 @@ refreshIsCachable(const StoreEntry * entry)
* minimum_expiry_time seconds delta (defaults to 60 seconds), to
* avoid objects which expire almost immediately, and which can't
* be refreshed.
*
* No hittingRequiresCollapsing() or didCollapse concerns here: This
* incoming response is fresh now, but we want to check whether it can be
* refreshed Config.minimum_expiry_time seconds later.
*/
int reason = refreshCheck(entry, nullptr, Config.minimum_expiry_time);
++ refreshCounts[rcStore].total;
Expand Down Expand Up @@ -569,6 +573,10 @@ refreshIsStaleIfHit(const int reason)
*
* \retval 1 if STALE
* \retval 0 if FRESH
*
* Do not call this when StoreClient::didCollapse is true. XXX: Callers should
* not have to remember to check didCollapse. TODO: Refactor by adding something
* like pure virtual StoreClient::refreshCheck() with protocol specializations?
*/
int
refreshCheckHTTP(const StoreEntry * entry, HttpRequest * request)
Expand All @@ -577,7 +585,6 @@ refreshCheckHTTP(const StoreEntry * entry, HttpRequest * request)
++ refreshCounts[rcHTTP].total;
++ refreshCounts[rcHTTP].status[reason];
request->flags.staleIfHit = refreshIsStaleIfHit(reason);
// TODO: Treat collapsed responses as fresh but second-hand.
return (Config.onoff.offline || reason < 200) ? 0 : 1;
}

Expand Down
1 change: 1 addition & 0 deletions src/store_client.cc
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@ StoreClient::startCollapsingOn(const StoreEntry &e, const bool doingRevalidation
tags->collapsingHistory.otherCollapses++;
}

didCollapse = true;
debugs(85, 5, e << " doingRevalidation=" << doingRevalidation);
return true;
}
Expand Down
4 changes: 4 additions & 0 deletions src/store_digest.cc
Original file line number Diff line number Diff line change
Expand Up @@ -260,6 +260,10 @@ storeDigestAddable(const StoreEntry * e)
}

/* still here? check staleness */
// Include hittingRequiresCollapsing() entries: They are fresh _now_, but we
// check future freshness. They should not have enough info to judge future
// freshness since we are still waiting for their response headers, but
// admins might configure Squid to consider such entries fresh anyway.
/* Note: We should use the time of the next rebuild, not (cur_time+period) */
if (refreshCheckDigest(e, Config.digest.rebuild_period)) {
debugs(71, 6, "storeDigestAdd: entry expires within " << Config.digest.rebuild_period << " secs, ignoring");
Expand Down
2 changes: 2 additions & 0 deletions src/urn.cc
Original file line number Diff line number Diff line change
Expand Up @@ -282,6 +282,8 @@ urnHandleReply(void *data, StoreIOBuffer result)
return;
}

// XXX: Missing reply freshness checks (e.g., calling refreshCheckHTTP()).

urls = urnParseReply(urnState->parsingBuffer.toSBuf(), urnState->request->method);

if (!urls) { /* unknown URN error */
Expand Down
1 change: 1 addition & 0 deletions test-suite/test-functionality.sh
Original file line number Diff line number Diff line change
Expand Up @@ -220,6 +220,7 @@ main() {
upgrade-protocols
cache-response
proxy-collapsed-forwarding
hit-revalidation
busy-restart
truncated-responses
malformed-request
Expand Down

0 comments on commit d58d286

Please sign in to comment.