@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,
_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
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):
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()