7f963cc
diff -ur m2crypto-0.18/M2Crypto/SSL/Connection.py m2crypto/M2Crypto/SSL/Connection.py
7f963cc
--- m2crypto-0.18/M2Crypto/SSL/Connection.py	2007-06-15 23:34:05.000000000 +0200
7f963cc
+++ m2crypto/M2Crypto/SSL/Connection.py	2007-07-31 23:30:51.000000000 +0200
7f963cc
@@ -37,9 +37,11 @@
af9de46
             self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
af9de46
             self.socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
af9de46
         self._fileno = self.socket.fileno()
af9de46
-        
af9de46
-        self.blocking = self.socket.gettimeout()
af9de46
-        
af9de46
+
af9de46
+        self._timeout = self.socket.gettimeout()
af9de46
+        if self._timeout is None:
af9de46
+            self._timeout = -1.0
af9de46
+
7f963cc
         self.ssl_close_flag = m2.bio_noclose
7f963cc
 
7f963cc
         
7f963cc
@@ -137,7 +139,7 @@
af9de46
         m2.ssl_set_accept_state(self.ssl)
af9de46
 
af9de46
     def accept_ssl(self):
af9de46
-        return m2.ssl_accept(self.ssl)
af9de46
+        return m2.ssl_accept(self.ssl, self._timeout)
af9de46
 
af9de46
     def accept(self):
af9de46
         """Accept an SSL connection. The return value is a pair (ssl, addr) where
7f963cc
@@ -159,7 +161,7 @@
af9de46
         m2.ssl_set_connect_state(self.ssl)
af9de46
 
af9de46
     def connect_ssl(self):
af9de46
-        return m2.ssl_connect(self.ssl)
af9de46
+        return m2.ssl_connect(self.ssl, self._timeout)
af9de46
 
af9de46
     def connect(self, addr):
af9de46
         self.socket.connect(addr)
7f963cc
@@ -186,7 +188,7 @@
af9de46
         return m2.ssl_pending(self.ssl)
af9de46
 
af9de46
     def _write_bio(self, data):
af9de46
-        return m2.ssl_write(self.ssl, data)
af9de46
+        return m2.ssl_write(self.ssl, data, self._timeout)
af9de46
 
af9de46
     def _write_nbio(self, data):
af9de46
         return m2.ssl_write_nbio(self.ssl, data)
7f963cc
@@ -194,7 +196,7 @@
af9de46
     def _read_bio(self, size=1024):
af9de46
         if size <= 0:
af9de46
             raise ValueError, 'size <= 0'
af9de46
-        return m2.ssl_read(self.ssl, size)
af9de46
+        return m2.ssl_read(self.ssl, size, self._timeout)
af9de46
 
af9de46
     def _read_nbio(self, size=1024):
af9de46
         if size <= 0:
7f963cc
@@ -202,13 +204,13 @@
af9de46
         return m2.ssl_read_nbio(self.ssl, size)
af9de46
 
af9de46
     def write(self, data):
af9de46
-        if self.blocking:
af9de46
+        if self._timeout != 0.0:
af9de46
             return self._write_bio(data)
af9de46
         return self._write_nbio(data)
af9de46
     sendall = send = write
af9de46
     
af9de46
     def read(self, size=1024):
af9de46
-        if self.blocking:
af9de46
+        if self._timeout != 0.0:
af9de46
             return self._read_bio(size)
af9de46
         return self._read_nbio(size)
af9de46
     recv = read
7f963cc
@@ -216,7 +218,17 @@
af9de46
     def setblocking(self, mode):
af9de46
         """Set this connection's underlying socket to _mode_."""
af9de46
         self.socket.setblocking(mode)
af9de46
-        self.blocking = mode
af9de46
+        if mode:
af9de46
+            self._timeout = -1.0
af9de46
+        else:
af9de46
+            self._timeout = 0.0
af9de46
+
af9de46
+    def settimeout(self, timeout):
af9de46
+        """Set this connection's underlying socket's timeout to _timeout_."""
af9de46
+        self.socket.settimeout(timeout)
af9de46
+        self._timeout = timeout
af9de46
+        if self._timeout is None:
af9de46
+            self._timeout = -1.0
af9de46
 
