chore(deps): update dependency tornado to v6.5.7 [security]#9
Open
renovate[bot] wants to merge 1 commit into
Open
chore(deps): update dependency tornado to v6.5.7 [security]#9renovate[bot] wants to merge 1 commit into
renovate[bot] wants to merge 1 commit into
Conversation
1b4c211 to
ef6dde5
Compare
ef6dde5 to
5306d3c
Compare
5306d3c to
54d789b
Compare
Coverage Report for CI Build 27870803688Warning No base build found for commit Coverage: 76.287%Details
Uncovered ChangesNo uncovered changes found. Coverage RegressionsRequires a base build to compare against. How to fix this → Coverage Stats
💛 - Coveralls |
54d789b to
2487a88
Compare
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.3.3→==6.5.7Inconsistent Interpretation of HTTP Requests ('HTTP Request/Response Smuggling') in tornado
GHSA-753j-mpmx-qq6g
More information
Details
Summary
When Tornado receives a request with two
Transfer-Encoding: chunkedheaders, it ignores them both. This enables request smuggling when Tornado is deployed behind a proxy server that emits such requests. Pound does this.PoC
Transfer-Encoding: chunkedheaders:This is because Tornado believes that the request has no message body, so it tries to interpret
1\r\nZ\r\n0\r\n\r\nas its own request, which causes a 400 response. With a little cleverness involvingchunk-exts, you can get Tornado to instead respond 405, which has the potential to desynchronize the connection, as opposed to 400 which should always result in a connection closure.Impact
Anyone using Tornado behind a proxy that forwards requests containing multiple
Transfer-Encoding: chunkedheaders is vulnerable to request smuggling, which may entail ACL bypass, cache poisoning, or connection desynchronization.Severity
CVSS:3.1/AV:N/AC:L/PR:N/UI:N/S:U/C:N/I:L/A:NReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado has a CRLF injection in CurlAsyncHTTPClient headers
GHSA-w235-7p84-xx57
More information
Details
Summary
Tornado’s
curl_httpclient.CurlAsyncHTTPClientclass is vulnerable to CRLF (carriage return/line feed) injection in the request headers.Details
When an HTTP request is sent using
CurlAsyncHTTPClient, Tornado does not reject carriage return (\r) or line feed (\n) characters in the request headers. As a result, if an application includes an attacker-controlled header value in a request sent usingCurlAsyncHTTPClient, the attacker can inject arbitrary headers into the request or cause the application to send arbitrary requests to the specified server.This behavior differs from that of the standard
AsyncHTTPClientclass, which does reject CRLF characters.This issue appears to stem from libcurl's (as well as pycurl's) lack of validation for the
HTTPHEADERoption. libcurl’s documentation states:pycurl similarly appears to assume that the headers adhere to the correct format. Therefore, without any validation on Tornado’s part, header names and values are included verbatim in the request sent by
CurlAsyncHTTPClient, including any control characters that have special meaning in HTTP semantics.PoC
The issue can be reproduced using the following script:
When the specified server receives the request, it contains the injected header (
Header: Injected) on its own line:The attacker can also construct entirely new requests using a payload with multiple CRLF sequences. For example, specifying a header value of
\r\n\r\nPOST /attacker-controlled-url HTTP/1.1\r\nHost: 727ymeu841qydmnwlol261ktkkqbe24qt.oastify.comresults in the server receiving an additional, attacker-controlled request:Impact
Applications using the Tornado library to send HTTP requests with untrusted header data are affected. This issue may facilitate the exploitation of server-side request forgery (SSRF) vulnerabilities.
Severity
CVSS:3.1/AV:N/AC:L/PR:N/UI:N/S:U/C:L/I:L/A:NReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado has an HTTP cookie parsing DoS vulnerability
CVE-2024-52804 / GHSA-8w49-h785-mj3c
More information
Details
The algorithm used for parsing HTTP cookies in Tornado versions prior to 6.4.2 sometimes has quadratic complexity, leading to excessive CPU consumption when parsing maliciously-crafted cookie headers. This parsing occurs in the event loop thread and may block the processing of other requests.
See also CVE-2024-7592 for a similar vulnerability in cpython.
Severity
CVSS:3.1/AV:N/AC:L/PR:N/UI:N/S:U/C:N/I:N/A:HReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado vulnerable to excessive logging caused by malformed multipart form data
CVE-2025-47287 / GHSA-7cx3-6m66-7c5m
More information
Details
Summary
When Tornado's
multipart/form-dataparser encounters certain errors, it logs a warning but continues trying to parse the remainder of the data. This allows remote attackers to generate an extremely high volume of logs, constituting a DoS attack. This DoS is compounded by the fact that the logging subsystem is synchronous.Affected versions
All versions of Tornado prior to 6.5 are affected. The vulnerable parser is enabled by default.
Solution
Upgrade to Tornado version 6.5. In the meantime, risk can be mitigated by blocking
Content-Type: multipart/form-datain a proxy.Severity
CVSS:3.1/AV:N/AC:L/PR:N/UI:N/S:U/C:N/I:N/A:HReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado has incomplete validation of cookie attributes
GHSA-78cv-mqj4-43f7
More information
Details
Values passed to the
domain,path, andsamesitearguments ofRequestHandler.set_cookiewere not completely validated in versions of Tornado prior to 6.5.5. In particular, semicolons would be allowed, which could be used to inject attacker-controlled values for other cookie attributes.Severity
CVSS:3.1/AV:N/AC:L/PR:N/UI:R/S:U/C:L/I:L/A:NReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado is vulnerable to DoS due to too many multipart parts
CVE-2026-31958 / GHSA-qjxf-f2mg-c6mc
More information
Details
In versions of Tornado prior to 6.5.5, the only limit on the number of parts in
multipart/form-datais themax_body_sizesetting (default 100MB). Since parsing occurs synchronously on the main thread, this creates the possibility of denial-of-service due to the cost of parsing very large multipart bodies with many parts.Tornado 6.5.5 introduces new limits on the size and complexity of multipart bodies, including a default limit of 100 parts per request. These limits are configurable if needed; see
tornado.httputil.ParseMultipartConfig. It is also now possible to disablemultipart/form-dataparsing entirely if it is not required for the application.Severity
CVSS:4.0/AV:N/AC:L/AT:N/PR:N/UI:N/VC:N/VI:N/VA:H/SC:N/SI:N/SA:NReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado has cookie attribute injection via .RequestHandler.set_cookie
CVE-2026-35536 / GHSA-fqwm-6jpj-5wxc
More information
Details
In Tornado before 6.5.5, cookie attribute injection could occur because the domain, path, and samesite arguments to
.RequestHandler.set_cookiewere not checked for crafted characters.Severity
CVSS:3.1/AV:N/AC:L/PR:N/UI:N/S:C/C:L/I:L/A:NReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado has out-of-bounds memory access via C extension
CVE-2026-49854 / GHSA-cx3h-4qpv-8hc9
More information
Details
Summary
Tornado's optional native extension
tornado.speedupsimplementswebsocket_maskwithout validating that themaskargument is exactly four bytes long. The C function reads four bytes frommaskunconditionally, even when Python passes a shorter byte string. This can read beyond the provided buffer, exposing up to 3 bytes of uninitialized memory.The behavior is reachable from Tornado's XSRF token decoder when
xsrf_cookies=Trueand the native extension is active.Mitigations
This bug is fixed in Tornado 6.5.6. Prior to upgrading to this version, setting the environment variable TORNADO_EXTENSION=0 will disable the vulnerable code (at the expense of reducing websocket performance).
Severity
CVSS:3.1/AV:N/AC:H/PR:N/UI:N/S:U/C:L/I:N/A:NReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
tornado AsyncHTTPClient accumulates decompressed chunks without size limit (gzip bomb)
CVE-2026-49855 / GHSA-mgf9-4vpg-hj56
More information
Details
Tornado's gzip decompression routines work in limited-size chunks, but have no overall limit for the total size of decompressed chunks that they will accumulate (There has always been a limit for the total compressed size). This allows a malicious server to consume effectively unlimited amounts of memory if it is accessed via SimpleAsyncHTTPClient in its default configuration.
HTTPServeris not affected in its default configuration, but it is ifdecompress_request=Trueis set.This bug is fixed in Tornado 6.5.6.
max_body_sizeis now checked both for the compressed and cumulative decompressed size of the response.Prior to upgrading, this issue can be mitigated by setting
decompress_response=Falseor usingCurlAsyncHTTPClient.Severity
CVSS:3.1/AV:N/AC:L/PR:N/UI:N/S:U/C:N/I:N/A:HReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado: Authorization header forwarded across cross-origin redirects in SimpleAsyncHTTPClient
CVE-2026-49853 / GHSA-3x9g-8vmp-wqvf
More information
Details
Summary
When SimpleAsyncHTTPClient follows a 3xx redirect, it shallow-copies the original HTTPRequest, rewrites the URL, decrements max_redirects, and removes only the Host header. It does not clear Authorization, auth_username, auth_password, or auth_mode when the redirect target changes origin.
As a result, credentials intended for one origin can be forwarded to a different origin when follow_redirects=True, which is the default.
Beginning in Tornado 6.5.6,
SimpleAsyncHTTPClientmatches the default behavior oflibcurl(and thereforeCurlAsyncHTTPClient): When a redirect changes the scheme, host, or port of the url, theAuthorizationandCookieheaders will be removed when following the redirect.Severity
CVSS:3.1/AV:N/AC:L/PR:L/UI:N/S:C/C:H/I:N/A:NReferences
This data is provided by the GitHub Advisory Database (CC-BY 4.0).
Tornado: 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
v6.5.6Compare Source
v6.5.5Compare Source
v6.5.4Compare Source
v6.5.3Compare Source
v6.5.2Compare Source
v6.5.1Compare Source
v6.5Compare Source
v6.4.2Compare Source
v6.4.1Compare Source
v6.4Compare 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.