Skip to content

Commit

Permalink
REQMOD stuck when adapted request (body) is not forwarded (#1966)
Browse files Browse the repository at this point in the history
Squid forwards request bodies using BodyPipe objects. A BodyPipe object
has two associated agents: producer and consumer. Those agents are set
independently, at different processing stages. If BodyPipe consumer is
not set, the producer may get stuck waiting for BodyPipe buffer space.
When producer creates a BodyPipe, it effectively relies on some code
somewhere to register a consumer (or declare a registration failure),
but automatically tracking that expectation fulfillment is impractical.

For REQMOD transactions involving adapted request bodies, including ICAP
204 transactions, Client::startRequestBodyFlow() sets body consumer. If
that method is not called, there will be no consumer, and REQMOD may get
stuck. Many `if` statements can block request forwarding altogether or
block a being-forwarded request from reaching that method. For example,
adapted_http_access and miss_access denials block request forwarding.

Without REQMOD, request processing can probably get stuck for similar
lack-of-consumer reasons, but regular requests ought to be killed by
various I/O or forwarding timeouts. There are no such timeouts for those
REQMOD transactions that are only waiting for request body consumer to
clear adapted BodyPipe space (e.g., after all ICAP 204 I/Os are over).
Relying on timeouts is also very inefficient.

For a `mgr:mem` observer, stuck REQMOD transactions look like a ModXact
memory leak. A `mgr:jobs` report shows ModXact jobs with RBS(1) status.
  • Loading branch information
rousskov authored and squid-anubis committed Dec 30, 2024
1 parent 16cafa1 commit 387bcd6
Show file tree
Hide file tree
Showing 2 changed files with 35 additions and 12 deletions.
24 changes: 12 additions & 12 deletions src/BodyPipe.cc
Original file line number Diff line number Diff line change
Expand Up @@ -266,18 +266,18 @@ BodyPipe::clearConsumer()
void
BodyPipe::expectNoConsumption()
{
// We may be called multiple times because multiple jobs on the consumption
// chain may realize that there will be no more setConsumer() calls (e.g.,
// consuming code and retrying code). It is both difficult and not really
// necessary for them to coordinate their expectNoConsumption() calls.

// As a consequence, we may be called when we are auto-consuming already.

if (!abortedConsumption && !exhausted()) {
// Before we abort, any regular consumption should be over and auto
// consumption must not be started.
Must(!theConsumer);

// We and enableAutoConsumption() may be called multiple times because
// multiple jobs on the consumption chain may realize that there will be no
// more setConsumer() calls (e.g., consuming code and retrying code). It is
// both difficult and unnecessary for them to coordinate their calls.

// As a consequence, we may be called when already auto-consuming, including
// cases where abortedConsumption is still false. We could try to harden
// this by also aborting consumption from enableAutoConsumption() when there
// is no consumer, but see errorAppendEntry() TODO for a better plan.

debugs(91, 7, status());
if (!abortedConsumption && !exhausted() && !theConsumer) {
AsyncCall::Pointer call= asyncCall(91, 7,
"BodyProducer::noteBodyConsumerAborted",
BodyProducerDialer(theProducer,
Expand Down
23 changes: 23 additions & 0 deletions src/errorpage.cc
Original file line number Diff line number Diff line change
Expand Up @@ -741,6 +741,29 @@ errorAppendEntry(StoreEntry * entry, ErrorState * err)
assert (entry->isEmpty());
debugs(4, 4, "storing " << err << " in " << *entry);

if (const auto &request = err->request) {
if (const auto &bodyPipe = request->body_pipe) {
// We cannot expectNoConsumption() here (yet): This request may be a
// virgin request being consumed by adaptation that should continue
// even in error-handling cases. startAutoConsumptionIfNeeded() call
// triggered by enableAutoConsumption() below skips such requests.
//
// Today, we also cannot enableAutoConsumption() earlier because it
// could result in premature consumption in BodyPipe::postAppend()
// followed by an unwanted setConsumerIfNotLate() failure.
//
// TODO: Simplify BodyPipe auto-consumption by automatically
// enabling it when no new consumers are expected, removing the need
// for explicit enableAutoConsumption() calls like the one below.
//
// Code like clientReplyContext::sendClientOldEntry() might use
// another StoreEntry for this master transaction, but we want to
// consume this request body even in those hypothetical error cases
// to prevent stuck (client-Squid or REQMOD) transactions.
bodyPipe->enableAutoConsumption();
}
}

if (entry->store_status != STORE_PENDING) {
debugs(4, 2, "Skipping error page due to store_status: " << entry->store_status);
/*
Expand Down

0 comments on commit 387bcd6

Please sign in to comment.