Security
Headlines
HeadlinesLatestCVEs

Headline

CVE-2022-21712: Merge pull request from GHSA-92x2-jw7w-xvvx · twisted/twisted@af8fe78

twisted is an event-driven networking engine written in Python. In affected versions twisted exposes cookies and authorization headers when following cross-origin redirects. This issue is present in the twited.web.RedirectAgent and twisted.web. BrowserLikeRedirectAgent functions. Users are advised to upgrade. There are no known workarounds.

CVE
#web#auth#ssl

@@ -8,7 +8,8 @@ import zlib from http.cookiejar import CookieJar from io import BytesIO from unittest import skipIf from typing import TYPE_CHECKING, List, Optional, Tuple from unittest import SkipTest, skipIf
from zope.interface.declarations import implementer from zope.interface.verify import verifyObject @@ -76,6 +77,15 @@ URIInjectionTestsMixin, )
# Creatively lie to mypy about the nature of inheritance, since dealing with # expectations of a mixin class is basically impossible (don’t use mixins). if TYPE_CHECKING: testMixinClass = TestCase runtimeTestCase = object else: testMixinClass = object runtimeTestCase = TestCase
try: from twisted.internet import ssl as _ssl except ImportError: @@ -108,8 +118,8 @@ class StubHTTPProtocol(Protocol): request method is appended to this list. “"”
def __init__(self): self.requests = [] def __init__(self) -> None: self.requests: List[Tuple[Request, Deferred[IResponse]]] = [] self.state = “QUIESCENT”
def request(self, request): @@ -2587,12 +2597,25 @@ def getConnection(this, key, ep): self.assertEqual(agent._pool.connected, True)

class _RedirectAgentTestsMixin: SENSITIVE_HEADERS = [ b"authorization", b"cookie", b"cookie2", b"proxy-authorization", b"www-authenticate", ]

class _RedirectAgentTestsMixin(testMixinClass): “"” Test cases mixin for L{RedirectAgentTests} and L{BrowserLikeRedirectAgentTests}. “"”
agent: IAgent reactor: MemoryReactorClock protocol: StubHTTPProtocol
def test_noRedirect(self): “"” L{client.RedirectAgent} behaves like L{client.Agent} if the response @@ -2611,32 +2634,56 @@ def test_noRedirect(self): self.assertIdentical(response, result) self.assertIdentical(result.previousResponse, None)
def _testRedirectDefault(self, code): def _testRedirectDefault( self, code: int, crossScheme: bool = False, crossDomain: bool = False, crossPort: bool = False, requestHeaders: Optional[Headers] = None, ) -> Request: “"” When getting a redirect, L{client.RedirectAgent} follows the URL specified in the L{Location} header field and make a new request. @param code: HTTP status code. “"” self.agent.request(b"GET", b"http://example.com/foo") startDomain = b"example.com" startScheme = b"https" if ssl is not None else b"http" startPort = 80 if startScheme == b"http" else 443 self.agent.request( b"GET", startScheme + b"://" + startDomain + b"/foo", headers=requestHeaders )
host, port = self.reactor.tcpClients.pop()[:2] self.assertEqual(EXAMPLE_COM_IP, host) self.assertEqual(80, port) self.assertEqual(startPort, port)
req, res = self.protocol.requests.pop()
# If possible (i.e.: SSL support is present), run the test with a # If possible (i.e.: TLS support is present), run the test with a # cross-scheme redirect to verify that the scheme is honored; if not, # let’s just make sure it works at all. if ssl is None: scheme = b"http" expectedPort = 80 else: scheme = b"https" expectedPort = 443
headers = http_headers.Headers({b"location": [scheme + b"://example.com/bar"]})
targetScheme = startScheme targetDomain = startDomain targetPort = startPort
if crossScheme: if ssl is None: raise SkipTest( “Cross-scheme redirects can’t be tested without TLS support.” ) targetScheme = b"https" if startScheme == b"http" else b"http" targetPort = 443 if startPort == 80 else 80
portSyntax = b"" if crossPort: targetPort = 8443 portSyntax = b":8443" targetDomain = b"example.net" if crossDomain else startDomain locationValue = targetScheme + b"://" + targetDomain + portSyntax + b"/bar" headers = http_headers.Headers({b"location": [locationValue]}) response = Response((b"HTTP", 1, 1), code, b"OK", headers, None) res.callback(response)
@@ -2645,15 +2692,25 @@ def _testRedirectDefault(self, code): self.assertEqual(b"/bar", req2.uri)
host, port = self.reactor.tcpClients.pop()[:2] self.assertEqual(EXAMPLE_COM_IP, host) self.assertEqual(expectedPort, port) self.assertEqual(EXAMPLE_NET_IP if crossDomain else EXAMPLE_COM_IP, host) self.assertEqual(targetPort, port) return req2
def test_redirect301(self): “"” L{client.RedirectAgent} follows redirects on status code 301. “"” self._testRedirectDefault(301)
def test_redirect301Scheme(self): “"” L{client.RedirectAgent} follows cross-scheme redirects. “"” self._testRedirectDefault( 301, crossScheme=True, )
def test_redirect302(self): “"” L{client.RedirectAgent} follows redirects on status code 302. @@ -2672,6 +2729,74 @@ def test_redirect308(self): “"” self._testRedirectDefault(308)
def _sensitiveHeadersTest( self, expectedHostHeader: bytes = b"example.com", **crossKwargs: bool ) -> None: “"” L{client.RedirectAgent} scrubs sensitive headers when redirecting between differing origins. “"” sensitiveHeaderValues = { b"authorization": [b"sensitive-authnz"], b"cookie": [b"sensitive-cookie-data"], b"cookie2": [b"sensitive-cookie2-data"], b"proxy-authorization": [b"sensitive-proxy-auth"], b"wWw-auThentiCate": [b"sensitive-authn"], b"x-custom-sensitive": [b"sensitive-custom"], } otherHeaderValues = {b"x-random-header": [b"x-random-value"]} allHeaders = Headers({**sensitiveHeaderValues, **otherHeaderValues}) redirected = self._testRedirectDefault(301, requestHeaders=allHeaders)
def normHeaders(headers: Headers) -> dict: return {k.lower(): v for (k, v) in headers.getAllRawHeaders()}
sameOriginHeaders = normHeaders(redirected.headers) self.assertEquals( sameOriginHeaders, { b"host": [b"example.com"], **normHeaders(allHeaders), }, )
redirectedElsewhere = self._testRedirectDefault( 301, **crossKwargs, requestHeaders=Headers({**sensitiveHeaderValues, **otherHeaderValues}), ) otherOriginHeaders = normHeaders(redirectedElsewhere.headers) self.assertEquals( otherOriginHeaders, { b"host": [expectedHostHeader], **normHeaders(Headers(otherHeaderValues)), }, )
def test_crossDomainHeaders(self) -> None: “"” L{client.RedirectAgent} scrubs sensitive headers when redirecting between differing domains. “"” self._sensitiveHeadersTest(crossDomain=True, expectedHostHeader=b"example.net")
def test_crossPortHeaders(self) -> None: “"” L{client.RedirectAgent} scrubs sensitive headers when redirecting between differing ports. “"” self._sensitiveHeadersTest( crossPort=True, expectedHostHeader=b"example.com:8443" )
def test_crossSchemeHeaders(self) -> None: “"” L{client.RedirectAgent} scrubs sensitive headers when redirecting between differing schemes. “"” self._sensitiveHeadersTest(crossScheme=True)
def _testRedirectToGet(self, code, method): “"” L{client.RedirectAgent} changes the method to I{GET} when getting @@ -2878,7 +3003,10 @@ def test_responseHistory(self):

