Skip to content

Commit 5330060

Browse files
committed
Merge remote-tracking branch 'upstream/main' into codegen-reduce-repetition
2 parents 9b5dc34 + c5d0517 commit 5330060

34 files changed

+673
-181
lines changed

Doc/library/pathlib.rst

+30-3
Original file line numberDiff line numberDiff line change
@@ -133,11 +133,13 @@ we also call *flavours*:
133133
PureWindowsPath('c:/Program Files')
134134

135135
Spurious slashes and single dots are collapsed, but double dots (``'..'``)
136-
are not, since this would change the meaning of a path in the face of
137-
symbolic links::
136+
and leading double slashes (``'//'``) are not, since this would change the
137+
meaning of a path for various reasons (e.g. symbolic links, UNC paths)::
138138

139139
>>> PurePath('foo//bar')
140140
PurePosixPath('foo/bar')
141+
>>> PurePath('//foo/bar')
142+
PurePosixPath('//foo/bar')
141143
>>> PurePath('foo/./bar')
142144
PurePosixPath('foo/bar')
143145
>>> PurePath('foo/../bar')
@@ -166,13 +168,17 @@ we also call *flavours*:
166168
.. class:: PureWindowsPath(*pathsegments)
167169

168170
A subclass of :class:`PurePath`, this path flavour represents Windows
169-
filesystem paths::
171+
filesystem paths, including `UNC paths`_::
170172

171173
>>> PureWindowsPath('c:/Program Files/')
172174
PureWindowsPath('c:/Program Files')
175+
>>> PureWindowsPath('//server/share/file')
176+
PureWindowsPath('//server/share/file')
173177

174178
*pathsegments* is specified similarly to :class:`PurePath`.
175179

180+
.. _unc paths: https://en.wikipedia.org/wiki/Path_(computing)#UNC
181+
176182
Regardless of the system you're running on, you can instantiate all of
177183
these classes, since they don't provide any operation that does system calls.
178184

@@ -309,6 +315,27 @@ Pure paths provide the following methods and properties:
309315
>>> PureWindowsPath('//host/share').root
310316
'\\'
311317

318+
If the path starts with more than two successive slashes,
319+
:class:`~pathlib.PurePosixPath` collapses them::
320+
321+
>>> PurePosixPath('//etc').root
322+
'//'
323+
>>> PurePosixPath('///etc').root
324+
'/'
325+
>>> PurePosixPath('////etc').root
326+
'/'
327+
328+
.. note::
329+
330+
This behavior conforms to *The Open Group Base Specifications Issue 6*,
331+
paragraph `4.11 *Pathname Resolution* <xbd_path_resolution>`_:
332+
333+
*"A pathname that begins with two successive slashes may be interpreted in
334+
an implementation-defined manner, although more than two leading slashes
335+
shall be treated as a single slash."*
336+
337+
.. xbd_path_resolution: https://pubs.opengroup.org/onlinepubs/009695399/basedefs/xbd_chap04.html#tag_04_11
338+
312339
.. data:: PurePath.anchor
313340

314341
The concatenation of the drive and root::

Doc/library/symtable.rst

+2-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,8 @@ Examining Symbol Tables
6969

7070
.. method:: get_identifiers()
7171

72-
Return a list of names of symbols in this table.
72+
Return a view object containing the names of symbols in the table.
73+
See the :ref:`documentation of view objects <dict-views>`.
7374

7475
.. method:: lookup(name)
7576

Doc/tools/susp-ignored.csv

