mirror of
https://github.com/python/cpython.git
synced 2025-08-01 15:43:13 +00:00
[ 1497053 ] Let dicts propagate the exceptions in user __eq__().
[ 1456209 ] dictresize() vulnerability ( <- backport candidate ).
This commit is contained in:
parent
e08940ef6c
commit
35f6d36951
6 changed files with 183 additions and 101 deletions
|
@ -5,27 +5,16 @@ print '3. Operations'
|
|||
print 'XXX Mostly not yet implemented'
|
||||
|
||||
|
||||
print '3.1 Dictionary lookups succeed even if __cmp__() raises an exception'
|
||||
|
||||
# SourceForge bug #112558:
|
||||
# http://sourceforge.net/bugs/?func=detailbug&bug_id=112558&group_id=5470
|
||||
print '3.1 Dictionary lookups fail if __cmp__() raises an exception'
|
||||
|
||||
class BadDictKey:
|
||||
already_printed_raising_error = 0
|
||||
|
||||
def __hash__(self):
|
||||
return hash(self.__class__)
|
||||
|
||||
def __cmp__(self, other):
|
||||
if isinstance(other, self.__class__):
|
||||
if not BadDictKey.already_printed_raising_error:
|
||||
# How many times __cmp__ gets called depends on the hash
|
||||
# code and the internals of the dict implementation; we
|
||||
# know it will be called at least once, but that's it.
|
||||
# already_printed_raising_error makes sure the expected-
|
||||
# output file prints the msg at most once.
|
||||
BadDictKey.already_printed_raising_error = 1
|
||||
print "raising error"
|
||||
print "raising error"
|
||||
raise RuntimeError, "gotcha"
|
||||
return other
|
||||
|
||||
|
@ -33,8 +22,21 @@ d = {}
|
|||
x1 = BadDictKey()
|
||||
x2 = BadDictKey()
|
||||
d[x1] = 1
|
||||
d[x2] = 2
|
||||
print "No exception passed through."
|
||||
for stmt in ['d[x2] = 2',
|
||||
'z = d[x2]',
|
||||
'x2 in d',
|
||||
'd.has_key(x2)',
|
||||
'd.get(x2)',
|
||||
'd.setdefault(x2, 42)',
|
||||
'd.pop(x2)',
|
||||
'd.update({x2: 2})']:
|
||||
try:
|
||||
exec stmt
|
||||
except RuntimeError:
|
||||
print "%s: caught the RuntimeError outside" % (stmt,)
|
||||
else:
|
||||
print "%s: No exception passed through!" # old CPython behavior
|
||||
|
||||
|
||||
# Dict resizing bug, found by Jack Jansen in 2.2 CVS development.
|
||||
# This version got an assert failure in debug build, infinite loop in
|
||||
|
@ -50,3 +52,27 @@ for i in range(5):
|
|||
del d[i]
|
||||
for i in range(5, 9): # i==8 was the problem
|
||||
d[i] = i
|
||||
|
||||
|
||||
# Another dict resizing bug (SF bug #1456209).
|
||||
# This caused Segmentation faults or Illegal instructions.
|
||||
|
||||
class X(object):
|
||||
def __hash__(self):
|
||||
return 5
|
||||
def __eq__(self, other):
|
||||
if resizing:
|
||||
d.clear()
|
||||
return False
|
||||
d = {}
|
||||
resizing = False
|
||||
d[X()] = 1
|
||||
d[X()] = 2
|
||||
d[X()] = 3
|
||||
d[X()] = 4
|
||||
d[X()] = 5
|
||||
# now trigger a resize
|
||||
resizing = True
|
||||
d[9] = 6
|
||||
|
||||
print 'resize bugs not triggered.'
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue