summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJohn W. Parent <45471568+johnwparent@users.noreply.github.com>2024-04-16 14:02:02 -0400
committerGitHub <noreply@github.com>2024-04-16 11:02:02 -0700
commitde3b324983d83267943c7747b063fd6ffb574939 (patch)
tree8fcaf07fce320a1505dca97159f989ccba061c05
parent747cd374df925703fc5355a1ddefddb25ad470f2 (diff)
downloadspack-de3b324983d83267943c7747b063fd6ffb574939.tar.gz
spack-de3b324983d83267943c7747b063fd6ffb574939.tar.bz2
spack-de3b324983d83267943c7747b063fd6ffb574939.tar.xz
spack-de3b324983d83267943c7747b063fd6ffb574939.zip
Windows filesystem utilities (bugfix): improve SFN usage (#43645)
Reduce incidence of spurious errors by: * Ensuring we're passing the buffer by reference * Get the correct short string size from Windows API instead of computing ourselves * Ensure sufficient space for null terminator character Add test for `windows_sfn`
-rw-r--r--lib/spack/llnl/util/filesystem.py6
-rw-r--r--lib/spack/spack/test/llnl/util/filesystem.py38
2 files changed, 41 insertions, 3 deletions
diff --git a/lib/spack/llnl/util/filesystem.py b/lib/spack/llnl/util/filesystem.py
index 722f51f3db..f9cee1e118 100644
--- a/lib/spack/llnl/util/filesystem.py
+++ b/lib/spack/llnl/util/filesystem.py
@@ -1234,10 +1234,12 @@ def windows_sfn(path: os.PathLike):
import ctypes
k32 = ctypes.WinDLL("kernel32", use_last_error=True)
+ # Method with null values returns size of short path name
+ sz = k32.GetShortPathNameW(path, None, 0)
# stub Windows types TCHAR[LENGTH]
- TCHAR_arr = ctypes.c_wchar * len(path)
+ TCHAR_arr = ctypes.c_wchar * sz
ret_str = TCHAR_arr()
- k32.GetShortPathNameW(path, ret_str, len(path))
+ k32.GetShortPathNameW(path, ctypes.byref(ret_str), sz)
return ret_str.value
diff --git a/lib/spack/spack/test/llnl/util/filesystem.py b/lib/spack/spack/test/llnl/util/filesystem.py
index a38faaef27..f04c2455cc 100644
--- a/lib/spack/spack/test/llnl/util/filesystem.py
+++ b/lib/spack/spack/test/llnl/util/filesystem.py
@@ -912,7 +912,6 @@ def test_find_first_file(tmpdir, bfs_depth):
def test_rename_dest_exists(tmpdir):
-
@contextmanager
def setup_test_files():
a = tmpdir.join("a", "file1")
@@ -999,3 +998,40 @@ def test_rename_dest_exists(tmpdir):
assert os.path.exists(link2)
assert os.path.realpath(link2) == a
shutil.rmtree(tmpdir.join("f"))
+
+
+@pytest.mark.skipif(sys.platform != "win32", reason="No-op on non Windows")
+def test_windows_sfn(tmpdir):
+ # first check some standard Windows locations
+ # we know require sfn names
+ # this is basically a smoke test
+ # ensure spaces are replaced + path abbreviated
+ assert fs.windows_sfn("C:\\Program Files (x86)") == "C:\\PROGRA~2"
+ # ensure path without spaces is still properly shortened
+ assert fs.windows_sfn("C:\\ProgramData") == "C:\\PROGRA~3"
+
+ # test user created paths
+ # ensure longer path with spaces is properly abbreviated
+ a = tmpdir.join("d", "this is a test", "a", "still test")
+ # ensure longer path is properly abbreviated
+ b = tmpdir.join("d", "long_path_with_no_spaces", "more_long_path")
+ # ensure path not in need of abbreviation is properly roundtripped
+ c = tmpdir.join("d", "this", "is", "short")
+ # ensure paths that are the same in the first six letters
+ # are incremented post tilde
+ d = tmpdir.join("d", "longerpath1")
+ e = tmpdir.join("d", "longerpath2")
+ fs.mkdirp(a)
+ fs.mkdirp(b)
+ fs.mkdirp(c)
+ fs.mkdirp(d)
+ fs.mkdirp(e)
+ # check only for path of path we can control,
+ # pytest prefix may or may not be mangled by windows_sfn
+ # based on user/pytest config
+ assert "d\\THISIS~1\\a\\STILLT~1" in fs.windows_sfn(a)
+ assert "d\\LONG_P~1\\MORE_L~1" in fs.windows_sfn(b)
+ assert "d\\this\\is\\short" in fs.windows_sfn(c)
+ assert "d\\LONGER~1" in fs.windows_sfn(d)
+ assert "d\\LONGER~2" in fs.windows_sfn(e)
+ shutil.rmtree(tmpdir.join("d"))