2015-01-16 16:19:41 -06:00
|
|
|
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
// Regression test for #20797.
|
|
|
|
|
|
|
|
use std::default::Default;
|
2015-03-17 15:33:26 -05:00
|
|
|
use std::io;
|
|
|
|
use std::fs;
|
|
|
|
use std::path::{PathBuf, Path};
|
|
|
|
|
|
|
|
pub trait PathExtensions {
|
|
|
|
fn is_dir(&self) -> bool { false }
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PathExtensions for PathBuf {}
|
2015-01-16 16:19:41 -06:00
|
|
|
|
|
|
|
/// A strategy for acquiring more subpaths to walk.
|
|
|
|
pub trait Strategy {
|
2015-03-17 15:33:26 -05:00
|
|
|
type P: PathExtensions;
|
|
|
|
/// Get additional subpaths from a given path.
|
|
|
|
fn get_more(&self, item: &Self::P) -> io::Result<Vec<Self::P>>;
|
|
|
|
/// Determine whether a path should be walked further.
|
|
|
|
/// This is run against each item from `get_more()`.
|
|
|
|
fn prune(&self, p: &Self::P) -> bool;
|
2015-01-16 16:19:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// The basic fully-recursive strategy. Nothing is pruned.
|
|
|
|
#[derive(Copy, Default)]
|
|
|
|
pub struct Recursive;
|
|
|
|
|
|
|
|
impl Strategy for Recursive {
|
2015-03-17 15:33:26 -05:00
|
|
|
type P = PathBuf;
|
|
|
|
fn get_more(&self, p: &PathBuf) -> io::Result<Vec<PathBuf>> {
|
|
|
|
Ok(fs::read_dir(p).unwrap().map(|s| s.unwrap().path()).collect())
|
|
|
|
}
|
2015-01-16 16:19:41 -06:00
|
|
|
|
2015-03-17 15:33:26 -05:00
|
|
|
fn prune(&self, _: &PathBuf) -> bool { false }
|
2015-01-16 16:19:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// A directory walker of `P` using strategy `S`.
|
|
|
|
pub struct Subpaths<S: Strategy> {
|
|
|
|
stack: Vec<S::P>,
|
|
|
|
strategy: S,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<S: Strategy> Subpaths<S> {
|
2015-03-17 15:33:26 -05:00
|
|
|
/// Create a directory walker with a root path and strategy.
|
|
|
|
pub fn new(p: &S::P, strategy: S) -> io::Result<Subpaths<S>> {
|
|
|
|
let stack = try!(strategy.get_more(p));
|
|
|
|
Ok(Subpaths { stack: stack, strategy: strategy })
|
|
|
|
}
|
2015-01-16 16:19:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<S: Default + Strategy> Subpaths<S> {
|
2015-03-17 15:33:26 -05:00
|
|
|
/// Create a directory walker with a root path and a default strategy.
|
|
|
|
pub fn walk(p: &S::P) -> io::Result<Subpaths<S>> {
|
|
|
|
Subpaths::new(p, Default::default())
|
|
|
|
}
|
2015-01-16 16:19:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<S: Default + Strategy> Default for Subpaths<S> {
|
2015-03-17 15:33:26 -05:00
|
|
|
fn default() -> Subpaths<S> {
|
|
|
|
Subpaths { stack: Vec::new(), strategy: Default::default() }
|
|
|
|
}
|
2015-01-16 16:19:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<S: Strategy> Iterator for Subpaths<S> {
|
2015-03-17 15:33:26 -05:00
|
|
|
type Item = S::P;
|
|
|
|
fn next (&mut self) -> Option<S::P> {
|
|
|
|
let mut opt_path = self.stack.pop();
|
|
|
|
while opt_path.is_some() && self.strategy.prune(opt_path.as_ref().unwrap()) {
|
|
|
|
opt_path = self.stack.pop();
|
|
|
|
}
|
|
|
|
match opt_path {
|
|
|
|
Some(path) => {
|
|
|
|
if path.is_dir() {
|
|
|
|
let result = self.strategy.get_more(&path);
|
|
|
|
match result {
|
|
|
|
Ok(dirs) => { self.stack.extend(dirs.into_iter()); },
|
|
|
|
Err(..) => { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Some(path)
|
|
|
|
}
|
|
|
|
None => None,
|
2015-01-16 16:19:41 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-17 15:33:26 -05:00
|
|
|
fn foo() {
|
|
|
|
let mut walker: Subpaths<Recursive> = Subpaths::walk(&PathBuf::new("/home")).unwrap();
|
2015-01-16 16:19:41 -06:00
|
|
|
}
|
2015-03-17 15:33:26 -05:00
|
|
|
|
|
|
|
fn main() {}
|