af9de46
     def fileno(self):
af9de46
         return self.socket.fileno()
7f963cc
@@ -293,15 +305,8 @@
af9de46
         """Set the cipher suites for this connection."""
af9de46
         return m2.ssl_set_cipher_list(self.ssl, cipher_list)
af9de46
 
af9de46
-    def makefile(self, mode='rb', bufsize='ignored'):
af9de46
-        r = 'r' in mode or '+' in mode
af9de46
-        w = 'w' in mode or 'a' in mode or '+' in mode
af9de46
-        b = 'b' in mode
af9de46
-        m2mode = ['', 'r'][r] + ['', 'w'][w] + ['', 'b'][b]      
af9de46
-        # XXX Need to dup().
af9de46
-        bio = BIO.BIO(self.sslbio, _close_cb=self.close)
af9de46
-        m2.bio_do_handshake(bio._ptr())
af9de46
-        return BIO.IOBuffer(bio, m2mode, _pyfree=0)
af9de46
+    def makefile(self, mode='rb', bufsize=-1):
af9de46
+        return socket._fileobject(self, mode, bufsize)
af9de46
 
af9de46
     def getsockname(self):
af9de46
         return self.socket.getsockname()
7f963cc
diff -ur m2crypto-0.18/M2Crypto/SSL/__init__.py m2crypto/M2Crypto/SSL/__init__.py
7f963cc
--- m2crypto-0.18/M2Crypto/SSL/__init__.py	2006-03-20 20:26:28.000000000 +0100
7f963cc
+++ m2crypto/M2Crypto/SSL/__init__.py	2007-07-31 23:29:21.000000000 +0200
af9de46
@@ -2,11 +2,14 @@
af9de46
 
