Update dependency tornado to v6.5.7 [SECURITY]#1071
Merged
Conversation
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Add this suggestion to a batch that can be applied as a single commit.This suggestion is invalid because no changes were made to the code.Suggestions cannot be applied while the pull request is closed.Suggestions cannot be applied while viewing a subset of changes.Only one suggestion per line can be applied in a batch.Add this suggestion to a batch that can be applied as a single commit.Applying suggestions on deleted lines is not supported.You must change the existing code in this line in order to create a valid suggestion.Outdated suggestions cannot be applied.This suggestion has been applied or marked resolved.Suggestions cannot be applied from pending reviews.Suggestions cannot be applied on multi-line comments.Suggestions cannot be applied while the pull request is queued to merge.Suggestion cannot be applied right now. Please check back later.
This PR contains the following updates:
6.5.6→6.5.7Tornado: CurlAsyncHTTPClient leaks per-request credentials on handle reuse
GHSA-pw6j-qg29-8w7f
More information
Details
CurlAsyncHTTPClient leaks per-request credentials on handle reuse
Summary
CurlAsyncHTTPClientpools and reusespycurlhandles across requests but doesnot reset them between requests, and several per-request options are applied with
no clearing branch. As a result, sensitive state set by one request persists onto
a later request on the same client that does not set it. Two credential vectors
are demonstrated below — a client TLS certificate (
SSLCERT/SSLKEY) and proxybasic-auth credentials (
PROXYUSERPWD) — both leaking to a different,unintended host. This affects all released versions through 6.5.6.
Details
In
tornado/curl_httpclient.py, handles are created once and returned to a freelist for reuse (
_process_queuepops the handle at line 200,_finishre-appends it at line 245), and
_curl_setup_requestis never preceded bycurl.reset(). The function clears some carried-over state on the reused handle—
unsetopt(PROXYUSERPWD)in the no-proxy branch (line 394),unsetopt(USERPWD)when no auth is set (line 495), and the HTTP-method flag reset (lines 428-432) —
but other options have no equivalent clearing path and persist until a later
request sets them again.
Vector A — client TLS certificate (
SSLCERT/SSLKEY). Set-only, no clearingbranch:
A request that sets
client_certleaves the certificate on the handle; a laterrequest without
client_certpresents it during its TLS handshake.Vector B — proxy credentials (
PROXYUSERPWD).PROXYUSERPWDis set onlyinside the credentials branch and unset only in the no-proxy
elsebranch:A request that sets a new
proxy_hostwithoutproxy_usernameupdatesPROXY/PROXYPORTbut never reaches theelse, so the previous request'scredentials persist and are sent to the new proxy.
The same class also affects
INTERFACE(lines 365-366: set only whenrequest.network_interfaceis truthy, with no clearing branch), which is alower-severity instance — a later request can be bound to a network interface it
did not request. A single fix addresses all three (see Mitigation).
PoC
Both reproduce against the pinned release using public API only
(
CurlAsyncHTTPClient,HTTPRequest, and the documented per-request arguments).Vector A — client TLS certificate
The two servers listen on different ports, so request B opens a fresh TCP+TLS
connection; the certificate can only reach server 2 via the persisted handle
option, not connection or session reuse.
Output (
pip show tornado→ 6.5.6, installed in the venv):Vector B — proxy credentials
Each proxy is a separate listener capturing the raw request bytes.
Output (
YWxpY2U6c2VjcmV0QQ==decodes toalice:secretA):Impact
of a resource whose sensitive state was not cleared (CWE-672).
options on a shared
CurlAsyncHTTPClient— for Vector A, mixing per-requestclient_certrequests with non-certificate requests; for Vector B,multiplexing requests across more than one proxy with per-proxy credentials.
handshake — proving possession of the private key and disclosing the
certificate subject and chain — to a host that was never meant to receive it.
For Vector B, proxy basic-auth credentials are transmitted (base64) to a
different proxy. If the unintended host/proxy is attacker-controlled or
attacker-influenced (a user-supplied URL, webhook target, SSRF-reachable
endpoint, or a proxy chosen from user-controlled configuration), the credential
is disclosed to the attacker.
CurlAsyncHTTPClientbackendwith the patterns above are affected. The default
SimpleAsyncHTTPClientis notaffected (and does not support proxies).
Proposed CWE: CWE-200 / CWE-672. Proposed CVSS 3.1:
CVSS:3.1/AV:N/AC:H/PR:N/UI:N/S:U/C:H/I:N/A:N(5.9, medium); attack complexity isHigh because exploitation depends on the application using differing per-request
options on a shared client and on handle scheduling.
Mitigation
A single fix closes all instances of this class: call
curl.reset()at the startof
_curl_setup_requestand then re-apply the per-request options, so no statefrom a prior request can persist on the reused handle. (Note
curl.reset()alsoclears
CAINFO, which the current code intentionally leaves untouched — see thecomment at lines 401-409 — so that default would need to be re-established after
the reset.)
Alternatively, add explicit clearing branches mirroring the existing
PROXYUSERPWD/USERPWDhandling:Until a fix is available, use a separate
CurlAsyncHTTPClientinstance perdistinct credential set (per client certificate / per proxy credential), or use
SimpleAsyncHTTPClientwhere applicable.Severity
CVSS:3.1/AV:N/AC:H/PR:N/UI:N/S:U/C:H/I:N/A:NReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Release Notes
tornadoweb/tornado (tornado)
v6.5.7Compare Source
Configuration
📅 Schedule: (UTC)
🚦 Automerge: Disabled by config. Please merge this manually once you are satisfied.
♻ Rebasing: Whenever PR becomes conflicted, or you tick the rebase/retry checkbox.
🔕 Ignore: Close this PR and you won't be reminded about this update again.
This PR was generated by Mend Renovate. View the repository job log.