2010-11-09 14:27:43 +00:00
|
|
|
import pickle
|
2012-11-27 15:10:13 +00:00
|
|
|
import socket
|
2016-07-07 16:33:23 +00:00
|
|
|
from copy import copy, deepcopy
|
2020-08-15 20:51:02 +00:00
|
|
|
from unittest.mock import Mock, patch
|
2016-08-22 18:17:59 +00:00
|
|
|
|
2021-07-20 13:07:49 +00:00
|
|
|
import pytest
|
|
|
|
|
2012-06-24 15:32:17 +00:00
|
|
|
from kombu import Connection, Consumer, Producer, parse_url
|
|
|
|
from kombu.connection import Resource
|
2016-07-26 21:11:37 +00:00
|
|
|
from kombu.exceptions import OperationalError
|
2016-04-01 22:04:01 +00:00
|
|
|
from kombu.utils.functional import lazy
|
2016-08-22 18:17:59 +00:00
|
|
|
from t.mocks import Transport
|
2012-11-27 15:10:13 +00:00
|
|
|
|
2010-06-28 22:54:13 +00:00
|
|
|
|
2016-08-22 18:17:59 +00:00
|
|
|
class test_connection_utils:
|
2011-11-02 05:17:42 +00:00
|
|
|
|
2014-04-23 22:00:03 +00:00
|
|
|
def setup(self):
|
2012-06-15 17:32:40 +00:00
|
|
|
self.url = 'amqp://user:pass@localhost:5672/my/vhost'
|
2014-05-20 18:05:01 +00:00
|
|
|
self.nopass = 'amqp://user:**@localhost:5672/my/vhost'
|
2011-11-02 05:17:42 +00:00
|
|
|
self.expected = {
|
2012-06-15 17:32:40 +00:00
|
|
|
'transport': 'amqp',
|
|
|
|
'userid': 'user',
|
|
|
|
'password': 'pass',
|
|
|
|
'hostname': 'localhost',
|
|
|
|
'port': 5672,
|
|
|
|
'virtual_host': 'my/vhost',
|
2011-11-02 05:17:42 +00:00
|
|
|
}
|
2019-09-03 11:57:27 +00:00
|
|
|
self.pg_url = 'sqla+postgresql://test:password@yms-pg/yms'
|
|
|
|
self.pg_nopass = 'sqla+postgresql://test:**@yms-pg/yms'
|
2011-11-02 05:17:42 +00:00
|
|
|
|
|
|
|
def test_parse_url(self):
|
|
|
|
result = parse_url(self.url)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert result == self.expected
|
2011-11-02 05:17:42 +00:00
|
|
|
|
|
|
|
def test_parse_generated_as_uri(self):
|
2012-06-24 15:32:17 +00:00
|
|
|
conn = Connection(self.url)
|
2011-11-02 05:17:42 +00:00
|
|
|
info = conn.info()
|
2013-01-17 13:16:32 +00:00
|
|
|
for k, v in self.expected.items():
|
2016-08-22 18:17:59 +00:00
|
|
|
assert info[k] == v
|
2011-11-02 05:17:42 +00:00
|
|
|
# by default almost the same- no password
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn.as_uri() == self.nopass
|
|
|
|
assert conn.as_uri(include_password=True) == self.url
|
2011-11-02 05:17:42 +00:00
|
|
|
|
2013-09-12 12:10:42 +00:00
|
|
|
def test_as_uri_when_prefix(self):
|
2020-08-16 19:54:10 +00:00
|
|
|
pytest.importorskip('redis')
|
2016-01-13 01:11:09 +00:00
|
|
|
conn = Connection('redis+socket:///var/spool/x/y/z/redis.sock')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn.as_uri() == 'redis+socket:///var/spool/x/y/z/redis.sock'
|
2013-09-12 12:10:42 +00:00
|
|
|
|
2012-01-14 00:02:59 +00:00
|
|
|
def test_as_uri_when_mongodb(self):
|
2020-08-16 19:54:10 +00:00
|
|
|
pytest.importorskip('pymongo')
|
2012-06-24 15:32:17 +00:00
|
|
|
x = Connection('mongodb://localhost')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert x.as_uri()
|
2012-01-14 00:02:59 +00:00
|
|
|
|
2011-11-02 05:17:42 +00:00
|
|
|
def test_bogus_scheme(self):
|
2016-08-22 18:17:59 +00:00
|
|
|
with pytest.raises(KeyError):
|
2012-06-24 15:32:17 +00:00
|
|
|
Connection('bogus://localhost:7421').transport
|
2011-11-02 05:17:42 +00:00
|
|
|
|
2012-02-10 14:52:17 +00:00
|
|
|
def assert_info(self, conn, **fields):
|
|
|
|
info = conn.info()
|
2020-07-23 13:33:40 +00:00
|
|
|
for field, expected in fields.items():
|
2016-08-22 18:17:59 +00:00
|
|
|
assert info[field] == expected
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('url,expected', [
|
|
|
|
('amqp://user:pass@host:10000/vhost',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'user', 'password': 'pass', 'hostname': 'host',
|
|
|
|
'port': 10000, 'virtual_host': 'vhost'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://user%61:%61pass@ho%61st:10000/v%2fhost',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'usera', 'password': 'apass', 'hostname': 'hoast',
|
|
|
|
'port': 10000, 'virtual_host': 'v/host'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'guest', 'password': 'guest', 'hostname': 'localhost',
|
|
|
|
'port': 5672, 'virtual_host': '/'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://:@/',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'guest', 'password': 'guest', 'hostname': 'localhost',
|
|
|
|
'port': 5672, 'virtual_host': '/'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://user@/',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'user', 'password': 'guest', 'hostname': 'localhost',
|
|
|
|
'port': 5672, 'virtual_host': '/'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://user:pass@/',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'user', 'password': 'pass', 'hostname': 'localhost',
|
|
|
|
'port': 5672, 'virtual_host': '/'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://host',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'guest', 'password': 'guest', 'hostname': 'host',
|
|
|
|
'port': 5672, 'virtual_host': '/'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://:10000',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'guest', 'password': 'guest', 'hostname': 'localhost',
|
|
|
|
'port': 10000, 'virtual_host': '/'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp:///vhost',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'guest', 'password': 'guest', 'hostname': 'localhost',
|
|
|
|
'port': 5672, 'virtual_host': 'vhost'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://host/',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'guest', 'password': 'guest', 'hostname': 'host',
|
|
|
|
'port': 5672, 'virtual_host': '/'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
('amqp://host/%2f',
|
2017-09-08 04:43:59 +00:00
|
|
|
{'userid': 'guest', 'password': 'guest', 'hostname': 'host',
|
|
|
|
'port': 5672, 'virtual_host': '/'}),
|
2016-08-22 18:17:59 +00:00
|
|
|
])
|
|
|
|
def test_rabbitmq_example_urls(self, url, expected):
|
2012-02-10 14:52:17 +00:00
|
|
|
# see Appendix A of http://www.rabbitmq.com/uri-spec.html
|
2016-08-22 18:17:59 +00:00
|
|
|
self.assert_info(Connection(url), **expected)
|
2012-02-10 14:52:17 +00:00
|
|
|
|
2022-02-02 13:23:14 +00:00
|
|
|
@pytest.mark.parametrize('url,expected', [
|
|
|
|
('sqs://user:pass@',
|
|
|
|
{'userid': None, 'password': None, 'hostname': None,
|
|
|
|
'port': None, 'virtual_host': '/'}),
|
|
|
|
('sqs://',
|
|
|
|
{'userid': None, 'password': None, 'hostname': None,
|
|
|
|
'port': None, 'virtual_host': '/'}),
|
|
|
|
])
|
|
|
|
def test_sqs_example_urls(self, url, expected, caplog):
|
|
|
|
pytest.importorskip('boto3')
|
|
|
|
self.assert_info(Connection('sqs://'), **expected)
|
|
|
|
assert not caplog.records
|
|
|
|
|
2020-08-15 22:28:25 +00:00
|
|
|
@pytest.mark.skip('TODO: urllib cannot parse ipv6 urls')
|
2012-02-10 14:52:17 +00:00
|
|
|
def test_url_IPV6(self):
|
|
|
|
self.assert_info(
|
2012-06-24 15:32:17 +00:00
|
|
|
Connection('amqp://[::1]'),
|
2013-01-17 13:50:01 +00:00
|
|
|
userid='guest', password='guest', hostname='[::1]',
|
|
|
|
port=5672, virtual_host='/',
|
|
|
|
)
|
2012-02-10 14:52:17 +00:00
|
|
|
|
2016-07-07 16:33:23 +00:00
|
|
|
def test_connection_copy(self):
|
|
|
|
conn = Connection(self.url, alternates=['amqp://host'])
|
|
|
|
clone = deepcopy(conn)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert clone.alt == ['amqp://host']
|
2016-07-07 16:33:23 +00:00
|
|
|
|
2019-09-03 11:57:27 +00:00
|
|
|
def test_parse_generated_as_uri_pg(self):
|
2020-08-16 19:54:10 +00:00
|
|
|
pytest.importorskip('sqlalchemy')
|
2019-09-03 11:57:27 +00:00
|
|
|
conn = Connection(self.pg_url)
|
|
|
|
assert conn.as_uri() == self.pg_nopass
|
|
|
|
assert conn.as_uri(include_password=True) == self.pg_url
|
|
|
|
|
2011-11-03 15:13:27 +00:00
|
|
|
|
2016-08-22 18:17:59 +00:00
|
|
|
class test_Connection:
|
2010-06-28 22:54:13 +00:00
|
|
|
|
2014-04-23 22:00:03 +00:00
|
|
|
def setup(self):
|
2012-06-24 15:32:17 +00:00
|
|
|
self.conn = Connection(port=5672, transport=Transport)
|
2010-11-09 14:27:43 +00:00
|
|
|
|
2010-06-28 22:54:13 +00:00
|
|
|
def test_establish_connection(self):
|
2010-11-09 14:27:43 +00:00
|
|
|
conn = self.conn
|
2020-06-01 21:23:05 +00:00
|
|
|
assert not conn.connected
|
2010-06-28 22:54:13 +00:00
|
|
|
conn.connect()
|
2020-06-01 21:23:05 +00:00
|
|
|
assert conn.connected
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn.connection.connected
|
|
|
|
assert conn.host == 'localhost:5672'
|
2010-06-28 22:54:13 +00:00
|
|
|
channel = conn.channel()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert channel.open
|
|
|
|
assert conn.drain_events() == 'event'
|
2010-06-28 22:54:13 +00:00
|
|
|
_connection = conn.connection
|
|
|
|
conn.close()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not _connection.connected
|
|
|
|
assert isinstance(conn.transport, Transport)
|
2010-06-29 15:31:56 +00:00
|
|
|
|
2020-06-02 07:45:59 +00:00
|
|
|
def test_reuse_connection(self):
|
|
|
|
conn = self.conn
|
|
|
|
assert conn.connect() is conn.connection is conn.connect()
|
|
|
|
|
2020-05-17 21:34:02 +00:00
|
|
|
def test_connect_no_transport_options(self):
|
|
|
|
conn = self.conn
|
2020-05-22 22:00:52 +00:00
|
|
|
conn._ensure_connection = Mock()
|
2020-05-17 21:34:02 +00:00
|
|
|
|
|
|
|
conn.connect()
|
2020-06-10 09:14:47 +00:00
|
|
|
# ensure_connection must be called to return immidiately
|
|
|
|
# and fail with transport exception
|
|
|
|
conn._ensure_connection.assert_called_with(
|
|
|
|
max_retries=1, reraise_as_library_errors=False
|
|
|
|
)
|
2020-05-17 21:34:02 +00:00
|
|
|
|
|
|
|
def test_connect_transport_options(self):
|
|
|
|
conn = self.conn
|
2020-06-10 09:14:47 +00:00
|
|
|
conn.transport_options = {
|
2020-05-17 21:34:02 +00:00
|
|
|
'max_retries': 1,
|
|
|
|
'interval_start': 2,
|
|
|
|
'interval_step': 3,
|
|
|
|
'interval_max': 4,
|
|
|
|
'ignore_this': True
|
|
|
|
}
|
2020-05-22 22:00:52 +00:00
|
|
|
conn._ensure_connection = Mock()
|
2020-05-17 21:34:02 +00:00
|
|
|
|
|
|
|
conn.connect()
|
2020-06-10 09:14:47 +00:00
|
|
|
# connect() is ignoring transport options
|
|
|
|
# ensure_connection must be called to return immidiately
|
|
|
|
# and fail with transport exception
|
|
|
|
conn._ensure_connection.assert_called_with(
|
|
|
|
max_retries=1, reraise_as_library_errors=False
|
|
|
|
)
|
2020-05-17 21:34:02 +00:00
|
|
|
|
2012-11-27 15:10:13 +00:00
|
|
|
def test_multiple_urls(self):
|
|
|
|
conn1 = Connection('amqp://foo;amqp://bar')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn1.hostname == 'foo'
|
|
|
|
assert conn1.alt == ['amqp://foo', 'amqp://bar']
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
conn2 = Connection(['amqp://foo', 'amqp://bar'])
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn2.hostname == 'foo'
|
|
|
|
assert conn2.alt == ['amqp://foo', 'amqp://bar']
|
2012-11-27 15:10:13 +00:00
|
|
|
|
2013-06-18 11:06:49 +00:00
|
|
|
def test_collect(self):
|
|
|
|
connection = Connection('memory://')
|
|
|
|
trans = connection._transport = Mock(name='transport')
|
|
|
|
_collect = trans._collect = Mock(name='transport._collect')
|
|
|
|
_close = connection._close = Mock(name='connection._close')
|
|
|
|
connection.declared_entities = Mock(name='decl_entities')
|
|
|
|
uconn = connection._connection = Mock(name='_connection')
|
|
|
|
connection.collect()
|
|
|
|
|
2016-04-09 04:10:31 +00:00
|
|
|
_close.assert_not_called()
|
2013-06-18 11:06:49 +00:00
|
|
|
_collect.assert_called_with(uconn)
|
|
|
|
connection.declared_entities.clear.assert_called_with()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert trans.client is None
|
|
|
|
assert connection._transport is None
|
|
|
|
assert connection._connection is None
|
2013-06-18 11:06:49 +00:00
|
|
|
|
2018-08-16 09:48:33 +00:00
|
|
|
def test_prefer_librabbitmq_over_amqp_when_available(self):
|
|
|
|
with patch('kombu.connection.supports_librabbitmq',
|
|
|
|
return_value=True):
|
|
|
|
connection = Connection('amqp://')
|
|
|
|
|
|
|
|
assert connection.transport_cls == 'librabbitmq'
|
|
|
|
|
|
|
|
def test_select_amqp_when_librabbitmq_is_not_available(self):
|
|
|
|
with patch('kombu.connection.supports_librabbitmq',
|
|
|
|
return_value=False):
|
|
|
|
connection = Connection('amqp://')
|
|
|
|
|
|
|
|
assert connection.transport_cls == 'amqp'
|
|
|
|
|
2013-06-18 11:06:49 +00:00
|
|
|
def test_collect_no_transport(self):
|
|
|
|
connection = Connection('memory://')
|
|
|
|
connection._transport = None
|
2015-10-23 20:48:55 +00:00
|
|
|
connection._do_close_self = Mock()
|
|
|
|
connection._do_close_transport = Mock()
|
2013-06-18 11:06:49 +00:00
|
|
|
connection.collect()
|
2015-10-23 20:48:55 +00:00
|
|
|
connection._do_close_self.assert_called_with()
|
|
|
|
connection._do_close_transport.assert_called_with()
|
2013-06-18 11:06:49 +00:00
|
|
|
|
2015-10-23 20:48:55 +00:00
|
|
|
connection._do_close_self.side_effect = socket.timeout()
|
2013-06-18 11:06:49 +00:00
|
|
|
connection.collect()
|
|
|
|
|
|
|
|
def test_collect_transport_gone(self):
|
|
|
|
connection = Connection('memory://')
|
|
|
|
uconn = connection._connection = Mock(name='conn._conn')
|
|
|
|
trans = connection._transport = Mock(name='transport')
|
|
|
|
collect = trans._collect = Mock(name='transport._collect')
|
|
|
|
|
|
|
|
def se(conn):
|
|
|
|
connection._transport = None
|
|
|
|
collect.side_effect = se
|
|
|
|
|
|
|
|
connection.collect()
|
|
|
|
collect.assert_called_with(uconn)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert connection._transport is None
|
2013-06-18 11:06:49 +00:00
|
|
|
|
2012-11-27 15:10:13 +00:00
|
|
|
def test_uri_passthrough(self):
|
2013-10-03 14:05:30 +00:00
|
|
|
transport = Mock(name='transport')
|
|
|
|
with patch('kombu.connection.get_transport_cls') as gtc:
|
|
|
|
gtc.return_value = transport
|
|
|
|
transport.can_parse_url = True
|
2012-11-27 15:10:13 +00:00
|
|
|
with patch('kombu.connection.parse_url') as parse_url:
|
|
|
|
c = Connection('foo+mysql://some_host')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert c.transport_cls == 'foo'
|
2016-04-09 04:10:31 +00:00
|
|
|
parse_url.assert_not_called()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert c.hostname == 'mysql://some_host'
|
|
|
|
assert c.as_uri().startswith('foo+')
|
2012-11-27 15:10:13 +00:00
|
|
|
with patch('kombu.connection.parse_url') as parse_url:
|
|
|
|
c = Connection('mysql://some_host', transport='foo')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert c.transport_cls == 'foo'
|
2016-04-09 04:10:31 +00:00
|
|
|
parse_url.assert_not_called()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert c.hostname == 'mysql://some_host'
|
2013-09-09 14:49:44 +00:00
|
|
|
c = Connection('pyamqp+sqlite://some_host')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert c.as_uri().startswith('pyamqp+')
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
def test_ensure_connection_on_error(self):
|
|
|
|
c = Connection('amqp://A;amqp://B')
|
|
|
|
with patch('kombu.connection.retry_over_time') as rot:
|
|
|
|
c.ensure_connection()
|
2016-04-09 04:10:31 +00:00
|
|
|
rot.assert_called()
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
args = rot.call_args[0]
|
|
|
|
cb = args[4]
|
|
|
|
intervals = iter([1, 2, 3, 4, 5])
|
2016-08-22 18:17:59 +00:00
|
|
|
assert cb(KeyError(), intervals, 0) == 0
|
|
|
|
assert cb(KeyError(), intervals, 1) == 1
|
|
|
|
assert cb(KeyError(), intervals, 2) == 0
|
|
|
|
assert cb(KeyError(), intervals, 3) == 2
|
|
|
|
assert cb(KeyError(), intervals, 4) == 0
|
|
|
|
assert cb(KeyError(), intervals, 5) == 3
|
|
|
|
assert cb(KeyError(), intervals, 6) == 0
|
|
|
|
assert cb(KeyError(), intervals, 7) == 4
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
errback = Mock()
|
|
|
|
c.ensure_connection(errback=errback)
|
|
|
|
args = rot.call_args[0]
|
|
|
|
cb = args[4]
|
2016-08-22 18:17:59 +00:00
|
|
|
assert cb(KeyError(), intervals, 0) == 0
|
2016-04-09 04:10:31 +00:00
|
|
|
errback.assert_called()
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
def test_supports_heartbeats(self):
|
|
|
|
c = Connection(transport=Mock)
|
2014-04-23 22:00:03 +00:00
|
|
|
c.transport.implements.heartbeats = False
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not c.supports_heartbeats
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
def test_is_evented(self):
|
|
|
|
c = Connection(transport=Mock)
|
2018-03-19 17:28:43 +00:00
|
|
|
c.transport.implements.asynchronous = False
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not c.is_evented
|
2012-11-27 15:10:13 +00:00
|
|
|
|
2013-09-25 15:48:00 +00:00
|
|
|
def test_register_with_event_loop(self):
|
2021-12-30 06:28:11 +00:00
|
|
|
transport = Mock(name='transport')
|
|
|
|
transport.connection_errors = []
|
|
|
|
c = Connection(transport=transport)
|
2013-09-25 15:48:00 +00:00
|
|
|
loop = Mock(name='loop')
|
|
|
|
c.register_with_event_loop(loop)
|
|
|
|
c.transport.register_with_event_loop.assert_called_with(
|
|
|
|
c.connection, loop,
|
|
|
|
)
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
def test_manager(self):
|
|
|
|
c = Connection(transport=Mock)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert c.manager is c.transport.manager
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
def test_copy(self):
|
|
|
|
c = Connection('amqp://example.com')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert copy(c).info() == c.info()
|
2012-11-27 15:10:13 +00:00
|
|
|
|
2013-06-18 11:06:49 +00:00
|
|
|
def test_copy_multiples(self):
|
|
|
|
c = Connection('amqp://A.example.com;amqp://B.example.com')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert c.alt
|
2013-06-18 11:06:49 +00:00
|
|
|
d = copy(c)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert d.alt == c.alt
|
2013-06-18 11:06:49 +00:00
|
|
|
|
2012-11-27 15:10:13 +00:00
|
|
|
def test_switch(self):
|
|
|
|
c = Connection('amqp://foo')
|
|
|
|
c._closed = True
|
|
|
|
c.switch('redis://example.com//3')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not c._closed
|
|
|
|
assert c.hostname == 'example.com'
|
|
|
|
assert c.transport_cls == 'redis'
|
|
|
|
assert c.virtual_host == '/3'
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
def test_maybe_switch_next(self):
|
|
|
|
c = Connection('amqp://foo;redis://example.com//3')
|
|
|
|
c.maybe_switch_next()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not c._closed
|
|
|
|
assert c.hostname == 'example.com'
|
|
|
|
assert c.transport_cls == 'redis'
|
|
|
|
assert c.virtual_host == '/3'
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
def test_maybe_switch_next_no_cycle(self):
|
|
|
|
c = Connection('amqp://foo')
|
|
|
|
c.maybe_switch_next()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not c._closed
|
|
|
|
assert c.hostname == 'foo'
|
|
|
|
assert c.transport_cls, ('librabbitmq', 'pyamqp' in 'amqp')
|
2012-11-27 15:10:13 +00:00
|
|
|
|
2019-06-24 05:39:18 +00:00
|
|
|
def test_switch_without_uri_identifier(self):
|
|
|
|
c = Connection('amqp://foo')
|
|
|
|
assert c.hostname == 'foo'
|
|
|
|
assert c.transport_cls, ('librabbitmq', 'pyamqp' in 'amqp')
|
|
|
|
c._closed = True
|
|
|
|
c.switch('example.com')
|
|
|
|
assert not c._closed
|
|
|
|
assert c.hostname == 'example.com'
|
|
|
|
assert c.transport_cls, ('librabbitmq', 'pyamqp' in 'amqp')
|
|
|
|
|
2012-11-27 15:10:13 +00:00
|
|
|
def test_heartbeat_check(self):
|
|
|
|
c = Connection(transport=Transport)
|
|
|
|
c.transport.heartbeat_check = Mock()
|
|
|
|
c.heartbeat_check(3)
|
|
|
|
c.transport.heartbeat_check.assert_called_with(c.connection, rate=3)
|
|
|
|
|
|
|
|
def test_completes_cycle_no_cycle(self):
|
|
|
|
c = Connection('amqp://')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert c.completes_cycle(0)
|
|
|
|
assert c.completes_cycle(1)
|
2012-11-27 15:10:13 +00:00
|
|
|
|
|
|
|
def test_completes_cycle(self):
|
|
|
|
c = Connection('amqp://a;amqp://b;amqp://c')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not c.completes_cycle(0)
|
|
|
|
assert not c.completes_cycle(1)
|
|
|
|
assert c.completes_cycle(2)
|
2012-11-27 15:10:13 +00:00
|
|
|
|
2016-04-01 22:04:01 +00:00
|
|
|
def test_get_heartbeat_interval(self):
|
|
|
|
self.conn.transport.get_heartbeat_interval = Mock(name='ghi')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert (self.conn.get_heartbeat_interval() is
|
|
|
|
self.conn.transport.get_heartbeat_interval.return_value)
|
2016-04-01 22:04:01 +00:00
|
|
|
self.conn.transport.get_heartbeat_interval.assert_called_with(
|
|
|
|
self.conn.connection)
|
|
|
|
|
|
|
|
def test_supports_exchange_type(self):
|
|
|
|
self.conn.transport.implements.exchange_type = {'topic'}
|
2016-08-22 18:17:59 +00:00
|
|
|
assert self.conn.supports_exchange_type('topic')
|
|
|
|
assert not self.conn.supports_exchange_type('fanout')
|
2016-04-01 22:04:01 +00:00
|
|
|
|
|
|
|
def test_qos_semantics_matches_spec(self):
|
2016-04-04 20:04:13 +00:00
|
|
|
qsms = self.conn.transport.qos_semantics_matches_spec = Mock()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert self.conn.qos_semantics_matches_spec is qsms.return_value
|
2016-04-01 22:04:01 +00:00
|
|
|
qsms.assert_called_with(self.conn.connection)
|
|
|
|
|
2010-06-29 15:31:56 +00:00
|
|
|
def test__enter____exit__(self):
|
2022-04-09 18:42:59 +00:00
|
|
|
with self.conn as context:
|
|
|
|
assert context is self.conn
|
|
|
|
self.conn.connect()
|
|
|
|
assert self.conn.connection.connected
|
|
|
|
assert self.conn.connection is None
|
|
|
|
self.conn.close() # again
|
2010-10-22 13:42:47 +00:00
|
|
|
|
2010-11-09 14:27:43 +00:00
|
|
|
def test_close_survives_connerror(self):
|
|
|
|
|
|
|
|
class _CustomError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
2015-07-13 19:22:16 +00:00
|
|
|
connection_errors = (_CustomError,)
|
2010-11-09 14:27:43 +00:00
|
|
|
|
|
|
|
def close_connection(self, connection):
|
2012-06-15 17:32:40 +00:00
|
|
|
raise _CustomError('foo')
|
2010-11-09 14:27:43 +00:00
|
|
|
|
2012-06-24 15:32:17 +00:00
|
|
|
conn = Connection(transport=MyTransport)
|
2010-11-09 14:27:43 +00:00
|
|
|
conn.connect()
|
|
|
|
conn.close()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn._closed
|
2010-11-09 14:27:43 +00:00
|
|
|
|
2012-01-11 14:20:03 +00:00
|
|
|
def test_close_when_default_channel(self):
|
|
|
|
conn = self.conn
|
|
|
|
conn._default_channel = Mock()
|
|
|
|
conn._close()
|
|
|
|
conn._default_channel.close.assert_called_with()
|
|
|
|
|
2020-06-05 07:20:56 +00:00
|
|
|
def test_auto_reconnect_default_channel(self):
|
|
|
|
# tests GH issue: #1208
|
|
|
|
# Tests that default_channel automatically reconnects when connection
|
|
|
|
# closed
|
|
|
|
c = Connection('memory://')
|
|
|
|
c._closed = True
|
|
|
|
with patch.object(
|
|
|
|
c, '_connection_factory', side_effect=c._connection_factory
|
|
|
|
) as cf_mock:
|
|
|
|
c.default_channel
|
|
|
|
cf_mock.assert_called_once_with()
|
|
|
|
|
2012-01-11 14:20:03 +00:00
|
|
|
def test_close_when_default_channel_close_raises(self):
|
|
|
|
|
2012-06-24 15:32:17 +00:00
|
|
|
class Conn(Connection):
|
2012-01-11 14:20:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def connection_errors(self):
|
2015-07-13 19:22:16 +00:00
|
|
|
return (KeyError,)
|
2012-01-11 14:20:03 +00:00
|
|
|
|
2012-06-15 17:32:40 +00:00
|
|
|
conn = Conn('memory://')
|
2012-01-11 14:20:03 +00:00
|
|
|
conn._default_channel = Mock()
|
|
|
|
conn._default_channel.close.side_effect = KeyError()
|
|
|
|
|
|
|
|
conn._close()
|
|
|
|
conn._default_channel.close.assert_called_with()
|
|
|
|
|
|
|
|
def test_revive_when_default_channel(self):
|
|
|
|
conn = self.conn
|
|
|
|
defchan = conn._default_channel = Mock()
|
|
|
|
conn.revive(Mock())
|
|
|
|
|
|
|
|
defchan.close.assert_called_with()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn._default_channel is None
|
2012-01-11 14:20:03 +00:00
|
|
|
|
2010-11-09 14:27:43 +00:00
|
|
|
def test_ensure_connection(self):
|
2016-08-22 18:17:59 +00:00
|
|
|
assert self.conn.ensure_connection()
|
2010-11-09 14:27:43 +00:00
|
|
|
|
2011-11-02 17:37:17 +00:00
|
|
|
def test_ensure_success(self):
|
|
|
|
def publish():
|
2012-06-15 17:32:40 +00:00
|
|
|
return 'foobar'
|
2011-11-02 17:37:17 +00:00
|
|
|
|
|
|
|
ensured = self.conn.ensure(None, publish)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert ensured() == 'foobar'
|
2011-11-02 17:37:17 +00:00
|
|
|
|
|
|
|
def test_ensure_failure(self):
|
|
|
|
class _CustomError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def publish():
|
2012-06-15 17:32:40 +00:00
|
|
|
raise _CustomError('bar')
|
2011-11-02 17:37:17 +00:00
|
|
|
|
|
|
|
ensured = self.conn.ensure(None, publish)
|
2016-08-22 18:17:59 +00:00
|
|
|
with pytest.raises(_CustomError):
|
2012-01-13 17:54:48 +00:00
|
|
|
ensured()
|
2011-11-02 17:37:17 +00:00
|
|
|
|
|
|
|
def test_ensure_connection_failure(self):
|
|
|
|
class _ConnectionError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def publish():
|
2012-06-15 17:32:40 +00:00
|
|
|
raise _ConnectionError('failed connection')
|
2011-11-02 17:37:17 +00:00
|
|
|
|
2021-12-30 06:28:11 +00:00
|
|
|
self.conn.get_transport_cls().connection_errors = (_ConnectionError,)
|
2011-11-02 17:37:17 +00:00
|
|
|
ensured = self.conn.ensure(self.conn, publish)
|
2016-08-22 18:17:59 +00:00
|
|
|
with pytest.raises(OperationalError):
|
2012-01-13 17:54:48 +00:00
|
|
|
ensured()
|
2011-11-02 17:37:17 +00:00
|
|
|
|
2012-01-14 00:02:59 +00:00
|
|
|
def test_autoretry(self):
|
|
|
|
myfun = Mock()
|
|
|
|
|
2021-12-30 06:28:11 +00:00
|
|
|
self.conn.get_transport_cls().connection_errors = (KeyError,)
|
2012-01-14 00:02:59 +00:00
|
|
|
|
|
|
|
def on_call(*args, **kwargs):
|
|
|
|
myfun.side_effect = None
|
2012-06-15 17:32:40 +00:00
|
|
|
raise KeyError('foo')
|
2012-01-14 00:02:59 +00:00
|
|
|
|
|
|
|
myfun.side_effect = on_call
|
|
|
|
insured = self.conn.autoretry(myfun)
|
|
|
|
insured()
|
|
|
|
|
2016-04-09 04:10:31 +00:00
|
|
|
myfun.assert_called()
|
2012-01-14 00:02:59 +00:00
|
|
|
|
2010-11-09 14:27:43 +00:00
|
|
|
def test_SimpleQueue(self):
|
|
|
|
conn = self.conn
|
2012-06-15 17:32:40 +00:00
|
|
|
q = conn.SimpleQueue('foo')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert q.channel is conn.default_channel
|
2010-11-09 14:27:43 +00:00
|
|
|
chan = conn.channel()
|
2012-06-15 17:32:40 +00:00
|
|
|
q2 = conn.SimpleQueue('foo', channel=chan)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert q2.channel is chan
|
2010-11-09 14:27:43 +00:00
|
|
|
|
|
|
|
def test_SimpleBuffer(self):
|
|
|
|
conn = self.conn
|
2012-06-15 17:32:40 +00:00
|
|
|
q = conn.SimpleBuffer('foo')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert q.channel is conn.default_channel
|
2010-11-09 14:27:43 +00:00
|
|
|
chan = conn.channel()
|
2012-06-15 17:32:40 +00:00
|
|
|
q2 = conn.SimpleBuffer('foo', channel=chan)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert q2.channel is chan
|
2010-11-09 14:27:43 +00:00
|
|
|
|
2019-11-22 18:19:03 +00:00
|
|
|
def test_SimpleQueue_with_parameters(self):
|
|
|
|
conn = self.conn
|
|
|
|
q = conn.SimpleQueue(
|
|
|
|
'foo', True, {'durable': True}, {'x-queue-mode': 'lazy'},
|
|
|
|
{'durable': True, 'type': 'fanout', 'delivery_mode': 'persistent'})
|
|
|
|
|
|
|
|
assert q.queue.exchange.type == 'fanout'
|
|
|
|
assert q.queue.exchange.durable
|
|
|
|
assert not q.queue.exchange.auto_delete
|
|
|
|
delivery_mode_code = q.queue.exchange.PERSISTENT_DELIVERY_MODE
|
|
|
|
assert q.queue.exchange.delivery_mode == delivery_mode_code
|
|
|
|
|
|
|
|
assert q.queue.queue_arguments['x-queue-mode'] == 'lazy'
|
|
|
|
|
|
|
|
assert q.queue.durable
|
|
|
|
assert not q.queue.auto_delete
|
|
|
|
|
|
|
|
def test_SimpleBuffer_with_parameters(self):
|
|
|
|
conn = self.conn
|
|
|
|
q = conn.SimpleBuffer(
|
|
|
|
'foo', True, {'durable': True}, {'x-queue-mode': 'lazy'},
|
|
|
|
{'durable': True, 'type': 'fanout', 'delivery_mode': 'persistent'})
|
|
|
|
assert q.queue.exchange.type == 'fanout'
|
|
|
|
assert q.queue.exchange.durable
|
|
|
|
assert q.queue.exchange.auto_delete
|
|
|
|
delivery_mode_code = q.queue.exchange.PERSISTENT_DELIVERY_MODE
|
|
|
|
assert q.queue.exchange.delivery_mode == delivery_mode_code
|
|
|
|
assert q.queue.queue_arguments['x-queue-mode'] == 'lazy'
|
|
|
|
assert q.queue.durable
|
|
|
|
assert q.queue.auto_delete
|
|
|
|
|
2012-01-11 14:20:03 +00:00
|
|
|
def test_Producer(self):
|
|
|
|
conn = self.conn
|
2016-08-22 18:17:59 +00:00
|
|
|
assert isinstance(conn.Producer(), Producer)
|
|
|
|
assert isinstance(conn.Producer(conn.default_channel), Producer)
|
2012-01-11 14:20:03 +00:00
|
|
|
|
|
|
|
def test_Consumer(self):
|
|
|
|
conn = self.conn
|
2016-08-22 18:17:59 +00:00
|
|
|
assert isinstance(conn.Consumer(queues=[]), Consumer)
|
|
|
|
assert isinstance(
|
|
|
|
conn.Consumer(queues=[], channel=conn.default_channel),
|
|
|
|
Consumer)
|
2012-01-11 14:20:03 +00:00
|
|
|
|
2010-11-09 14:27:43 +00:00
|
|
|
def test__repr__(self):
|
2016-08-22 18:17:59 +00:00
|
|
|
assert repr(self.conn)
|
2010-11-09 14:27:43 +00:00
|
|
|
|
|
|
|
def test__reduce__(self):
|
|
|
|
x = pickle.loads(pickle.dumps(self.conn))
|
2016-08-22 18:17:59 +00:00
|
|
|
assert x.info() == self.conn.info()
|
2010-11-09 14:27:43 +00:00
|
|
|
|
|
|
|
def test_channel_errors(self):
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
channel_errors = (KeyError, ValueError)
|
|
|
|
|
2012-06-24 15:32:17 +00:00
|
|
|
conn = Connection(transport=MyTransport)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn.channel_errors == (KeyError, ValueError)
|
2010-11-09 14:27:43 +00:00
|
|
|
|
2021-12-30 06:28:11 +00:00
|
|
|
def test_channel_errors__exception_no_cache(self):
|
|
|
|
"""Ensure the channel_errors can be retrieved without an initialized
|
|
|
|
transport.
|
|
|
|
"""
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
channel_errors = (KeyError,)
|
|
|
|
|
|
|
|
conn = Connection(transport=MyTransport)
|
|
|
|
MyTransport.__init__ = Mock(side_effect=Exception)
|
|
|
|
assert conn.channel_errors == (KeyError,)
|
|
|
|
|
2010-11-09 14:27:43 +00:00
|
|
|
def test_connection_errors(self):
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
connection_errors = (KeyError, ValueError)
|
|
|
|
|
2012-06-24 15:32:17 +00:00
|
|
|
conn = Connection(transport=MyTransport)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn.connection_errors == (KeyError, ValueError)
|
2010-11-09 14:27:43 +00:00
|
|
|
|
2021-12-30 06:28:11 +00:00
|
|
|
def test_connection_errors__exception_no_cache(self):
|
|
|
|
"""Ensure the connection_errors can be retrieved without an
|
|
|
|
initialized transport.
|
|
|
|
"""
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
connection_errors = (KeyError,)
|
|
|
|
|
|
|
|
conn = Connection(transport=MyTransport)
|
|
|
|
MyTransport.__init__ = Mock(side_effect=Exception)
|
|
|
|
assert conn.connection_errors == (KeyError,)
|
|
|
|
|
|
|
|
def test_recoverable_connection_errors(self):
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
recoverable_connection_errors = (KeyError, ValueError)
|
|
|
|
|
|
|
|
conn = Connection(transport=MyTransport)
|
|
|
|
assert conn.recoverable_connection_errors == (KeyError, ValueError)
|
|
|
|
|
|
|
|
def test_recoverable_connection_errors__fallback(self):
|
|
|
|
"""Ensure missing recoverable_connection_errors on the Transport does
|
|
|
|
not cause a fatal error.
|
|
|
|
"""
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
connection_errors = (KeyError,)
|
|
|
|
channel_errors = (ValueError,)
|
|
|
|
|
|
|
|
conn = Connection(transport=MyTransport)
|
|
|
|
assert conn.recoverable_connection_errors == (KeyError, ValueError)
|
|
|
|
|
|
|
|
def test_recoverable_connection_errors__exception_no_cache(self):
|
|
|
|
"""Ensure the recoverable_connection_errors can be retrieved without
|
|
|
|
an initialized transport.
|
|
|
|
"""
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
recoverable_connection_errors = (KeyError,)
|
|
|
|
|
|
|
|
conn = Connection(transport=MyTransport)
|
|
|
|
MyTransport.__init__ = Mock(side_effect=Exception)
|
|
|
|
assert conn.recoverable_connection_errors == (KeyError,)
|
|
|
|
|
|
|
|
def test_recoverable_channel_errors(self):
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
recoverable_channel_errors = (KeyError, ValueError)
|
|
|
|
|
|
|
|
conn = Connection(transport=MyTransport)
|
|
|
|
assert conn.recoverable_channel_errors == (KeyError, ValueError)
|
|
|
|
|
|
|
|
def test_recoverable_channel_errors__fallback(self):
|
|
|
|
"""Ensure missing recoverable_channel_errors on the Transport does not
|
|
|
|
cause a fatal error.
|
|
|
|
"""
|
|
|
|
|
|
|
|
class MyTransport(Transport):
|
|
|
|
pass
|
|
|
|
|
|
|
|
conn = Connection(transport=MyTransport)
|
|
|
|
assert conn.recoverable_channel_errors == ()
|
|
|
|
|
|
|
|
def test_recoverable_channel_errors__exception_no_cache(self):
|
|
|
|
"""Ensure the recoverable_channel_errors can be retrieved without an
|
|
|
|
initialized transport.
|
|
|
|
"""
|
|
|
|
class MyTransport(Transport):
|
|
|
|
recoverable_channel_errors = (KeyError,)
|
|
|
|
|
|
|
|
conn = Connection(transport=MyTransport)
|
|
|
|
MyTransport.__init__ = Mock(side_effect=Exception)
|
|
|
|
assert conn.recoverable_channel_errors == (KeyError,)
|
|
|
|
|
2019-09-10 16:46:13 +00:00
|
|
|
def test_multiple_urls_hostname(self):
|
|
|
|
conn = Connection(['example.com;amqp://example.com'])
|
|
|
|
assert conn.as_uri() == 'amqp://guest:**@example.com:5672//'
|
|
|
|
conn = Connection(['example.com', 'amqp://example.com'])
|
|
|
|
assert conn.as_uri() == 'amqp://guest:**@example.com:5672//'
|
|
|
|
conn = Connection('example.com;example.com;')
|
|
|
|
assert conn.as_uri() == 'amqp://guest:**@example.com:5672//'
|
|
|
|
|
2021-12-24 09:21:44 +00:00
|
|
|
def test_connection_respect_its_timeout(self):
|
|
|
|
invalid_port = 1222
|
|
|
|
with Connection(
|
|
|
|
f'amqp://guest:guest@localhost:{invalid_port}//',
|
|
|
|
transport_options={'max_retries': 2},
|
|
|
|
connect_timeout=1
|
|
|
|
) as conn:
|
|
|
|
with pytest.raises(OperationalError):
|
|
|
|
conn.default_channel
|
|
|
|
|
2010-10-22 13:42:47 +00:00
|
|
|
|
2016-08-22 18:17:59 +00:00
|
|
|
class test_Connection_with_transport_options:
|
2011-03-13 17:02:32 +00:00
|
|
|
|
2012-06-15 17:32:40 +00:00
|
|
|
transport_options = {'pool_recycler': 3600, 'echo': True}
|
2011-03-13 17:02:32 +00:00
|
|
|
|
2014-04-23 22:00:03 +00:00
|
|
|
def setup(self):
|
2012-06-24 15:32:17 +00:00
|
|
|
self.conn = Connection(port=5672, transport=Transport,
|
|
|
|
transport_options=self.transport_options)
|
2011-03-13 17:02:32 +00:00
|
|
|
|
|
|
|
def test_establish_connection(self):
|
|
|
|
conn = self.conn
|
2016-08-22 18:17:59 +00:00
|
|
|
assert conn.transport_options == self.transport_options
|
2011-03-13 17:02:32 +00:00
|
|
|
|
|
|
|
|
2012-01-14 00:02:59 +00:00
|
|
|
class xResource(Resource):
|
|
|
|
|
|
|
|
def setup(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2016-08-22 18:17:59 +00:00
|
|
|
class ResourceCase:
|
2010-10-22 13:42:47 +00:00
|
|
|
|
2015-12-09 01:43:04 +00:00
|
|
|
def create_resource(self, limit):
|
2012-06-15 17:32:40 +00:00
|
|
|
raise NotImplementedError('subclass responsibility')
|
2010-10-22 13:42:47 +00:00
|
|
|
|
2016-08-22 18:17:59 +00:00
|
|
|
def assert_state(self, P, avail, dirty):
|
|
|
|
assert P._resource.qsize() == avail
|
|
|
|
assert len(P._dirty) == dirty
|
2010-10-22 13:42:47 +00:00
|
|
|
|
2010-11-09 14:27:43 +00:00
|
|
|
def test_setup(self):
|
2016-08-22 18:17:59 +00:00
|
|
|
with pytest.raises(NotImplementedError):
|
|
|
|
Resource()
|
2010-11-09 14:27:43 +00:00
|
|
|
|
2010-10-22 13:42:47 +00:00
|
|
|
def test_acquire__release(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2016-08-22 18:17:59 +00:00
|
|
|
self.assert_state(P, 10, 0)
|
2013-02-13 11:15:47 +00:00
|
|
|
chans = [P.acquire() for _ in range(10)]
|
2016-08-22 18:17:59 +00:00
|
|
|
self.assert_state(P, 0, 10)
|
|
|
|
with pytest.raises(P.LimitExceeded):
|
2012-01-13 17:54:48 +00:00
|
|
|
P.acquire()
|
2010-10-22 13:42:47 +00:00
|
|
|
chans.pop().release()
|
2016-08-22 18:17:59 +00:00
|
|
|
self.assert_state(P, 1, 9)
|
2010-10-22 13:42:47 +00:00
|
|
|
[chan.release() for chan in chans]
|
2016-08-22 18:17:59 +00:00
|
|
|
self.assert_state(P, 10, 0)
|
2010-10-22 13:42:47 +00:00
|
|
|
|
2012-11-27 15:10:13 +00:00
|
|
|
def test_acquire_prepare_raises(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2012-11-27 15:10:13 +00:00
|
|
|
|
2016-08-22 18:17:59 +00:00
|
|
|
assert len(P._resource.queue) == 10
|
2012-11-27 15:10:13 +00:00
|
|
|
P.prepare = Mock()
|
|
|
|
P.prepare.side_effect = IOError()
|
2016-08-22 18:17:59 +00:00
|
|
|
with pytest.raises(IOError):
|
2012-11-27 15:10:13 +00:00
|
|
|
P.acquire(block=True)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert len(P._resource.queue) == 10
|
2012-11-27 15:10:13 +00:00
|
|
|
|
2012-01-14 00:02:59 +00:00
|
|
|
def test_acquire_no_limit(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(None)
|
2012-01-14 00:02:59 +00:00
|
|
|
P.acquire().release()
|
|
|
|
|
|
|
|
def test_replace_when_limit(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2012-01-14 00:02:59 +00:00
|
|
|
r = P.acquire()
|
|
|
|
P._dirty = Mock()
|
|
|
|
P.close_resource = Mock()
|
|
|
|
|
|
|
|
P.replace(r)
|
|
|
|
P._dirty.discard.assert_called_with(r)
|
|
|
|
P.close_resource.assert_called_with(r)
|
|
|
|
|
|
|
|
def test_replace_no_limit(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(None)
|
2012-01-14 00:02:59 +00:00
|
|
|
r = P.acquire()
|
|
|
|
P._dirty = Mock()
|
|
|
|
P.close_resource = Mock()
|
|
|
|
|
|
|
|
P.replace(r)
|
2016-04-09 04:10:31 +00:00
|
|
|
P._dirty.discard.assert_not_called()
|
2012-01-14 00:02:59 +00:00
|
|
|
P.close_resource.assert_called_with(r)
|
|
|
|
|
|
|
|
def test_interface_prepare(self):
|
|
|
|
x = xResource()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert x.prepare(10) == 10
|
2012-01-14 00:02:59 +00:00
|
|
|
|
|
|
|
def test_force_close_all_handles_AttributeError(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2013-06-18 11:29:00 +00:00
|
|
|
cr = P.collect_resource = Mock()
|
2012-06-15 17:32:40 +00:00
|
|
|
cr.side_effect = AttributeError('x')
|
2012-01-14 00:02:59 +00:00
|
|
|
|
|
|
|
P.acquire()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert P._dirty
|
2012-01-14 00:02:59 +00:00
|
|
|
|
|
|
|
P.force_close_all()
|
|
|
|
|
|
|
|
def test_force_close_all_no_mutex(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2012-01-14 00:02:59 +00:00
|
|
|
P.close_resource = Mock()
|
|
|
|
|
|
|
|
m = P._resource = Mock()
|
|
|
|
m.mutex = None
|
|
|
|
m.queue.pop.side_effect = IndexError
|
|
|
|
|
|
|
|
P.force_close_all()
|
|
|
|
|
|
|
|
def test_add_when_empty(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(None)
|
|
|
|
P._resource.queue.clear()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not P._resource.queue
|
2012-01-14 00:02:59 +00:00
|
|
|
P._add_when_empty()
|
2016-08-22 18:17:59 +00:00
|
|
|
assert P._resource.queue
|
2012-01-14 00:02:59 +00:00
|
|
|
|
2010-10-22 13:42:47 +00:00
|
|
|
|
|
|
|
class test_ConnectionPool(ResourceCase):
|
|
|
|
|
2015-12-09 01:43:04 +00:00
|
|
|
def create_resource(self, limit):
|
|
|
|
return Connection(port=5672, transport=Transport).Pool(limit)
|
2010-10-22 13:42:47 +00:00
|
|
|
|
2016-04-01 22:04:01 +00:00
|
|
|
def test_collect_resource__does_not_collect_lazy_resource(self):
|
|
|
|
P = self.create_resource(10)
|
|
|
|
res = lazy(object())
|
|
|
|
res.collect = Mock(name='collect')
|
|
|
|
P.collect_resource(res)
|
2016-04-09 04:10:31 +00:00
|
|
|
res.collect.assert_not_called()
|
2016-04-01 22:04:01 +00:00
|
|
|
|
|
|
|
def test_collect_resource(self):
|
|
|
|
res = Mock(name='res')
|
|
|
|
P = self.create_resource(10)
|
|
|
|
P.collect_resource(res, socket_timeout=10.3)
|
|
|
|
res.collect.assert_called_with(10.3)
|
|
|
|
|
2010-10-22 13:42:47 +00:00
|
|
|
def test_setup(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2010-10-22 13:42:47 +00:00
|
|
|
q = P._resource.queue
|
2016-08-22 18:17:59 +00:00
|
|
|
assert q[0]()._connection is None
|
|
|
|
assert q[1]()._connection is None
|
|
|
|
assert q[2]()._connection is None
|
2010-10-22 13:42:47 +00:00
|
|
|
|
2013-06-18 11:29:00 +00:00
|
|
|
def test_acquire_raises_evaluated(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(1)
|
2013-06-18 11:29:00 +00:00
|
|
|
# evaluate the connection first
|
|
|
|
r = P.acquire()
|
|
|
|
r.release()
|
|
|
|
P.prepare = Mock()
|
|
|
|
P.prepare.side_effect = MemoryError()
|
|
|
|
P.release = Mock()
|
2016-08-22 18:17:59 +00:00
|
|
|
with pytest.raises(MemoryError):
|
2013-06-18 12:09:02 +00:00
|
|
|
with P.acquire():
|
2013-06-18 11:29:00 +00:00
|
|
|
assert False
|
|
|
|
P.release.assert_called_with(r)
|
|
|
|
|
2012-11-27 15:10:13 +00:00
|
|
|
def test_release_no__debug(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2012-11-27 15:10:13 +00:00
|
|
|
R = Mock()
|
|
|
|
R._debug.side_effect = AttributeError()
|
|
|
|
P.release_resource(R)
|
|
|
|
|
2012-01-11 14:20:03 +00:00
|
|
|
def test_setup_no_limit(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(None)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not P._resource.queue
|
|
|
|
assert P.limit is None
|
2012-01-11 14:20:03 +00:00
|
|
|
|
|
|
|
def test_prepare_not_callable(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(None)
|
2012-06-24 15:32:17 +00:00
|
|
|
conn = Connection('memory://')
|
2016-08-22 18:17:59 +00:00
|
|
|
assert P.prepare(conn) is conn
|
2012-01-11 14:20:03 +00:00
|
|
|
|
|
|
|
def test_acquire_channel(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2012-01-11 14:20:03 +00:00
|
|
|
with P.acquire_channel() as (conn, channel):
|
2016-08-22 18:17:59 +00:00
|
|
|
assert channel is conn.default_channel
|
2012-01-11 14:20:03 +00:00
|
|
|
|
2010-10-22 13:42:47 +00:00
|
|
|
|
|
|
|
class test_ChannelPool(ResourceCase):
|
|
|
|
|
2015-12-09 01:43:04 +00:00
|
|
|
def create_resource(self, limit):
|
|
|
|
return Connection(port=5672, transport=Transport).ChannelPool(limit)
|
2010-10-22 13:42:47 +00:00
|
|
|
|
|
|
|
def test_setup(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2010-10-22 13:42:47 +00:00
|
|
|
q = P._resource.queue
|
2016-08-22 18:17:59 +00:00
|
|
|
with pytest.raises(AttributeError):
|
2015-12-09 01:43:04 +00:00
|
|
|
q[0].basic_consume
|
2012-01-11 14:20:03 +00:00
|
|
|
|
|
|
|
def test_setup_no_limit(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(None)
|
2016-08-22 18:17:59 +00:00
|
|
|
assert not P._resource.queue
|
|
|
|
assert P.limit is None
|
2012-01-11 14:20:03 +00:00
|
|
|
|
|
|
|
def test_prepare_not_callable(self):
|
2015-12-09 01:43:04 +00:00
|
|
|
P = self.create_resource(10)
|
2012-06-24 15:32:17 +00:00
|
|
|
conn = Connection('memory://')
|
2012-01-11 14:20:03 +00:00
|
|
|
chan = conn.default_channel
|
2016-08-22 18:17:59 +00:00
|
|
|
assert P.prepare(chan) is chan
|