[2.7] bpo-31478: Prevent unwanted behavior in _random.Random.seed() in case the arg has a bad __abs__() method (GH-3596) (#3845)
This commit is contained in:
parent
20cbc1d29f
commit
13da1a60f1
|
@ -307,6 +307,22 @@ class SystemRandom_TestBasicOps(TestBasicOps):
|
||||||
class MersenneTwister_TestBasicOps(TestBasicOps):
|
class MersenneTwister_TestBasicOps(TestBasicOps):
|
||||||
gen = random.Random()
|
gen = random.Random()
|
||||||
|
|
||||||
|
@test_support.cpython_only
|
||||||
|
def test_bug_31478(self):
|
||||||
|
# _random.Random.seed() should ignore the __abs__() method of a
|
||||||
|
# long/int subclass argument.
|
||||||
|
class BadInt(int):
|
||||||
|
def __abs__(self):
|
||||||
|
1/0
|
||||||
|
class BadLong(long):
|
||||||
|
def __abs__(self):
|
||||||
|
1/0
|
||||||
|
self.gen.seed(42)
|
||||||
|
expected_value = self.gen.random()
|
||||||
|
for seed_arg in [42L, BadInt(42), BadLong(42)]:
|
||||||
|
self.gen.seed(seed_arg)
|
||||||
|
self.assertEqual(self.gen.random(), expected_value)
|
||||||
|
|
||||||
def test_setstate_first_arg(self):
|
def test_setstate_first_arg(self):
|
||||||
self.assertRaises(ValueError, self.gen.setstate, (1, None, None))
|
self.assertRaises(ValueError, self.gen.setstate, (1, None, None))
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
Prevent unwanted behavior in `_random.Random.seed()` in case the argument
|
||||||
|
has a bad ``__abs__()`` method. Patch by Oren Milman.
|
|
@ -230,9 +230,15 @@ random_seed(RandomObject *self, PyObject *args)
|
||||||
}
|
}
|
||||||
/* If the arg is an int or long, use its absolute value; else use
|
/* If the arg is an int or long, use its absolute value; else use
|
||||||
* the absolute value of its hash code.
|
* the absolute value of its hash code.
|
||||||
|
* Calling int.__abs__() or long.__abs__() prevents calling arg.__abs__(),
|
||||||
|
* which might return an invalid value. See issue #31478.
|
||||||
*/
|
*/
|
||||||
if (PyInt_Check(arg) || PyLong_Check(arg))
|
if (PyInt_Check(arg)) {
|
||||||
n = PyNumber_Absolute(arg);
|
n = PyInt_Type.tp_as_number->nb_absolute(arg);
|
||||||
|
}
|
||||||
|
else if (PyLong_Check(arg)) {
|
||||||
|
n = PyLong_Type.tp_as_number->nb_absolute(arg);
|
||||||
|
}
|
||||||
else {
|
else {
|
||||||
long hash = PyObject_Hash(arg);
|
long hash = PyObject_Hash(arg);
|
||||||
if (hash == -1)
|
if (hash == -1)
|
||||||
|
|
Loading…
Reference in New Issue