summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMike Bayer <mike_mp@zzzcomputing.com>2013-07-04 13:18:00 -0400
committerMike Bayer <mike_mp@zzzcomputing.com>2013-07-04 13:18:00 -0400
commitf390639bf1a7a5a2a47bcd6df7106cf5855a44c1 (patch)
treec907a2683dfa4e8ad05be2ad2730cabe8a031743
parent82f5ff7938d002a3cd67d43af04f77ce3daa8b71 (diff)
downloadsqlalchemy-f390639bf1a7a5a2a47bcd6df7106cf5855a44c1.tar.gz
Fixed bug where :class:`.QueuePool` would lose the correct
checked out count if an existing pooled connection failed to reconnect after an invalidate or recycle event. Also in 0.8.3. [ticket:2772]
-rw-r--r--doc/build/changelog/changelog_08.rst11
-rw-r--r--doc/build/changelog/changelog_09.rst8
-rw-r--r--lib/sqlalchemy/pool.py7
-rw-r--r--test/engine/test_pool.py51
4 files changed, 75 insertions, 2 deletions
diff --git a/doc/build/changelog/changelog_08.rst b/doc/build/changelog/changelog_08.rst
index 1d52a994d..04c7f1f31 100644
--- a/doc/build/changelog/changelog_08.rst
+++ b/doc/build/changelog/changelog_08.rst
@@ -4,6 +4,17 @@
==============
.. changelog::
+ :version: 0.8.3
+
+ .. change::
+ :tags: bug, engine, pool
+ :tickets: 2772
+
+ Fixed bug where :class:`.QueuePool` would lose the correct
+ checked out count if an existing pooled connection failed to reconnect
+ after an invalidate or recycle event.
+
+.. changelog::
:version: 0.8.2
:released: July 3, 2013
diff --git a/doc/build/changelog/changelog_09.rst b/doc/build/changelog/changelog_09.rst
index 2b33a5cb1..c2abc2504 100644
--- a/doc/build/changelog/changelog_09.rst
+++ b/doc/build/changelog/changelog_09.rst
@@ -7,6 +7,14 @@
:version: 0.9.0
.. change::
+ :tags: bug, engine, pool
+ :tickets: 2772
+
+ Fixed bug where :class:`.QueuePool` would lose the correct
+ checked out count if an existing pooled connection failed to reconnect
+ after an invalidate or recycle event. Also in 0.8.3.
+
+ .. change::
:tags: bug, mysql
:tickets: 2768
diff --git a/lib/sqlalchemy/pool.py b/lib/sqlalchemy/pool.py
index 97411dd3a..498b001c1 100644
--- a/lib/sqlalchemy/pool.py
+++ b/lib/sqlalchemy/pool.py
@@ -328,7 +328,11 @@ class _ConnectionRecord(object):
@classmethod
def checkout(cls, pool):
rec = pool._do_get()
- dbapi_connection = rec.get_connection()
+ try:
+ dbapi_connection = rec.get_connection()
+ except:
+ rec.checkin()
+ raise
fairy = _ConnectionFairy(dbapi_connection, rec)
rec.fairy_ref = weakref.ref(
fairy,
@@ -565,6 +569,7 @@ class _ConnectionFairy(object):
_refs.remove(self._connection_record)
self._connection_record.fairy_ref = None
self._connection_record.connection = None
+ # TODO: should this be _return_conn?
self._pool._do_return_conn(self._connection_record)
self.info = self.info.copy()
self._connection_record = None
diff --git a/test/engine/test_pool.py b/test/engine/test_pool.py
index 8c4bcd8b5..3e0188bd7 100644
--- a/test/engine/test_pool.py
+++ b/test/engine/test_pool.py
@@ -14,11 +14,20 @@ def MockDBAPI():
def cursor():
while True:
yield Mock()
+
def connect():
while True:
yield Mock(cursor=Mock(side_effect=cursor()))
- return Mock(connect=Mock(side_effect=connect()))
+ def shutdown(value):
+ if value:
+ db.connect = Mock(side_effect=Exception("connect failed"))
+ else:
+ db.connect = Mock(side_effect=connect())
+
+ db = Mock(connect=Mock(side_effect=connect()),
+ shutdown=shutdown, _shutdown=False)
+ return db
class PoolTestBase(fixtures.TestBase):
def setup(self):
@@ -1073,6 +1082,46 @@ class QueuePoolTest(PoolTestBase):
c3 = p.connect()
assert id(c3.connection) != c_id
+ def _assert_cleanup_on_pooled_reconnect(self, dbapi, p):
+ # p is QueuePool with size=1, max_overflow=2,
+ # and one connection in the pool that will need to
+ # reconnect when next used (either due to recycle or invalidate)
+ eq_(p.checkedout(), 0)
+ eq_(p._overflow, 0)
+ dbapi.shutdown(True)
+ assert_raises(
+ Exception,
+ p.connect
+ )
+ eq_(p._overflow, 0)
+ eq_(p.checkedout(), 0) # and not 1
+
+ dbapi.shutdown(False)
+
+ c1 = p.connect()
+ assert p._pool.empty() # poolsize is one, so we're empty OK
+ c2 = p.connect()
+ eq_(p._overflow, 1) # and not 2
+
+ # this hangs if p._overflow is 2
+ c3 = p.connect()
+
+ def test_error_on_pooled_reconnect_cleanup_invalidate(self):
+ dbapi, p = self._queuepool_dbapi_fixture(pool_size=1, max_overflow=2)
+ c1 = p.connect()
+ c1.invalidate()
+ c1.close()
+ self._assert_cleanup_on_pooled_reconnect(dbapi, p)
+
+ def test_error_on_pooled_reconnect_cleanup_recycle(self):
+ dbapi, p = self._queuepool_dbapi_fixture(pool_size=1,
+ max_overflow=2, recycle=1)
+ c1 = p.connect()
+ c1.close()
+ time.sleep(1)
+ self._assert_cleanup_on_pooled_reconnect(dbapi, p)
+
+
def test_invalidate(self):
p = self._queuepool_fixture(pool_size=1, max_overflow=0)
c1 = p.connect()