| Viewing file:  injectionhelpers.py (5.5 KB)      -rw-r--r-- Select action/file-type:
 
  (+) |  (+) |  (+) | Code (+) | Session (+) |  (+) | SDB (+) |  (+) |  (+) |  (+) |  (+) |  (+) | 
 
"""Helpers for URI and method injection tests.
 
 @see: U{CVE-2019-12387}
 """
 
 import string
 
 
 UNPRINTABLE_ASCII = (
 frozenset(range(0, 128)) -
 frozenset(bytearray(string.printable, 'ascii'))
 )
 
 NONASCII = frozenset(range(128, 256))
 
 
 
 class MethodInjectionTestsMixin(object):
 """
 A mixin that runs HTTP method injection tests.  Define
 L{MethodInjectionTestsMixin.attemptRequestWithMaliciousMethod} in
 a L{twisted.trial.unittest.SynchronousTestCase} subclass to test
 how HTTP client code behaves when presented with malicious HTTP
 methods.
 
 @see: U{CVE-2019-12387}
 """
 
 def attemptRequestWithMaliciousMethod(self, method):
 """
 Attempt to send a request with the given method.  This should
 synchronously raise a L{ValueError} if either is invalid.
 
 @param method: the method (e.g. C{GET\x00})
 
 @param uri: the URI
 
 @type method:
 """
 raise NotImplementedError()
 
 
 def test_methodWithCLRFRejected(self):
 """
 Issuing a request with a method that contains a carriage
 return and line feed fails with a L{ValueError}.
 """
 with self.assertRaises(ValueError) as cm:
 method = b"GET\r\nX-Injected-Header: value"
 self.attemptRequestWithMaliciousMethod(method)
 self.assertRegex(str(cm.exception), "^Invalid method")
 
 
 def test_methodWithUnprintableASCIIRejected(self):
 """
 Issuing a request with a method that contains unprintable
 ASCII characters fails with a L{ValueError}.
 """
 for c in UNPRINTABLE_ASCII:
 method = b"GET%s" % (bytearray([c]),)
 with self.assertRaises(ValueError) as cm:
 self.attemptRequestWithMaliciousMethod(method)
 self.assertRegex(str(cm.exception), "^Invalid method")
 
 
 def test_methodWithNonASCIIRejected(self):
 """
 Issuing a request with a method that contains non-ASCII
 characters fails with a L{ValueError}.
 """
 for c in NONASCII:
 method = b"GET%s" % (bytearray([c]),)
 with self.assertRaises(ValueError) as cm:
 self.attemptRequestWithMaliciousMethod(method)
 self.assertRegex(str(cm.exception), "^Invalid method")
 
 
 
 class URIInjectionTestsMixin(object):
 """
 A mixin that runs HTTP URI injection tests.  Define
 L{MethodInjectionTestsMixin.attemptRequestWithMaliciousURI} in a
 L{twisted.trial.unittest.SynchronousTestCase} subclass to test how
 HTTP client code behaves when presented with malicious HTTP
 URIs.
 """
 
 def attemptRequestWithMaliciousURI(self, method):
 """
 Attempt to send a request with the given URI.  This should
 synchronously raise a L{ValueError} if either is invalid.
 
 @param uri: the URI.
 
 @type method:
 """
 raise NotImplementedError()
 
 
 def test_hostWithCRLFRejected(self):
 """
 Issuing a request with a URI whose host contains a carriage
 return and line feed fails with a L{ValueError}.
 """
 with self.assertRaises(ValueError) as cm:
 uri = b"http://twisted\r\n.invalid/path"
 self.attemptRequestWithMaliciousURI(uri)
 self.assertRegex(str(cm.exception), "^Invalid URI")
 
 
 def test_hostWithWithUnprintableASCIIRejected(self):
 """
 Issuing a request with a URI whose host contains unprintable
 ASCII characters fails with a L{ValueError}.
 """
 for c in UNPRINTABLE_ASCII:
 uri = b"http://twisted%s.invalid/OK" % (bytearray([c]),)
 with self.assertRaises(ValueError) as cm:
 self.attemptRequestWithMaliciousURI(uri)
 self.assertRegex(str(cm.exception), "^Invalid URI")
 
 
 def test_hostWithNonASCIIRejected(self):
 """
 Issuing a request with a URI whose host contains non-ASCII
 characters fails with a L{ValueError}.
 """
 for c in NONASCII:
 uri = b"http://twisted%s.invalid/OK" % (bytearray([c]),)
 with self.assertRaises(ValueError) as cm:
 self.attemptRequestWithMaliciousURI(uri)
 self.assertRegex(str(cm.exception), "^Invalid URI")
 
 
 def test_pathWithCRLFRejected(self):
 """
 Issuing a request with a URI whose path contains a carriage
 return and line feed fails with a L{ValueError}.
 """
 with self.assertRaises(ValueError) as cm:
 uri = b"http://twisted.invalid/\r\npath"
 self.attemptRequestWithMaliciousURI(uri)
 self.assertRegex(str(cm.exception), "^Invalid URI")
 
 
 def test_pathWithWithUnprintableASCIIRejected(self):
 """
 Issuing a request with a URI whose path contains unprintable
 ASCII characters fails with a L{ValueError}.
 """
 for c in UNPRINTABLE_ASCII:
 uri = b"http://twisted.invalid/OK%s" % (bytearray([c]),)
 with self.assertRaises(ValueError) as cm:
 self.attemptRequestWithMaliciousURI(uri)
 self.assertRegex(str(cm.exception), "^Invalid URI")
 
 
 def test_pathWithNonASCIIRejected(self):
 """
 Issuing a request with a URI whose path contains non-ASCII
 characters fails with a L{ValueError}.
 """
 for c in NONASCII:
 uri = b"http://twisted.invalid/OK%s" % (bytearray([c]),)
 with self.assertRaises(ValueError) as cm:
 self.attemptRequestWithMaliciousURI(uri)
 self.assertRegex(str(cm.exception), "^Invalid URI")
 
 |