class RedirectAgentTests( TestCase, FakeReactorAndConnectMixin, _RedirectAgentTestsMixin, AgentTestsMixin FakeReactorAndConnectMixin, _RedirectAgentTestsMixin, AgentTestsMixin, runtimeTestCase, ): “"” Tests for L{client.RedirectAgent}. @@ -2888,7 +3016,10 @@ def makeAgent(self): “"” @return: a new L{twisted.web.client.RedirectAgent} “"” return client.RedirectAgent(self.buildAgentForWrapperTest(self.reactor)) return client.RedirectAgent( self.buildAgentForWrapperTest(self.reactor), sensitiveHeaderNames=[b"X-Custom-sensitive"], )
def setUp(self): self.reactor = self.createReactor() @@ -2912,7 +3043,10 @@ def test_302OnPost(self):

class BrowserLikeRedirectAgentTests( TestCase, FakeReactorAndConnectMixin, _RedirectAgentTestsMixin, AgentTestsMixin FakeReactorAndConnectMixin, _RedirectAgentTestsMixin, AgentTestsMixin, runtimeTestCase, ): “"” Tests for L{client.BrowserLikeRedirectAgent}. @@ -2923,7 +3057,8 @@ def makeAgent(self): @return: a new L{twisted.web.client.BrowserLikeRedirectAgent} “"” return client.BrowserLikeRedirectAgent( self.buildAgentForWrapperTest(self.reactor) self.buildAgentForWrapperTest(self.reactor), sensitiveHeaderNames=[b"x-Custom-sensitive"], )
def setUp(self):

Related news

CVE-2023-32449: DSA-2023-173: Dell PowerStore Family Security Update for Multiple Vulnerabilities

Dell PowerStore versions prior to 3.5 contain an improper verification of cryptographic signature vulnerability. An attacker can trick a high privileged user to install a malicious binary by bypassing the existing cryptographic signature checks

Gentoo Linux Security Advisory 202301-02

Gentoo Linux Security Advisory 202301-2 - Multiple vulnerabilities have been discovered in Twisted, the worst of which could result in denial of service. Versions less than 22.10.0 are affected.

CVE: Latest News

CVE-2023-50976: Transactions API Authorization by oleiman · Pull Request #14969 · redpanda-data/redpanda