mirror of
				https://github.com/python/cpython.git
				synced 2025-11-03 23:21:29 +00:00 
			
		
		
		
	[3.9] bpo-40592: shutil.which will not return None anymore if ; is the last char in PATHEXT (GH-20088) (GH-22912)
shutil.which will not return None anymore for empty str in PATHEXT
Empty PATHEXT will now be defaulted to _WIN_DEFAULT_PATHEXT
(cherry picked from commit da6f098188)
Co-authored-by: Christopher Marchfelder <marchfelder@googlemail.com>
			
			
This commit is contained in:
		
							parent
							
								
									f8d96b98a4
								
							
						
					
					
						commit
						c437fe39cf
					
				
					 3 changed files with 24 additions and 1 deletions
				
			
		| 
						 | 
					@ -53,6 +53,9 @@
 | 
				
			||||||
_USE_CP_SENDFILE = hasattr(os, "sendfile") and sys.platform.startswith("linux")
 | 
					_USE_CP_SENDFILE = hasattr(os, "sendfile") and sys.platform.startswith("linux")
 | 
				
			||||||
_HAS_FCOPYFILE = posix and hasattr(posix, "_fcopyfile")  # macOS
 | 
					_HAS_FCOPYFILE = posix and hasattr(posix, "_fcopyfile")  # macOS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# CMD defaults in Windows 10
 | 
				
			||||||
 | 
					_WIN_DEFAULT_PATHEXT = ".COM;.EXE;.BAT;.CMD;.VBS;.JS;.WS;.MSC"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
__all__ = ["copyfileobj", "copyfile", "copymode", "copystat", "copy", "copy2",
 | 
					__all__ = ["copyfileobj", "copyfile", "copymode", "copystat", "copy", "copy2",
 | 
				
			||||||
           "copytree", "move", "rmtree", "Error", "SpecialFileError",
 | 
					           "copytree", "move", "rmtree", "Error", "SpecialFileError",
 | 
				
			||||||
           "ExecError", "make_archive", "get_archive_formats",
 | 
					           "ExecError", "make_archive", "get_archive_formats",
 | 
				
			||||||
| 
						 | 
					@ -1415,7 +1418,9 @@ def which(cmd, mode=os.F_OK | os.X_OK, path=None):
 | 
				
			||||||
            path.insert(0, curdir)
 | 
					            path.insert(0, curdir)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # PATHEXT is necessary to check on Windows.
 | 
					        # PATHEXT is necessary to check on Windows.
 | 
				
			||||||
        pathext = os.environ.get("PATHEXT", "").split(os.pathsep)
 | 
					        pathext_source = os.getenv("PATHEXT") or _WIN_DEFAULT_PATHEXT
 | 
				
			||||||
 | 
					        pathext = [ext for ext in pathext_source.split(os.pathsep) if ext]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if use_bytes:
 | 
					        if use_bytes:
 | 
				
			||||||
            pathext = [os.fsencode(ext) for ext in pathext]
 | 
					            pathext = [os.fsencode(ext) for ext in pathext]
 | 
				
			||||||
        # See if the given file matches any of the expected path extensions.
 | 
					        # See if the given file matches any of the expected path extensions.
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1848,6 +1848,23 @@ def test_pathext(self):
 | 
				
			||||||
            rv = shutil.which(program, path=self.temp_dir)
 | 
					            rv = shutil.which(program, path=self.temp_dir)
 | 
				
			||||||
            self.assertEqual(rv, temp_filexyz.name)
 | 
					            self.assertEqual(rv, temp_filexyz.name)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # Issue 40592: See https://bugs.python.org/issue40592
 | 
				
			||||||
 | 
					    @unittest.skipUnless(sys.platform == "win32", 'test specific to Windows')
 | 
				
			||||||
 | 
					    def test_pathext_with_empty_str(self):
 | 
				
			||||||
 | 
					        ext = ".xyz"
 | 
				
			||||||
 | 
					        temp_filexyz = tempfile.NamedTemporaryFile(dir=self.temp_dir,
 | 
				
			||||||
 | 
					                                                   prefix="Tmp2", suffix=ext)
 | 
				
			||||||
 | 
					        self.addCleanup(temp_filexyz.close)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        # strip path and extension
 | 
				
			||||||
 | 
					        program = os.path.basename(temp_filexyz.name)
 | 
				
			||||||
 | 
					        program = os.path.splitext(program)[0]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        with support.EnvironmentVarGuard() as env:
 | 
				
			||||||
 | 
					            env['PATHEXT'] = f"{ext};"  # note the ;
 | 
				
			||||||
 | 
					            rv = shutil.which(program, path=self.temp_dir)
 | 
				
			||||||
 | 
					            self.assertEqual(rv, temp_filexyz.name)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class TestWhichBytes(TestWhich):
 | 
					class TestWhichBytes(TestWhich):
 | 
				
			||||||
    def setUp(self):
 | 
					    def setUp(self):
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1 @@
 | 
				
			||||||
 | 
					:func:`shutil.which` now ignores empty entries in :envvar:`PATHEXT` instead of treating them as a match.
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue