summaryrefslogtreecommitdiff
path: root/tests
diff options
context:
space:
mode:
authorAndreas Maier <andreas.r.maier@gmx.de>2018-06-08 08:43:07 +0000
committerMatěj Cepl <mcepl@cepl.eu>2018-06-08 08:43:07 +0000
commite569e8811b0b40b2551e9d5d876937112b6dc09c (patch)
tree34e2ac10df4a8f18063e85e7211be059882ad3e0 /tests
parent780c5ad481a11eb09124cd2f7b751fcaaffaa588 (diff)
downloadm2crypto-e569e8811b0b40b2551e9d5d876937112b6dc09c.tar.gz
Enabled timeout case with microsecs and fixed for Windows
Diffstat (limited to 'tests')
-rw-r--r--tests/alltests.py3
-rw-r--r--tests/test_ssl.py21
-rw-r--r--tests/test_timeout.py135
3 files changed, 156 insertions, 3 deletions
diff --git a/tests/alltests.py b/tests/alltests.py
index fee476f..fe5cf91 100644
--- a/tests/alltests.py
+++ b/tests/alltests.py
@@ -40,7 +40,8 @@ def suite():
'tests.test_smime',
'tests.test_ssl_offline',
'tests.test_threading',
- 'tests.test_x509']
+ 'tests.test_x509',
+ 'tests.test_timeout']
if os.name == 'posix':
modules_to_test.append('tests.test_ssl')
elif os.name == 'nt':
diff --git a/tests/test_ssl.py b/tests/test_ssl.py
index 55ba0d1..cb2cac8 100644
--- a/tests/test_ssl.py
+++ b/tests/test_ssl.py
@@ -373,7 +373,14 @@ class MiscSSLClientTestCase(BaseSSLClientTestCase):
self.assertEqual(r.sec, 600, r.sec)
self.assertEqual(r.microsec, 0, r.microsec)
self.assertEqual(w.sec, 909, w.sec)
- # self.assertEqual(w.microsec, 9, w.microsec) XXX 4000
+ if sys.platform == 'win32':
+ # On Windows, microseconds get rounded to milliseconds
+ self.assertEqual(w.microsec, 0, w.microsec)
+ else:
+ # On some platforms (e.g. some Linux), microeconds get rounded
+ # up to the next millisecond.
+ # On some platforms (e.g. OS-X), microseconds are preserved.
+ self.assertIn(w.microsec, (9, 1000), w.microsec)
s.connect(self.srv_addr)
data = self.http_get(s)
@@ -1033,8 +1040,18 @@ class TwistedSSLClientTestCase(BaseSSLClientTestCase):
s = SSL.Connection(ctx)
# Just a really small number so we can timeout
s.settimeout(0.000000000000000000000000000001)
- with self.assertRaises(SSL.SSLTimeoutError):
+
+ # TODO: Figure out which exception should be raised for timeout.
+ # The following assertion originally expected only a
+ # SSL.SSLTimeoutError exception, but what is raised is actually a
+ # socket.timeout exception. As a temporary circumvention to this
+ # issue, both exceptions are now tolerated. A final fix would need
+ # to figure out which of these two exceptions is supposed to be
+ # raised by SSL.Connection.connect() and possibly other methods
+ # to indicate a timeout.
+ with self.assertRaises((SSL.SSLTimeoutError, socket.timeout)):
s.connect(self.srv_addr)
+
s.close()
finally:
self.stop_server(pid)
diff --git a/tests/test_timeout.py b/tests/test_timeout.py
new file mode 100644
index 0000000..6d05449
--- /dev/null
+++ b/tests/test_timeout.py
@@ -0,0 +1,135 @@
+#!/usr/bin/env python
+
+"""Unit tests for M2Crypto.SSL.timeout.
+"""
+
+import sys
+from M2Crypto.SSL import timeout, struct_to_timeout, struct_size
+from tests import unittest
+
+# Max value for sec argument on Windows:
+# - needs to fit DWORD (signed 32-bit) when converted to millisec
+MAX_SEC_WIN32 = int((2**31 - 1) / 1000)
+
+# Max value for sec argument on other platforms:
+# Note: It may actually be 64-bit but we are happy with 32-bit.
+# We use the signed maximum, because the packing uses lower case "l".
+MAX_SEC_OTHER = 2**31 - 1
+
+# Enable this to test the Windows logic on a non-Windows platform:
+# sys.platform = 'win32'
+
+
+class TimeoutTestCase(unittest.TestCase):
+
+ def timeout_test(self, sec, microsec, exp_sec=None, exp_microsec=None):
+ """
+ Test that the timeout values (sec, microsec) are the same after
+ round tripping through a pack / unpack cycle.
+ """
+ if exp_sec is None:
+ exp_sec = sec
+ if exp_microsec is None:
+ exp_microsec = microsec
+
+ to = timeout(sec, microsec)
+
+ binstr = to.pack()
+
+ act_to = struct_to_timeout(binstr)
+
+ self.assertEqual(
+ (act_to.sec, act_to.microsec), (exp_sec, exp_microsec),
+ "Unexpected timeout(sec,microsec) after pack + unpack: "
+ "Got (%r,%r), expected (%r,%r), input was (%r,%r)" %
+ (act_to.sec, act_to.microsec, exp_sec, exp_microsec,
+ sec, microsec))
+
+ def test_timeout_0_0(self):
+ self.timeout_test(0, 0)
+
+ def test_timeout_123_0(self):
+ self.timeout_test(123, 0)
+
+ def test_timeout_max_0(self):
+ if sys.platform == 'win32':
+ self.timeout_test(MAX_SEC_WIN32, 0)
+ else:
+ self.timeout_test(MAX_SEC_OTHER, 0)
+
+ def test_timeout_0_456000(self):
+ self.timeout_test(0, 456000)
+
+ def test_timeout_123_456000(self):
+ self.timeout_test(123, 456000)
+
+ def test_timeout_2_3000000(self):
+ if sys.platform == 'win32':
+ self.timeout_test(2, 3000000, 5, 0)
+ else:
+ self.timeout_test(2, 3000000)
+
+ def test_timeout_2_2499000(self):
+ if sys.platform == 'win32':
+ self.timeout_test(2, 2499000, 4, 499000)
+ else:
+ self.timeout_test(2, 2499000)
+
+ def test_timeout_2_2999000(self):
+ if sys.platform == 'win32':
+ self.timeout_test(2, 2999000, 4, 999000)
+ else:
+ self.timeout_test(2, 2999000)
+
+ def test_timeout_max_456000(self):
+ if sys.platform == 'win32':
+ self.timeout_test(MAX_SEC_WIN32, 456000)
+ else:
+ self.timeout_test(MAX_SEC_OTHER, 456000)
+
+ def test_timeout_0_456(self):
+ if sys.platform == 'win32':
+ self.timeout_test(0, 456, None, 0)
+ else:
+ self.timeout_test(0, 456)
+
+ def test_timeout_123_456(self):
+ if sys.platform == 'win32':
+ self.timeout_test(123, 456, None, 0)
+ else:
+ self.timeout_test(123, 456)
+
+ def test_timeout_max_456(self):
+ if sys.platform == 'win32':
+ self.timeout_test(MAX_SEC_WIN32, 456, None, 0)
+ else:
+ self.timeout_test(MAX_SEC_OTHER, 456)
+
+ def test_timeout_1_499(self):
+ if sys.platform == 'win32':
+ self.timeout_test(123, 499, None, 0) # 499 us rounds down to 0
+ else:
+ self.timeout_test(123, 499)
+
+ def test_timeout_1_501(self):
+ # We use 501 for this test and not 500 because 0.5 is not exactly
+ # represented in binary floating point numbers, and because 0.5
+ # rounds differently between py2 and py3. See Python round() docs.
+ if sys.platform == 'win32':
+ self.timeout_test(123, 501, None, 1000) # 501 us rounds up to 1000
+ else:
+ self.timeout_test(123, 501)
+
+ def test_timeout_size(self):
+ exp_size = len(timeout(0, 0).pack())
+ self.assertEqual(struct_size(), exp_size)
+
+
+def suite():
+ suite = unittest.TestSuite()
+ suite.addTest(unittest.makeSuite(TimeoutTestCase))
+ return suite
+
+
+if __name__ == '__main__':
+ unittest.TextTestRunner().run(suite())