Skip to content

Commit

Permalink
Upgrade redis-py to 3.4.0 (#166)
Browse files Browse the repository at this point in the history
* Upgrade to redis-py 3.4.0

* Work around Redis __eq__() / __hash__() stupidity

* Prefer Knuth style; break before binary operators

https://www.python.org/dev/peps/pep-0008/#should-a-line-break-before-or-after-a-binary-operator

* Bump version number

* Fix typo in Redis equality test

I copypastaed these tests from my PR that got merged upstream:
https://github.com/andymccurdy/redis-py/pull/1240/files#diff-6fe40adc2096d3f6caba21161a37a7dcR26

I should probably fix it upstream too.
  • Loading branch information
brainix authored Feb 1, 2020
1 parent a490749 commit bd8da6a
Show file tree
Hide file tree
Showing 8 changed files with 49 additions and 47 deletions.
5 changes: 2 additions & 3 deletions .flake8
Original file line number Diff line number Diff line change
Expand Up @@ -20,14 +20,13 @@
# E711: comparison to None should be 'if cond is None:'
# E713: test for membership should be 'not in'
# F401: '.monkey' imported but unused
# W504: line break after binary operator
# W503: line break after binary operator

ignore = E226,E501
ignore = E226,E501,W503

per-file-ignores =
pottery/__init__.py:E402
pottery/base.py:F401
pottery/bloom.py:W504
pottery/exceptions.py:E302
pottery/monkey.py:E302,E305,E402
pottery/redlock.py:E127
Expand Down
2 changes: 1 addition & 1 deletion pottery/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@


__title__ = 'pottery'
__version__ = '0.66'
__version__ = '0.67'
__description__, __long_description__ = (
s.strip() for s in __doc__.split(sep='\n\n', maxsplit=1)
)
Expand Down
4 changes: 2 additions & 2 deletions pottery/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -116,8 +116,8 @@ def __eq__(self, other):
equals = True
elif (
isinstance(other, _Comparable)
and self.redis == other.redis # NoQA: W503
and self.key == other.key # NoQA: W503
and self.redis.connection_pool == other.redis.connection_pool
and self.key == other.key
):
equals = True
else:
Expand Down
12 changes: 6 additions & 6 deletions pottery/bloom.py
Original file line number Diff line number Diff line change
Expand Up @@ -113,9 +113,9 @@ def size(self):
return self._size
except AttributeError:
self._size = (
-self.num_values *
math.log(self.false_positives) /
math.log(2)**2
-self.num_values
* math.log(self.false_positives)
/ math.log(2)**2
)
self._size = math.ceil(self._size)
return self.size()
Expand Down Expand Up @@ -236,9 +236,9 @@ def __len__(self):
https://en.wikipedia.org/wiki/Bloom_filter#Approximating_the_number_of_items_in_a_Bloom_filter
'''
len_ = (
-self.size() /
self.num_hashes() *
math.log(1 - self._num_bits_set() / self.size())
-self.size()
/ self.num_hashes()
* math.log(1 - self._num_bits_set() / self.size())
)
return math.floor(len_)

Expand Down
38 changes: 5 additions & 33 deletions pottery/monkey.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,41 +34,13 @@ def _listdir(path=None, *, absolute=False):
_logger.info('Monkey patched os.listdir() to optionally return absolute paths')


# The Redis client doesn't have a sane equality test. So monkey patch equality
# comparisons on to the Redis client. We consider two Redis clients to be
# equal if they're connected to the same host, port, and database.
#
# I've submitted this change to redis-py:
# https://github.com/andymccurdy/redis-py/pull/1240
#
# If it gets merged upstream, then I'll be able to delete this monkey patch.

from redis import ConnectionPool
from redis import Redis

def __eq__(self, other):
try:
return self.connection_kwargs == other.connection_kwargs
except AttributeError: # pragma: no cover
return False

ConnectionPool.__eq__ = __eq__
# lolwut for some reason, Redis doesn't have a __hash__() method.

def __eq__(self, other):
'''True if two Redis clients are equal.
The Redis client doesn't have a sane equality test. So we monkey patch
this method on to the Redis client so that two client instances are equal
if they're connected to the same Redis host, port, and database.
'''
try:
return self.connection_pool == other.connection_pool
except AttributeError:
return False
from redis import Redis

Redis.__eq__ = __eq__
Redis.__hash__ = lambda self: hash(str(self))

_logger.info(
'Monkey patched ConnectionPool.__eq__() and Redis.__eq__() to compare '
'Redis clients by connection params'
'Monkey patched Redis.__hash__() in order to be able to put Redis clients '
'into sets'
)
5 changes: 4 additions & 1 deletion pottery/set.py
Original file line number Diff line number Diff line change
Expand Up @@ -81,7 +81,10 @@ def remove(self, value):
def isdisjoint(self, other):
'Return True if two sets have a null intersection. O(n)'
with self._watch(other):
if isinstance(other, self.__class__) and self.redis == other.redis:
if (
isinstance(other, self.__class__)
and self.redis.connection_pool == other.redis.connection_pool
):
self.redis.multi()
self.redis.sinter(self.key, other.key)
disjoint = not self.redis.execute()[0]
Expand Down
2 changes: 1 addition & 1 deletion requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ pycodestyle==2.5.0
pyflakes==2.1.1
Pygments==2.5.1
readme-renderer==24.0
redis==3.3.11
redis==3.4.0
requests==2.22.0
requests-toolbelt==0.9.1
six==1.13.0
Expand Down
28 changes: 28 additions & 0 deletions tests/test_redis.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,3 +25,31 @@ def test_redis_clients_equal_if_same_url(self):
assert not redis1 != redis2
assert redis1 != None
assert not redis1 == None

def test_client_equality(self):
r1 = Redis.from_url('redis://localhost:6379/9')
r2 = Redis.from_url('redis://localhost:6379/9')
assert r1 == r2
assert hash(r1) == hash(r2)

def test_clients_unequal_if_different_types(self):
r = Redis.from_url('redis://localhost:6379/9')
assert r != 0

def test_clients_unequal_if_different_hosts(self):
r1 = Redis.from_url('redis://localhost:6379/9')
r2 = Redis.from_url('redis://127.0.0.1:6379/9')
assert r1 != r2
assert hash(r1) != hash(r2)

def test_clients_unequal_if_different_ports(self):
r1 = Redis.from_url('redis://localhost:6379/9')
r2 = Redis.from_url('redis://localhost:6380/9')
assert r1 != r2
assert hash(r1) != hash(r2)

def test_clients_unequal_if_different_dbs(self):
r1 = Redis.from_url('redis://localhost:6379/9')
r2 = Redis.from_url('redis://localhost:6379/10')
assert r1 != r2
assert hash(r1) != hash(r2)

0 comments on commit bd8da6a

Please sign in to comment.