1+ use std:: fs:: FileType ;
12use std:: io;
23use std:: path:: { Path , PathBuf } ;
34
@@ -6,12 +7,15 @@ use std::path::{Path, PathBuf};
67pub fn copy_symlink ( src : impl AsRef < Path > , dst : impl AsRef < Path > ) {
78 let src = src. as_ref ( ) ;
89 let dst = dst. as_ref ( ) ;
9- if let Err ( e) = copy_symlink_raw ( src, dst) {
10+ let metadata = symlink_metadata ( src) ;
11+ if let Err ( e) = copy_symlink_raw ( metadata. file_type ( ) , src, dst) {
1012 panic ! ( "failed to copy symlink from `{}` to `{}`: {e}" , src. display( ) , dst. display( ) , ) ;
1113 }
1214}
1315
14- fn copy_symlink_raw ( src : impl AsRef < Path > , dst : impl AsRef < Path > ) -> io:: Result < ( ) > {
16+ fn copy_symlink_raw ( ty : FileType , src : impl AsRef < Path > , dst : impl AsRef < Path > ) -> io:: Result < ( ) > {
17+ #![ allow( unused_variables) ]
18+
1519 // Traverse symlink once to find path of target entity.
1620 let target_path = std:: fs:: read_link ( src) ?;
1721
@@ -54,7 +58,7 @@ pub fn copy_dir_all(src: impl AsRef<Path>, dst: impl AsRef<Path>) {
5458 if ty. is_dir ( ) {
5559 copy_dir_all_inner ( entry. path ( ) , dst. join ( entry. file_name ( ) ) ) ?;
5660 } else if ty. is_symlink ( ) {
57- copy_symlink_raw ( entry. path ( ) , dst. join ( entry. file_name ( ) ) ) ?;
61+ copy_symlink_raw ( ty , entry. path ( ) , dst. join ( entry. file_name ( ) ) ) ?;
5862 } else {
5963 std:: fs:: copy ( entry. path ( ) , dst. join ( entry. file_name ( ) ) ) ?;
6064 }
@@ -80,6 +84,21 @@ pub fn read_dir_entries<P: AsRef<Path>, F: FnMut(&Path)>(dir: P, mut callback: F
8084 }
8185}
8286
87+ /// A wrapper around [`build_helper::fs::recursive_remove`] which includes the file path in the
88+ /// panic message.
89+ ///
90+ /// This handles removing symlinks on Windows (e.g. symlink-to-file will be removed via
91+ /// [`std::fs::remove_file`] while symlink-to-dir will be removed via [`std::fs::remove_dir`]).
92+ #[ track_caller]
93+ pub fn recursive_remove < P : AsRef < Path > > ( path : P ) {
94+ if let Err ( e) = build_helper:: fs:: recursive_remove ( path. as_ref ( ) ) {
95+ panic ! (
96+ "failed to recursive remove filesystem entities at `{}`: {e}" ,
97+ path. as_ref( ) . display( )
98+ ) ;
99+ }
100+ }
101+
83102/// A wrapper around [`std::fs::remove_file`] which includes the file path in the panic message.
84103#[ track_caller]
85104pub fn remove_file < P : AsRef < Path > > ( path : P ) {
0 commit comments