Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit ba4e4fc

Browse files
committedApr 28, 2015
std: Implement fs::DirBuilder
This is the last remaining portion of #24796
1 parent 9348700 commit ba4e4fc

File tree

4 files changed

+105
-17
lines changed

4 files changed

+105
-17
lines changed
 

‎src/libstd/fs.rs

+57-5
Original file line numberDiff line numberDiff line change
@@ -152,6 +152,15 @@ pub struct Permissions(fs_imp::FilePermissions);
152152
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
153153
pub struct FileType(fs_imp::FileType);
154154

155+
/// A builder used to create directories in various manners.
156+
///
157+
/// This builder also supports platform-specific options.
158+
#[unstable(feature = "dir_builder", reason = "recently added API")]
159+
pub struct DirBuilder {
160+
inner: fs_imp::DirBuilder,
161+
recursive: bool,
162+
}
163+
155164
impl File {
156165
/// Attempts to open a file in read-only mode.
157166
///
@@ -997,7 +1006,7 @@ pub fn canonicalize<P: AsRef<Path>>(path: P) -> io::Result<PathBuf> {
9971006
/// ```
9981007
#[stable(feature = "rust1", since = "1.0.0")]
9991008
pub fn create_dir<P: AsRef<Path>>(path: P) -> io::Result<()> {
1000-
fs_imp::mkdir(path.as_ref())
1009+
DirBuilder::new().create(path.as_ref())
10011010
}
10021011

10031012
/// Recursively create a directory and all of its parent components if they
@@ -1022,10 +1031,7 @@ pub fn create_dir<P: AsRef<Path>>(path: P) -> io::Result<()> {
10221031
/// ```
10231032
#[stable(feature = "rust1", since = "1.0.0")]
10241033
pub fn create_dir_all<P: AsRef<Path>>(path: P) -> io::Result<()> {
1025-
let path = path.as_ref();
1026-
if path == Path::new("") || path.is_dir() { return Ok(()) }
1027-
if let Some(p) = path.parent() { try!(create_dir_all(p)) }
1028-
create_dir(path)
1034+
DirBuilder::new().recursive(true).create(path.as_ref())
10291035
}
10301036

10311037
/// Removes an existing, empty directory.
@@ -1286,6 +1292,52 @@ pub fn set_permissions<P: AsRef<Path>>(path: P, perm: Permissions) -> io::Result
12861292
fs_imp::set_perm(path.as_ref(), perm.0)
12871293
}
12881294

1295+
impl DirBuilder {
1296+
/// Creates a new set of options with default mode/security settings for all
1297+
/// platforms and also non-recursive.
1298+
pub fn new() -> DirBuilder {
1299+
DirBuilder {
1300+
inner: fs_imp::DirBuilder::new(),
1301+
recursive: false,
1302+
}
1303+
}
1304+
1305+
/// Indicate that directories create should be created recursively, creating
1306+
/// all parent directories if they do not exist with the same security and
1307+
/// permissions settings.
1308+
///
1309+
/// This option defaults to `false`
1310+
pub fn recursive(&mut self, recursive: bool) -> &mut Self {
1311+
self.recursive = recursive;
1312+
self
1313+
}
1314+
1315+
/// Create the specified directory with the options configured in this
1316+
/// builder.
1317+
pub fn create<P: AsRef<Path>>(&self, path: P) -> io::Result<()> {
1318+
let path = path.as_ref();
1319+
if self.recursive {
1320+
self.create_dir_all(path)
1321+
} else {
1322+
self.inner.mkdir(path)
1323+
}
1324+
}
1325+
1326+
fn create_dir_all(&self, path: &Path) -> io::Result<()> {
1327+
if path == Path::new("") || path.is_dir() { return Ok(()) }
1328+
if let Some(p) = path.parent() {
1329+
try!(self.create_dir_all(p))
1330+
}
1331+
self.inner.mkdir(path)
1332+
}
1333+
}
1334+
1335+
impl AsInnerMut<fs_imp::DirBuilder> for DirBuilder {
1336+
fn as_inner_mut(&mut self) -> &mut fs_imp::DirBuilder {
1337+
&mut self.inner
1338+
}
1339+
}
1340+
12891341
#[cfg(test)]
12901342
mod tests {
12911343
#![allow(deprecated)] //rand

‎src/libstd/sys/unix/ext/fs.rs

+16
Original file line numberDiff line numberDiff line change
@@ -189,3 +189,19 @@ pub fn symlink<P: AsRef<Path>, Q: AsRef<Path>>(src: P, dst: Q) -> io::Result<()>
189189
{
190190
sys::fs2::symlink(src.as_ref(), dst.as_ref())
191191
}
192+
193+
#[unstable(feature = "dir_builder", reason = "recently added API")]
194+
/// An extension trait for `fs::DirBuilder` for unix-specific options.
195+
pub trait DirBuilderExt {
196+
/// Sets the mode to create new directories with. This option defaults to
197+
/// 0o777.
198+
fn mode(&mut self, mode: raw::mode_t) -> &mut Self;
199+
}
200+
201+
impl DirBuilderExt for fs::DirBuilder {
202+
fn mode(&mut self, mode: raw::mode_t) -> &mut fs::DirBuilder {
203+
self.as_inner_mut().set_mode(mode);
204+
self
205+
}
206+
}
207+

‎src/libstd/sys/unix/fs2.rs

+18-6
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,8 @@ pub struct FilePermissions { mode: mode_t }
6161
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
6262
pub struct FileType { mode: mode_t }
6363

64+
pub struct DirBuilder { mode: mode_t }
65+
6466
impl FileAttr {
6567
pub fn size(&self) -> u64 { self.stat.st_size as u64 }
6668
pub fn perm(&self) -> FilePermissions {
@@ -342,6 +344,22 @@ impl File {
342344
pub fn fd(&self) -> &FileDesc { &self.0 }
343345
}
344346

347+
impl DirBuilder {
348+
pub fn new() -> DirBuilder {
349+
DirBuilder { mode: 0o777 }
350+
}
351+
352+
pub fn mkdir(&self, p: &Path) -> io::Result<()> {
353+
let p = try!(cstr(p));
354+
try!(cvt(unsafe { libc::mkdir(p.as_ptr(), self.mode) }));
355+
Ok(())
356+
}
357+
358+
pub fn set_mode(&mut self, mode: mode_t) {
359+
self.mode = mode;
360+
}
361+
}
362+
345363
fn cstr(path: &Path) -> io::Result<CString> {
346364
path.as_os_str().to_cstring().ok_or(
347365
io::Error::new(io::ErrorKind::InvalidInput, "path contained a null"))
@@ -401,12 +419,6 @@ impl fmt::Debug for File {
401419
}
402420
}
403421

404-
pub fn mkdir(p: &Path) -> io::Result<()> {
405-
let p = try!(cstr(p));
406-
try!(cvt(unsafe { libc::mkdir(p.as_ptr(), 0o777) }));
407-
Ok(())
408-
}
409-
410422
pub fn readdir(p: &Path) -> io::Result<ReadDir> {
411423
let root = Arc::new(p.to_path_buf());
412424
let p = try!(cstr(p));

‎src/libstd/sys/windows/fs2.rs

+14-6
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,8 @@ pub struct OpenOptions {
7070
#[derive(Clone, PartialEq, Eq, Debug)]
7171
pub struct FilePermissions { attrs: libc::DWORD }
7272

73+
pub struct DirBuilder;
74+
7375
impl Iterator for ReadDir {
7476
type Item = io::Result<DirEntry>;
7577
fn next(&mut self) -> Option<io::Result<DirEntry>> {
@@ -425,12 +427,18 @@ impl FileType {
425427
pub fn is_symlink(&self) -> bool { *self == FileType::Symlink }
426428
}
427429

428-
pub fn mkdir(p: &Path) -> io::Result<()> {
429-
let p = to_utf16(p);
430-
try!(cvt(unsafe {
431-
libc::CreateDirectoryW(p.as_ptr(), ptr::null_mut())
432-
}));
433-
Ok(())
430+
impl DirBuilder {
431+
pub fn new() -> DirBuilder {
432+
DirBuilder { mode: 0o777 }
433+
}
434+
435+
pub fn mkdir(&self, p: &Path) -> io::Result<()> {
436+
let p = to_utf16(p);
437+
try!(cvt(unsafe {
438+
libc::CreateDirectoryW(p.as_ptr(), ptr::null_mut())
439+
}));
440+
Ok(())
441+
}
434442
}
435443

436444
pub fn readdir(p: &Path) -> io::Result<ReadDir> {

0 commit comments

Comments
 (0)
Please sign in to comment.