af9de46
 Copyright (c) 1999-2004 Ng Pheng Siong. All rights reserved."""
af9de46
 
af9de46
+import socket
af9de46
+
af9de46
 # M2Crypto
af9de46
 from M2Crypto import m2
af9de46
 
af9de46
 class SSLError(Exception): pass
af9de46
-m2.ssl_init(SSLError)
af9de46
+class SSLTimeoutError(SSLError, socket.timeout): pass
af9de46
+m2.ssl_init(SSLError, SSLTimeoutError)
af9de46
 
af9de46
 # M2Crypto.SSL
af9de46
 from Cipher import Cipher, Cipher_Stack
7f963cc
diff -ur m2crypto-0.18/SWIG/_ssl.i m2crypto/SWIG/_ssl.i
7f963cc
--- m2crypto-0.18/SWIG/_ssl.i	2007-06-05 02:30:11.000000000 +0200
7f963cc
+++ m2crypto/SWIG/_ssl.i	2007-08-01 00:06:34.000000000 +0200
af9de46
@@ -8,10 +8,13 @@
af9de46
 
af9de46
 %{
af9de46
 #include <pythread.h>
af9de46
+#include <limits.h>
af9de46
 #include <openssl/bio.h>
af9de46
 #include <openssl/dh.h>
af9de46
 #include <openssl/ssl.h>
af9de46
 #include <openssl/x509.h>
af9de46
+#include <poll.h>
af9de46
+#include <sys/time.h>
af9de46
 %}
af9de46
 
af9de46
 %apply Pointer NONNULL { SSL_CTX * };
af9de46
@@ -142,6 +145,11 @@
af9de46
 %rename(ssl_session_get_timeout) SSL_SESSION_get_timeout;
af9de46
 extern long SSL_SESSION_get_timeout(CONST SSL_SESSION *);
af9de46
 
af9de46
+extern PyObject *ssl_accept(SSL *ssl, double timeout = -1);
af9de46
+extern PyObject *ssl_connect(SSL *ssl, double timeout = -1);
af9de46
+extern PyObject *ssl_read(SSL *ssl, int num, double timeout = -1);
af9de46
+extern int ssl_write(SSL *ssl, PyObject *blob, double timeout = -1);
af9de46
+
af9de46
 %constant int ssl_error_none              = SSL_ERROR_NONE;
af9de46
 %constant int ssl_error_ssl               = SSL_ERROR_SSL;
af9de46
 %constant int ssl_error_want_read         = SSL_ERROR_WANT_READ;
7f963cc
@@ -197,14 +205,19 @@
af9de46
 %constant int SSL_MODE_ACCEPT_MOVING_WRITE_BUFFER = SSL_MODE_ENABLE_PARTIAL_WRITE;
af9de46
 %constant int SSL_MODE_AUTO_RETRY           = SSL_MODE_AUTO_RETRY;
af9de46
 
af9de46
+%ignore ssl_handle_error;
af9de46
+%ignore ssl_sleep_with_timeout;
af9de46
 %inline %{
af9de46
 static PyObject *_ssl_err;
af9de46
+static PyObject *_ssl_timeout_err;
af9de46
 
af9de46
-void ssl_init(PyObject *ssl_err) {
af9de46
+void ssl_init(PyObject *ssl_err, PyObject *ssl_timeout_err) {
af9de46
     SSL_library_init();
af9de46
     SSL_load_error_strings();
af9de46
     Py_INCREF(ssl_err);
af9de46
+    Py_INCREF(ssl_timeout_err);
af9de46
     _ssl_err = ssl_err;
af9de46
+    _ssl_timeout_err = ssl_timeout_err;
af9de46
 }
af9de46
 
af9de46
 void ssl_ctx_passphrase_callback(SSL_CTX *ctx, PyObject *pyfunc) {
7f963cc
@@ -358,36 +371,130 @@
af9de46
     return ret;
af9de46
 }
af9de46
 
af9de46
-PyObject *ssl_accept(SSL *ssl) {
af9de46
+static void ssl_handle_error(int ssl_err, int ret) {
af9de46
+    int err;
af9de46
+
af9de46
+    switch (ssl_err) {
af9de46
+        case SSL_ERROR_SSL:
af9de46
+            PyErr_SetString(_ssl_err,
af9de46
+                            ERR_reason_error_string(ERR_get_error()));
af9de46
+            break;
af9de46
+        case SSL_ERROR_SYSCALL:
af9de46
+            err = ERR_get_error();
af9de46
+            if (err)
af9de46
+                PyErr_SetString(_ssl_err, ERR_reason_error_string(err));
af9de46
+            else if (ret == 0)
af9de46
+                PyErr_SetString(_ssl_err, "unexpected eof");
af9de46
+            else if (ret == -1)
af9de46
+                PyErr_SetFromErrno(_ssl_err);
af9de46
+            else
af9de46
+                assert(0);
af9de46
+            break;
af9de46
+		default:
af9de46
+            PyErr_SetString(_ssl_err, "unexpected SSL error");
af9de46
+     }
af9de46
+}
af9de46
+
af9de46
+static int ssl_sleep_with_timeout(SSL *ssl, const struct timeval *start,
af9de46
+                                  double timeout, int ssl_err) {
af9de46
+    struct pollfd fd;
af9de46
+    struct timeval tv;
7f963cc
+    int ms, tmp;
af9de46
+
af9de46
+    assert(timeout > 0);
af9de46
+ again:
af9de46
+    gettimeofday(&tv, NULL);
af9de46
+    /* tv >= start */
af9de46
+    if ((timeout + start->tv_sec - tv.tv_sec) > INT_MAX / 1000)
af9de46
+        ms = -1;
af9de46
+    else {
af9de46
+        int fract;
af9de46
+
af9de46
+        ms = ((start->tv_sec + (int)timeout) - tv.tv_sec) * 1000;
af9de46
+        fract = (start->tv_usec + (timeout - (int)timeout) * 1000000
af9de46
+                 - tv.tv_usec + 999) / 1000;
af9de46
+        if (ms > 0 && fract > INT_MAX - ms)
af9de46
+            ms = -1;
af9de46
+        else {
af9de46
+            ms += fract;
af9de46
+            if (ms <= 0)
af9de46
+                goto timeout;
af9de46
+        }
af9de46
+    }
af9de46
+    switch (ssl_err) {
af9de46
+	    case SSL_ERROR_WANT_READ:
af9de46
+            fd.fd = SSL_get_rfd(ssl);
af9de46
+            fd.events = POLLIN;
af9de46
+            break;
af9de46
+
af9de46
+	    case SSL_ERROR_WANT_WRITE:
af9de46
+            fd.fd = SSL_get_wfd(ssl);
af9de46
+            fd.events = POLLOUT;
af9de46
+            break;
af9de46
+
af9de46
+	    case SSL_ERROR_WANT_X509_LOOKUP:
af9de46
+            return 0; /* FIXME: is this correct? */
af9de46
+
af9de46
+	    default:
af9de46
+            assert(0);
af9de46
+    }
af9de46
+    if (fd.fd == -1) {
af9de46
+        PyErr_SetString(_ssl_err, "timeout on a non-FD SSL");
af9de46
+        return -1;
af9de46
+    }
7f963cc
+    Py_BEGIN_ALLOW_THREADS
7f963cc
+    tmp = poll(&fd, 1, ms);
7f963cc
+    Py_END_ALLOW_THREADS
7f963cc
+    switch (tmp) {
af9de46
+    	case 1:
af9de46
+            return 0;
af9de46
+    	case 0:
af9de46
+            goto timeout;
af9de46
+    	case -1:
af9de46
+            if (errno == EINTR)
af9de46
+                goto again;
af9de46
+            PyErr_SetFromErrno(_ssl_err);
af9de46
+            return -1;
af9de46
+    }
af9de46
+    return 0;
af9de46
+
af9de46
+ timeout:
af9de46
+    PyErr_SetNone(_ssl_timeout_err);
af9de46
+    return -1;
af9de46
+}
af9de46
+
af9de46
+PyObject *ssl_accept(SSL *ssl, double timeout) {
af9de46
     PyObject *obj = NULL;
af9de46
-    int r, err;
af9de46
+    int r, ssl_err;
af9de46
+    struct timeval tv;
af9de46
 
af9de46
+    if (timeout > 0)
af9de46
+        gettimeofday(&tv, NULL);
af9de46
+ again:
7f963cc
     Py_BEGIN_ALLOW_THREADS
af9de46
     r = SSL_accept(ssl);
af9de46
+    ssl_err = SSL_get_error(ssl, r);
7f963cc
     Py_END_ALLOW_THREADS
af9de46
 
af9de46
 
af9de46
-    switch (SSL_get_error(ssl, r)) {
af9de46
+    switch (ssl_err) {
af9de46
         case SSL_ERROR_NONE:
af9de46
         case SSL_ERROR_ZERO_RETURN:
af9de46
             obj = PyInt_FromLong((long)1);
af9de46
             break;
af9de46
         case SSL_ERROR_WANT_WRITE:
af9de46
         case SSL_ERROR_WANT_READ:
af9de46
-            obj = PyInt_FromLong((long)0);
af9de46
-            break;
af9de46
-        case SSL_ERROR_SSL:
af9de46
-            PyErr_SetString(_ssl_err, ERR_reason_error_string(ERR_get_error()));
af9de46
+            if (timeout <= 0) {
af9de46
+                obj = PyInt_FromLong((long)0);
af9de46
+                break;
af9de46
+            }
af9de46
+            if (ssl_sleep_with_timeout(ssl, &tv, timeout, ssl_err) == 0)
af9de46
+                goto again;
af9de46
             obj = NULL;
af9de46
             break;
af9de46
+        case SSL_ERROR_SSL:
af9de46
         case SSL_ERROR_SYSCALL:
af9de46
-            err = ERR_get_error();
af9de46
-            if (err)
af9de46
-                PyErr_SetString(_ssl_err, ERR_reason_error_string(err));
af9de46
-            else if (r == 0)
af9de46
-                PyErr_SetString(_ssl_err, "unexpected eof");
af9de46
-            else if (r == -1)
af9de46
-                PyErr_SetFromErrno(_ssl_err);
af9de46
+            ssl_handle_error(ssl_err, r);
af9de46
             obj = NULL;
af9de46
             break;
af9de46
     }
7f963cc
@@ -396,36 +503,38 @@
af9de46
     return obj;
af9de46
 }
af9de46
 
af9de46
-PyObject *ssl_connect(SSL *ssl) {
af9de46
+PyObject *ssl_connect(SSL *ssl, double timeout) {
af9de46
     PyObject *obj = NULL;
af9de46
-    int r, err;
af9de46
+    int r, ssl_err;
af9de46
+    struct timeval tv;
af9de46
 
af9de46
+    if (timeout > 0)
af9de46
+        gettimeofday(&tv, NULL);
af9de46
+ again:
7f963cc
     Py_BEGIN_ALLOW_THREADS
af9de46
     r = SSL_connect(ssl);
af9de46
+    ssl_err = SSL_get_error(ssl, r);
7f963cc
     Py_END_ALLOW_THREADS
af9de46
 
af9de46
     
af9de46
-    switch (SSL_get_error(ssl, r)) {
af9de46
+    switch (ssl_err) {
af9de46
         case SSL_ERROR_NONE:
af9de46
         case SSL_ERROR_ZERO_RETURN:
af9de46
             obj = PyInt_FromLong((long)1);
af9de46
             break;
af9de46
         case SSL_ERROR_WANT_WRITE:
af9de46
         case SSL_ERROR_WANT_READ:
af9de46
-            obj = PyInt_FromLong((long)0);
af9de46
-            break;
af9de46
-        case SSL_ERROR_SSL:
af9de46
-            PyErr_SetString(_ssl_err, ERR_reason_error_string(ERR_get_error()));
af9de46
+            if (timeout <= 0) {
af9de46
+                obj = PyInt_FromLong((long)0);
af9de46
+                break;
af9de46
+            }
af9de46
+            if (ssl_sleep_with_timeout(ssl, &tv, timeout, ssl_err) == 0)
af9de46
+                goto again;
af9de46
             obj = NULL;
af9de46
             break;
af9de46
+        case SSL_ERROR_SSL:
af9de46
         case SSL_ERROR_SYSCALL:
af9de46
-            err = ERR_get_error();
af9de46
-            if (err)
af9de46
-                PyErr_SetString(_ssl_err, ERR_reason_error_string(err));
af9de46
-            else if (r == 0)
af9de46
-                PyErr_SetString(_ssl_err, "unexpected eof");
af9de46
-            else if (r == -1)
af9de46
-                PyErr_SetFromErrno(_ssl_err);
af9de46
+            ssl_handle_error(ssl_err, r);
af9de46
             obj = NULL;
af9de46
             break;
af9de46
     }
7f963cc
@@ -438,10 +547,11 @@
af9de46
     SSL_set_shutdown(ssl, mode);
af9de46
 }
af9de46
 
af9de46
-PyObject *ssl_read(SSL *ssl, int num) {
af9de46
+PyObject *ssl_read(SSL *ssl, int num, double timeout) {
af9de46
     PyObject *obj = NULL;
af9de46
     void *buf;
af9de46
-    int r, err;
af9de46
+    int r;
af9de46
+    struct timeval tv;
af9de46
 
7f963cc
     if (!(buf = PyMem_Malloc(num))) {
7f963cc
         PyErr_SetString(PyExc_MemoryError, "ssl_read");
7f963cc
@@ -449,37 +559,44 @@
7f963cc
     }
af9de46
 
af9de46
 
af9de46
+    if (timeout > 0)
af9de46
+        gettimeofday(&tv, NULL);
af9de46
+ again:
7f963cc
     Py_BEGIN_ALLOW_THREADS
af9de46
     r = SSL_read(ssl, buf, num);
7f963cc
     Py_END_ALLOW_THREADS
af9de46
 
af9de46
 
af9de46
-    switch (SSL_get_error(ssl, r)) {
af9de46
-        case SSL_ERROR_NONE:
af9de46
-        case SSL_ERROR_ZERO_RETURN:
af9de46
-            buf = PyMem_Realloc(buf, r);
af9de46
-            obj = PyString_FromStringAndSize(buf, r);
af9de46
-            break;
af9de46
-        case SSL_ERROR_WANT_WRITE:
af9de46
-        case SSL_ERROR_WANT_READ:
af9de46
-        case SSL_ERROR_WANT_X509_LOOKUP:
af9de46
-            Py_INCREF(Py_None);
af9de46
-            obj = Py_None;
af9de46
-            break;
af9de46
-        case SSL_ERROR_SSL:
af9de46
-            PyErr_SetString(_ssl_err, ERR_reason_error_string(ERR_get_error()));
af9de46
-            obj = NULL;
af9de46
-            break;
af9de46
-        case SSL_ERROR_SYSCALL:
af9de46
-            err = ERR_get_error();
af9de46
-            if (err)
af9de46
-                PyErr_SetString(_ssl_err, ERR_reason_error_string(err));
af9de46
-            else if (r == 0)
af9de46
-                PyErr_SetString(_ssl_err, "unexpected eof");
af9de46
-            else if (r == -1)
af9de46
-                PyErr_SetFromErrno(_ssl_err);
af9de46
-            obj = NULL;
af9de46
-            break;
af9de46
+    if (r >= 0) {
af9de46
+        buf = PyMem_Realloc(buf, r);
af9de46
+        obj = PyString_FromStringAndSize(buf, r);
af9de46
+    } else {
af9de46
+        int ssl_err;
af9de46
+
af9de46
+        ssl_err = SSL_get_error(ssl, r);
af9de46
+        switch (ssl_err) {
af9de46
+            case SSL_ERROR_NONE:
af9de46
+            case SSL_ERROR_ZERO_RETURN:
af9de46
+                assert(0);
af9de46
+
af9de46
+            case SSL_ERROR_WANT_WRITE:
af9de46
+            case SSL_ERROR_WANT_READ:
af9de46
+            case SSL_ERROR_WANT_X509_LOOKUP:
af9de46
+                if (timeout <= 0) {
af9de46
+                    Py_INCREF(Py_None);
af9de46
+                    obj = Py_None;
af9de46
+                    break;
af9de46
+                }
af9de46
+                if (ssl_sleep_with_timeout(ssl, &tv, timeout, ssl_err) == 0)
af9de46
+                    goto again;
af9de46
+                obj = NULL;
af9de46
+                break;
af9de46
+            case SSL_ERROR_SSL:
af9de46
+            case SSL_ERROR_SYSCALL:
af9de46
+                ssl_handle_error(ssl_err, r);
af9de46
+                obj = NULL;
af9de46
+                break;
af9de46
+        }
af9de46
     }
af9de46
     PyMem_Free(buf);
af9de46
 
7f963cc
@@ -537,22 +654,26 @@
af9de46
     return obj;
af9de46
 }
af9de46
 
af9de46
-int ssl_write(SSL *ssl, PyObject *blob) {
af9de46
+int ssl_write(SSL *ssl, PyObject *blob, double timeout) {
af9de46
     const void *buf;
af9de46
-    int len, r, err, ret;
af9de46
+    int len, r, ssl_err, ret;
af9de46
+    struct timeval tv;
af9de46
 
7f963cc
 
7f963cc
     if (m2_PyObject_AsReadBufferInt(blob, &buf, &len) == -1) {
7f963cc
         return -1;
af9de46
     }
af9de46
 
af9de46
-    
af9de46
+    if (timeout > 0)
af9de46
+        gettimeofday(&tv, NULL);
af9de46
+ again:
7f963cc
     Py_BEGIN_ALLOW_THREADS
af9de46
     r = SSL_write(ssl, buf, len);
af9de46
+    ssl_err = SSL_get_error(ssl, r);
7f963cc
     Py_END_ALLOW_THREADS
af9de46
 
af9de46
 
af9de46
-    switch (SSL_get_error(ssl, r)) {
af9de46
+    switch (ssl_err) {
af9de46
         case SSL_ERROR_NONE:
af9de46
         case SSL_ERROR_ZERO_RETURN:
af9de46
             ret = r;
7f963cc
@@ -560,20 +681,17 @@
af9de46
         case SSL_ERROR_WANT_WRITE:
af9de46
         case SSL_ERROR_WANT_READ:
af9de46
         case SSL_ERROR_WANT_X509_LOOKUP:
af9de46
+            if (timeout <= 0) {
af9de46
+                ret = -1;
af9de46
+                break;
af9de46
+            }
af9de46
+            if (ssl_sleep_with_timeout(ssl, &tv, timeout, ssl_err) == 0)
af9de46
+                goto again;
af9de46
             ret = -1;
af9de46
             break;
af9de46
         case SSL_ERROR_SSL:
af9de46
-            PyErr_SetString(_ssl_err, ERR_reason_error_string(ERR_get_error()));
af9de46
-            ret = -1;
af9de46
-            break;
af9de46
         case SSL_ERROR_SYSCALL:
af9de46
-            err = ERR_get_error();
af9de46
-            if (err)
af9de46
-                PyErr_SetString(_ssl_err, ERR_reason_error_string(ERR_get_error()));
af9de46
-            else if (r == 0)
af9de46
-                PyErr_SetString(_ssl_err, "unexpected eof");
af9de46
-            else if (r == -1)
af9de46
-                PyErr_SetFromErrno(_ssl_err);
af9de46
+            ssl_handle_error(ssl_err, r);
af9de46
         default:
af9de46
             ret = -1;
af9de46
     }
7f963cc
diff -ur m2crypto-0.18/tests/test_ssl.py m2crypto/tests/test_ssl.py
7f963cc
--- m2crypto-0.18/tests/test_ssl.py	2007-07-02 22:25:45.000000000 +0200
7f963cc
+++ m2crypto/tests/test_ssl.py	2007-07-31 23:29:21.000000000 +0200
7f963cc
@@ -887,6 +887,53 @@
7f963cc
 
7f963cc
 class TwistedSSLClientTestCase(BaseSSLClientTestCase):
af9de46
 
af9de46
+    def test_timeout(self):
af9de46
+        pid = self.start_server(self.args)
af9de46
+        try:
af9de46
+            ctx = SSL.Context()
af9de46
+            s = SSL.Connection(ctx)
af9de46
+            # Just a really small number so we can timeout
af9de46
+            s.settimeout(0.000000000000000000000000000001)
af9de46
+            self.assertRaises(SSL.SSLTimeoutError, s.connect, self.srv_addr)
af9de46
+            s.close()
af9de46
+        finally:
af9de46
+            self.stop_server(pid)
af9de46
+
af9de46
+    def test_makefile_timeout(self):
af9de46
+        # httpslib uses makefile to read the response
af9de46
+        pid = self.start_server(self.args)
af9de46
+        try:
af9de46
+            from M2Crypto import httpslib
af9de46
+            c = httpslib.HTTPS(srv_host, srv_port)
af9de46
+            c.putrequest('GET', '/')
af9de46
+            c.putheader('Accept', 'text/html')
af9de46
+            c.putheader('Accept', 'text/plain')
af9de46
+            c.endheaders()
af9de46
+            c._conn.sock.settimeout(100)
af9de46
+            err, msg, headers = c.getreply()
af9de46
+            assert err == 200, err
af9de46
+            f = c.getfile()
af9de46
+            data = f.read()
af9de46
+            c.close()
af9de46
+        finally:
af9de46
+            self.stop_server(pid)
af9de46
+        self.failIf(string.find(data, 's_server -quiet -www') == -1)
af9de46
+
af9de46
+    def test_makefile_timeout_fires(self):
af9de46
+        pid = self.start_server(self.args)
af9de46
+        try:
af9de46
+            from M2Crypto import httpslib
af9de46
+            c = httpslib.HTTPS(srv_host, srv_port)
af9de46
+            c.putrequest('GET', '/')
af9de46
+            c.putheader('Accept', 'text/html')
af9de46
+            c.putheader('Accept', 'text/plain')
af9de46
+            c.endheaders()
af9de46
+            c._conn.sock.settimeout(0.0000000001)
af9de46
+            self.assertRaises(socket.timeout, c.getreply)
af9de46
+            c.close()
af9de46
+        finally:
af9de46
+            self.stop_server(pid)
af9de46
+
af9de46
     def test_twisted_wrapper(self):
7f963cc
         # Test only when twisted and ZopeInterfaces are present
7f963cc
         try: