mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 13:41:24 +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") | ||||
| _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", | ||||
|            "copytree", "move", "rmtree", "Error", "SpecialFileError", | ||||
|            "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) | ||||
| 
 | ||||
|         # 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: | ||||
|             pathext = [os.fsencode(ext) for ext in pathext] | ||||
|         # 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) | ||||
|             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): | ||||
|     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
	
	 Miss Skeleton (bot)
						Miss Skeleton (bot)