Summary
The CrowdSec AppSec component fails to read the HTTP request body for any request whose Content-Length is not positive — most notably HTTP/1.1 requests using Transfer-Encoding: chunked and HTTP/2 requests sent without a content-length header. Coraza is then evaluated against an empty body, so every WAF rule targeting REQUEST_BODY, BODY_ARGS, ARGS_POST, JSON, or XML silently fails to match.
An unauthenticated remote attacker can bypass the entire AppSec body-inspection pipeline by changing a single framing header on an otherwise-malicious request. The bypassed request is forwarded as allow and produces no WAF log entry.
Affected versions
github.com/crowdsecurity/crowdsec — all releases up to and including v1.7.7.
Affected component
pkg/appsec/request.go, function NewParsedRequestFromRequest.
Root cause
func NewParsedRequestFromRequest(r *http.Request, logger *log.Entry) (ParsedRequest, error) {
var err error
contentLength := max(r.ContentLength, 0)
body := make([]byte, contentLength)
if r.Body != nil {
_, err = io.ReadFull(r.Body, body)
if err != nil {
return ParsedRequest{}, fmt.Errorf("unable to read body: %s", err)
}
r.Body = io.NopCloser(bytes.NewBuffer(body))
}
...
}
Go's net/http server sets r.ContentLength = -1 when the request uses Transfer-Encoding: chunked with no Content-Length header, or when an HTTP/2 request omits the content-length pseudo-header (DATA-frame-only body). With ContentLength == -1:
max(-1, 0) evaluates to 0.
make([]byte, 0) allocates a zero-length slice.
io.ReadFull on a zero-length buffer needs zero bytes and returns immediately without touching r.Body.
- The empty buffer is written back onto the request and onto the cloned request constructed later in the same function.
Every downstream consumer then sees an empty body. In the AppSec runner, WriteRequestBody is skipped because the parsed body has zero length, and ProcessRequestBody runs against nothing.
Impact
Every body-scanning rule is bypassed for any request whose framing makes Content-Length non-positive. In default CrowdSec deployments using the standard AppSec collections, the bypass affects any rule with zones containing BODY_ARGS, JSON, XML, REQUEST_BODY, or ARGS_POST.
No configuration option mitigates the issue — the defect is in the request parser, not in any ruleset. Bypassed requests do not produce a WAF log entry, so operators have no signal that rules are being skipped.
Header-only and URI-only rules are unaffected.
Workarounds
No complete workaround is available.
References
Summary
The CrowdSec AppSec component fails to read the HTTP request body for any request whose
Content-Lengthis not positive — most notably HTTP/1.1 requests usingTransfer-Encoding: chunkedand HTTP/2 requests sent without acontent-lengthheader. Coraza is then evaluated against an empty body, so every WAF rule targetingREQUEST_BODY,BODY_ARGS,ARGS_POST,JSON, orXMLsilently fails to match.An unauthenticated remote attacker can bypass the entire AppSec body-inspection pipeline by changing a single framing header on an otherwise-malicious request. The bypassed request is forwarded as
allowand produces no WAF log entry.Affected versions
github.com/crowdsecurity/crowdsec— all releases up to and including v1.7.7.Affected component
pkg/appsec/request.go, functionNewParsedRequestFromRequest.Root cause
Go's
net/httpserver setsr.ContentLength = -1when the request usesTransfer-Encoding: chunkedwith noContent-Lengthheader, or when an HTTP/2 request omits thecontent-lengthpseudo-header (DATA-frame-only body). WithContentLength == -1:max(-1, 0)evaluates to0.make([]byte, 0)allocates a zero-length slice.io.ReadFullon a zero-length buffer needs zero bytes and returns immediately without touchingr.Body.Every downstream consumer then sees an empty body. In the AppSec runner,
WriteRequestBodyis skipped because the parsed body has zero length, andProcessRequestBodyruns against nothing.Impact
Every body-scanning rule is bypassed for any request whose framing makes
Content-Lengthnon-positive. In default CrowdSec deployments using the standard AppSec collections, the bypass affects any rule withzonescontainingBODY_ARGS,JSON,XML,REQUEST_BODY, orARGS_POST.No configuration option mitigates the issue — the defect is in the request parser, not in any ruleset. Bypassed requests do not produce a WAF log entry, so operators have no signal that rules are being skipped.
Header-only and URI-only rules are unaffected.
Workarounds
No complete workaround is available.
References