Skip to content

Commit 8e30df7

Browse files
authored
Rollup merge of #138671 - ChrisDenton:filetype, r=joshtriplett
Fix `FileType` `PartialEq` implementation on Windows Fixes #138668 On Windows the [`FileType`](https://doc.rust-lang.org/stable/std/fs/struct.FileType.html) struct was deriving `PartialEq` which in turn means it was doing a bit-for-bit comparison on the file attributes and reparse point. This is wrong because `attributes` may contain many things unrelated to file type. `FileType` on Windows allows for four possible combinations (see also [`FileTypeExt`](https://doc.rust-lang.org/stable/std/os/windows/fs/trait.FileTypeExt.html)): `file`, `dir`, `symlink_file` and `symlink_dir`. So the new implementation makes sure both symlink and directory information match (and only those things). This could be considered just a bug fix but it is a behaviour change so someone from libs-api might want to FCP this (or might not)...
2 parents 0fc6279 + 6b2fa32 commit 8e30df7

File tree

2 files changed

+32
-18
lines changed

2 files changed

+32
-18
lines changed

library/std/src/fs/tests.rs

+17
Original file line numberDiff line numberDiff line change
@@ -1719,6 +1719,23 @@ fn test_eq_direntry_metadata() {
17191719
}
17201720
}
17211721

1722+
/// Test that windows file type equality is not affected by attributes unrelated
1723+
/// to the file type.
1724+
#[test]
1725+
#[cfg(target_os = "windows")]
1726+
fn test_eq_windows_file_type() {
1727+
let tmpdir = tmpdir();
1728+
let file1 = File::create(tmpdir.join("file1")).unwrap();
1729+
let file2 = File::create(tmpdir.join("file2")).unwrap();
1730+
assert_eq!(file1.metadata().unwrap().file_type(), file2.metadata().unwrap().file_type());
1731+
1732+
// Change the readonly attribute of one file.
1733+
let mut perms = file1.metadata().unwrap().permissions();
1734+
perms.set_readonly(true);
1735+
file1.set_permissions(perms).unwrap();
1736+
assert_eq!(file1.metadata().unwrap().file_type(), file2.metadata().unwrap().file_type());
1737+
}
1738+
17221739
/// Regression test for https://github.com/rust-lang/rust/issues/50619.
17231740
#[test]
17241741
#[cfg(target_os = "linux")]

library/std/src/sys/fs/windows.rs

+15-18
Original file line numberDiff line numberDiff line change
@@ -41,8 +41,8 @@ pub struct FileAttr {
4141

4242
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
4343
pub struct FileType {
44-
attributes: u32,
45-
reparse_tag: u32,
44+
is_directory: bool,
45+
is_symlink: bool,
4646
}
4747

4848
pub struct ReadDir {
@@ -1111,32 +1111,29 @@ impl FileTimes {
11111111
}
11121112

11131113
impl FileType {
1114-
fn new(attrs: u32, reparse_tag: u32) -> FileType {
1115-
FileType { attributes: attrs, reparse_tag }
1114+
fn new(attributes: u32, reparse_tag: u32) -> FileType {
1115+
let is_directory = attributes & c::FILE_ATTRIBUTE_DIRECTORY != 0;
1116+
let is_symlink = {
1117+
let is_reparse_point = attributes & c::FILE_ATTRIBUTE_REPARSE_POINT != 0;
1118+
let is_reparse_tag_name_surrogate = reparse_tag & 0x20000000 != 0;
1119+
is_reparse_point && is_reparse_tag_name_surrogate
1120+
};
1121+
FileType { is_directory, is_symlink }
11161122
}
11171123
pub fn is_dir(&self) -> bool {
1118-
!self.is_symlink() && self.is_directory()
1124+
!self.is_symlink && self.is_directory
11191125
}
11201126
pub fn is_file(&self) -> bool {
1121-
!self.is_symlink() && !self.is_directory()
1127+
!self.is_symlink && !self.is_directory
11221128
}
11231129
pub fn is_symlink(&self) -> bool {
1124-
self.is_reparse_point() && self.is_reparse_tag_name_surrogate()
1130+
self.is_symlink
11251131
}
11261132
pub fn is_symlink_dir(&self) -> bool {
1127-
self.is_symlink() && self.is_directory()
1133+
self.is_symlink && self.is_directory
11281134
}
11291135
pub fn is_symlink_file(&self) -> bool {
1130-
self.is_symlink() && !self.is_directory()
1131-
}
1132-
fn is_directory(&self) -> bool {
1133-
self.attributes & c::FILE_ATTRIBUTE_DIRECTORY != 0
1134-
}
1135-
fn is_reparse_point(&self) -> bool {
1136-
self.attributes & c::FILE_ATTRIBUTE_REPARSE_POINT != 0
1137-
}
1138-
fn is_reparse_tag_name_surrogate(&self) -> bool {
1139-
self.reparse_tag & 0x20000000 != 0
1136+
self.is_symlink && !self.is_directory
11401137
}
11411138
}
11421139

0 commit comments

Comments
 (0)