+2
Original file line numberDiff line numberDiff line change
@@ -172,6 +172,8 @@ library/itertools,,:step,elements from seq[start:stop:step]
172172
library/itertools,,::,kernel = tuple(kernel)[::-1]
173173
library/itertools,,:stop,elements from seq[start:stop:step]
174174
library/logging.handlers,,:port,host:port
175+
library/logging,,:root,WARNING:root:Watch out!
176+
library/logging,,:Watch,WARNING:root:Watch out!
175177
library/mmap,,:i2,obj[i1:i2]
176178
library/multiprocessing,,`,# Add more tasks using `put()`
177179
library/multiprocessing,,:queue,">>> QueueManager.register('get_queue', callable=lambda:queue)"

Doc/whatsnew/3.11.rst

+3
Original file line numberDiff line numberDiff line change
@@ -373,6 +373,9 @@ Other Language Changes
373373
the current directory, the script's directory or an empty string.
374374
(Contributed by Victor Stinner in :gh:`57684`.)
375375

376+
* A ``"z"`` option was added to the format specification mini-language that
377+
coerces negative zero to zero after rounding to the format precision. See
378+
:pep:`682` for more details. (Contributed by John Belmonte in :gh:`90153`.)
376379

377380
Other CPython Implementation Changes
378381
====================================

Include/internal/pycore_global_strings.h

+3
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,7 @@ struct _Py_global_strings {
9090
STRUCT_FOR_ID(__delete__)
9191
STRUCT_FOR_ID(__delitem__)
9292
STRUCT_FOR_ID(__dict__)
93+
STRUCT_FOR_ID(__dictoffset__)
9394
STRUCT_FOR_ID(__dir__)
9495
STRUCT_FOR_ID(__divmod__)
9596
STRUCT_FOR_ID(__doc__)
@@ -202,9 +203,11 @@ struct _Py_global_strings {
202203
STRUCT_FOR_ID(__truediv__)
203204
STRUCT_FOR_ID(__trunc__)
204205
STRUCT_FOR_ID(__typing_is_unpacked_typevartuple__)
206+
STRUCT_FOR_ID(__typing_prepare_subst__)
205207
STRUCT_FOR_ID(__typing_subst__)
206208
STRUCT_FOR_ID(__typing_unpacked_tuple_args__)
207209
STRUCT_FOR_ID(__warningregistry__)
210+
STRUCT_FOR_ID(__weaklistoffset__)
208211
STRUCT_FOR_ID(__weakref__)
209212
STRUCT_FOR_ID(__xor__)
210213
STRUCT_FOR_ID(_abc_impl)

Include/internal/pycore_runtime_init.h

+3
Original file line numberDiff line numberDiff line change
@@ -712,6 +712,7 @@ extern "C" {
712712
INIT_ID(__delete__), \
713713
INIT_ID(__delitem__), \
714714
INIT_ID(__dict__), \
715+
INIT_ID(__dictoffset__), \
715716
INIT_ID(__dir__), \
716717
INIT_ID(__divmod__), \
717718
INIT_ID(__doc__), \
@@ -824,9 +825,11 @@ extern "C" {
824825
INIT_ID(__truediv__), \
825826
INIT_ID(__trunc__), \
826827
INIT_ID(__typing_is_unpacked_typevartuple__), \
828+
INIT_ID(__typing_prepare_subst__), \
827829
INIT_ID(__typing_subst__), \
828830
INIT_ID(__typing_unpacked_tuple_args__), \
829831
INIT_ID(__warningregistry__), \
832+
INIT_ID(__weaklistoffset__), \
830833
INIT_ID(__weakref__), \
831834
INIT_ID(__xor__), \
832835
INIT_ID(_abc_impl), \

Lib/importlib/_bootstrap_external.py

+9-4
Original file line numberDiff line numberDiff line change
@@ -1399,7 +1399,9 @@ def invalidate_caches():
13991399
"""Call the invalidate_caches() method on all path entry finders
14001400
stored in sys.path_importer_caches (where implemented)."""
14011401
for name, finder in list(sys.path_importer_cache.items()):
1402-
if finder is None:
1402+
# Drop entry if finder name is a relative path. The current
1403+
# working directory may have changed.
1404+
if finder is None or not _path_isabs(name):
14031405
del sys.path_importer_cache[name]
14041406
elif hasattr(finder, 'invalidate_caches'):
14051407
finder.invalidate_caches()
@@ -1567,9 +1569,12 @@ def __init__(self, path, *loader_details):
15671569
loaders.extend((suffix, loader) for suffix in suffixes)
15681570
self._loaders = loaders
15691571
# Base (directory) path
1570-
self.path = path or '.'
1571-
if not _path_isabs(self.path):
1572-
self.path = _path_join(_os.getcwd(), self.path)
1572+
if not path or path == '.':
1573+
self.path = _os.getcwd()
1574+
elif not _path_isabs(path):
1575+
self.path = _path_join(_os.getcwd(), path)
1576+
else:
1577+
self.path = path
15731578
self._path_mtime = -1
15741579
self._path_cache = set()
15751580
self._relaxed_path_cache = set()

Lib/symtable.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -111,7 +111,7 @@ def has_children(self):
111111
return bool(self._table.children)
112112

113113
def get_identifiers(self):
114-
"""Return a list of names of symbols in the table.
114+
"""Return a view object containing the names of symbols in the table.
115115
"""
116116
return self._table.symbols.keys()
117117

Lib/test/pickletester.py

+61
Original file line numberDiff line numberDiff line change
@@ -3035,6 +3035,67 @@ def check_array(arr):
30353035
# 2-D, non-contiguous
30363036
check_array(arr[::2])
30373037

3038+
def test_evil_class_mutating_dict(self):
3039+
# https://github.com/python/cpython/issues/92930
3040+
from random import getrandbits
3041+
3042+
global Bad
3043+
class Bad:
3044+
def __eq__(self, other):
3045+
return ENABLED
3046+
def __hash__(self):
3047+
return 42
3048+
def __reduce__(self):
3049+
if getrandbits(6) == 0:
3050+
collection.clear()
3051+
return (Bad, ())
3052+
3053+
for proto in protocols:
3054+
for _ in range(20):
3055+
ENABLED = False
3056+
collection = {Bad(): Bad() for _ in range(20)}
3057+
for bad in collection:
3058+
bad.bad = bad
3059+
bad.collection = collection
3060+
ENABLED = True
3061+
try:
3062+
data = self.dumps(collection, proto)
3063+
self.loads(data)
3064+
except RuntimeError as e:
3065+
expected = "changed size during iteration"
3066+
self.assertIn(expected, str(e))
3067+
3068+
def test_evil_pickler_mutating_collection(self):
3069+
# https://github.com/python/cpython/issues/92930
3070+
if not hasattr(self, "pickler"):
3071+
raise self.skipTest(f"{type(self)} has no associated pickler type")
3072+
3073+
global Clearer
3074+
class Clearer:
3075+
pass
3076+
3077+
def check(collection):
3078+
class EvilPickler(self.pickler):
3079+
def persistent_id(self, obj):
3080+
if isinstance(obj, Clearer):
3081+
collection.clear()
3082+
return None
3083+
pickler = EvilPickler(io.BytesIO(), proto)
3084+
try:
3085+
pickler.dump(collection)
3086+
except RuntimeError as e:
3087+
expected = "changed size during iteration"
3088+
self.assertIn(expected, str(e))
3089+
3090+
for proto in protocols:
3091+
check([Clearer()])
3092+
check([Clearer(), Clearer()])
3093+
check({Clearer()})
3094+
check({Clearer(), Clearer()})
3095+
check({Clearer(): 1})
3096+
check({Clearer(): 1, Clearer(): 2})
3097+
check({1: Clearer(), 2: Clearer()})
3098+
30383099

30393100
class BigmemPickleTests:
30403101

Lib/test/test_import/__init__.py

+7-7
Original file line numberDiff line numberDiff line change
@@ -927,15 +927,15 @@ def test_missing_source_legacy(self):
927927
m = __import__(TESTFN)
928928
try:
929929
self.assertEqual(m.__file__,
930-
os.path.join(os.getcwd(), os.curdir, os.path.relpath(pyc_file)))
930+
os.path.join(os.getcwd(), os.path.relpath(pyc_file)))
931931
finally:
932932
os.remove(pyc_file)
933933

934934
def test___cached__(self):
935935
# Modules now also have an __cached__ that points to the pyc file.
936936
m = __import__(TESTFN)
937937
pyc_file = importlib.util.cache_from_source(TESTFN + '.py')
938-
self.assertEqual(m.__cached__, os.path.join(os.getcwd(), os.curdir, pyc_file))
938+
self.assertEqual(m.__cached__, os.path.join(os.getcwd(), pyc_file))
939939

940940
@skip_if_dont_write_bytecode
941941
def test___cached___legacy_pyc(self):
@@ -951,7 +951,7 @@ def test___cached___legacy_pyc(self):
951951
importlib.invalidate_caches()
952952
m = __import__(TESTFN)
953953
self.assertEqual(m.__cached__,
954-
os.path.join(os.getcwd(), os.curdir, os.path.relpath(pyc_file)))
954+
os.path.join(os.getcwd(), os.path.relpath(pyc_file)))
955955

956956
@skip_if_dont_write_bytecode
957957
def test_package___cached__(self):
@@ -971,10 +971,10 @@ def cleanup():
971971
m = __import__('pep3147.foo')
972972
init_pyc = importlib.util.cache_from_source(
973973
os.path.join('pep3147', '__init__.py'))
974-
self.assertEqual(m.__cached__, os.path.join(os.getcwd(), os.curdir, init_pyc))
974+
self.assertEqual(m.__cached__, os.path.join(os.getcwd(), init_pyc))
975975
foo_pyc = importlib.util.cache_from_source(os.path.join('pep3147', 'foo.py'))
976976
self.assertEqual(sys.modules['pep3147.foo'].__cached__,
977-
os.path.join(os.getcwd(), os.curdir, foo_pyc))
977+
os.path.join(os.getcwd(), foo_pyc))
978978

979979
def test_package___cached___from_pyc(self):
980980
# Like test___cached__ but ensuring __cached__ when imported from a
@@ -998,10 +998,10 @@ def cleanup():
998998
m = __import__('pep3147.foo')
999999
init_pyc = importlib.util.cache_from_source(
10001000
os.path.join('pep3147', '__init__.py'))
1001-
self.assertEqual(m.__cached__, os.path.join(os.getcwd(), os.curdir, init_pyc))
1001+
self.assertEqual(m.__cached__, os.path.join(os.getcwd(), init_pyc))
10021002
foo_pyc = importlib.util.cache_from_source(os.path.join('pep3147', 'foo.py'))
10031003
self.assertEqual(sys.modules['pep3147.foo'].__cached__,
1004-
os.path.join(os.getcwd(), os.curdir, foo_pyc))
1004+
os.path.join(os.getcwd(), foo_pyc))
10051005

10061006
def test_recompute_pyc_same_second(self):
10071007
# Even when the source file doesn't change timestamp, a change in

Lib/test/test_importlib/import_/test_path.py

+13-2
Original file line numberDiff line numberDiff line change
@@ -202,10 +202,11 @@ def __init__(self):
202202
def invalidate_caches(self):
203203
self.called = True
204204

205-
cache = {'leave_alone': object(), 'finder_to_invalidate': FakeFinder()}
205+
key = os.path.abspath('finder_to_invalidate')
206+
cache = {'leave_alone': object(), key: FakeFinder()}
206207
with util.import_state(path_importer_cache=cache):
207208
self.machinery.PathFinder.invalidate_caches()
208-
self.assertTrue(cache['finder_to_invalidate'].called)
209+
self.assertTrue(cache[key].called)
209210

210211
def test_invalidate_caches_clear_out_None(self):
211212
# Clear out None in sys.path_importer_cache() when invalidating caches.
@@ -214,6 +215,16 @@ def test_invalidate_caches_clear_out_None(self):
214215
self.machinery.PathFinder.invalidate_caches()
215216
self.assertEqual(len(cache), 0)
216217

218+
def test_invalidate_caches_clear_out_relative_path(self):
219+
class FakeFinder:
220+
def invalidate_caches(self):
221+
pass
222+
223+
cache = {'relative_path': FakeFinder()}
224+
with util.import_state(path_importer_cache=cache):
225+
self.machinery.PathFinder.invalidate_caches()
226+
self.assertEqual(cache, {})
227+
217228

218229
class FindModuleTests(FinderTests):
219230
def find(self, *args, **kwargs):

Lib/test/test_importlib/test_api.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -396,7 +396,7 @@ def find_module(self, *args):
396396
def invalidate_caches(self):
397397
self.called = True
398398

399-
key = 'gobledeegook'
399+
key = os.path.abspath('gobledeegook')
400400
meta_ins = InvalidatingNullFinder()
401401
path_ins = InvalidatingNullFinder()
402402
sys.meta_path.insert(0, meta_ins)

Lib/test/test_queue.py

+2-3
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@
1010
from test.support import import_helper
1111
from test.support import threading_helper
1212

13+
# queue module depends on threading primitives
14+
threading_helper.requires_working_threading(module=True)
1315

1416
py_queue = import_helper.import_fresh_module('queue', blocked=['_queue'])
1517
c_queue = import_helper.import_fresh_module('queue', fresh=['_queue'])
@@ -87,7 +89,6 @@ def do_exceptional_blocking_test(self,block_func, block_args, trigger_func,
8789
self.fail("trigger thread ended but event never set")
8890

8991

90-
@threading_helper.requires_working_threading()
9192
class BaseQueueTestMixin(BlockingTestMixin):
9293
def setUp(self):
9394
self.cum = 0
@@ -291,7 +292,6 @@ class CPriorityQueueTest(PriorityQueueTest, unittest.TestCase):
291292
class FailingQueueException(Exception): pass
292293

293294

294-
@threading_helper.requires_working_threading()
295295
class FailingQueueTest(BlockingTestMixin):
296296

297297
def setUp(self):
@@ -467,7 +467,6 @@ def consume_timeout(self, q, results, sentinel):
467467
return
468468
results.append(val)
469469

470-
@threading_helper.requires_working_threading()
471470
def run_threads(self, n_threads, q, inputs, feed_func, consume_func):
472471
results = []
473472
sentinel = None

0 commit comments

Comments
 (0)