]> git.ipfire.org Git - thirdparty/Python/cpython.git/commitdiff
bpo-42470: Do not warn on sequences which are also sets in random.sample() (GH-23665)
authormasklinn <github.com@masklinn.net>
Sat, 19 Dec 2020 04:33:36 +0000 (05:33 +0100)
committerGitHub <noreply@github.com>
Sat, 19 Dec 2020 04:33:36 +0000 (20:33 -0800)
Lib/random.py
Lib/test/test_random.py
Misc/NEWS.d/next/Library/2020-12-06-12-00-00.bpo-42470.iqtC4L.rst [new file with mode: 0644]

index 139e8a40bb2724b4eea784dcac30310403138a94..66433baa8cc18f2781c082c15ac97184dd14693a 100644 (file)
@@ -424,13 +424,14 @@ class Random(_random.Random):
         # too many calls to _randbelow(), making them slower and
         # causing them to eat more entropy than necessary.
 
-        if isinstance(population, _Set):
-            _warn('Sampling from a set deprecated\n'
-                  'since Python 3.9 and will be removed in a subsequent version.',
-                  DeprecationWarning, 2)
-            population = tuple(population)
         if not isinstance(population, _Sequence):
-            raise TypeError("Population must be a sequence.  For dicts or sets, use sorted(d).")
+            if isinstance(population, _Set):
+                _warn('Sampling from a set deprecated\n'
+                      'since Python 3.9 and will be removed in a subsequent version.',
+                      DeprecationWarning, 2)
+                population = tuple(population)
+            else:
+                raise TypeError("Population must be a sequence.  For dicts or sets, use sorted(d).")
         n = len(population)
         if counts is not None:
             cum_counts = list(_accumulate(counts))
index 0c1fdeec9915ee8720fa902f22bfcd8b0b7a6b3e..327bfa3bbda1bd2460cf89e3b1bf301bc7888ca5 100644 (file)
@@ -11,7 +11,7 @@ from functools import partial
 from math import log, exp, pi, fsum, sin, factorial
 from test import support
 from fractions import Fraction
-from collections import Counter
+from collections import abc, Counter
 
 class TestBasicOps:
     # Superclass with tests common to all generators.
@@ -163,6 +163,22 @@ class TestBasicOps:
             population = {10, 20, 30, 40, 50, 60, 70}
             self.gen.sample(population, k=5)
 
+    def test_sample_on_seqsets(self):
+        class SeqSet(abc.Sequence, abc.Set):
+            def __init__(self, items):
+                self._items = items
+
+            def __len__(self):
+                return len(self._items)
+
+            def __getitem__(self, index):
+                return self._items[index]
+
+        population = SeqSet([2, 4, 1, 3])
+        with warnings.catch_warnings():
+            warnings.simplefilter("error", DeprecationWarning)
+            self.gen.sample(population, k=2)
+
     def test_sample_with_counts(self):
         sample = self.gen.sample
 
diff --git a/Misc/NEWS.d/next/Library/2020-12-06-12-00-00.bpo-42470.iqtC4L.rst b/Misc/NEWS.d/next/Library/2020-12-06-12-00-00.bpo-42470.iqtC4L.rst
new file mode 100644 (file)
index 0000000..cd2edb6
--- /dev/null
@@ -0,0 +1 @@
+:func:`random.sample` no longer warns on a sequence which is also a set.