1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
use std::fs::{DirEntry, ReadDir, read_dir, metadata};
use std::io;
use std::path::{Path, PathBuf};

pub struct Walker {
  root: PathBuf,
  cur: Option<ReadDir>,
  stack: Vec<ReadDir>,
}

impl Walker {
  pub fn new(path: &Path) -> io::Result<Self> {
    let cur = try!(read_dir(path));
    Ok(Walker {
      root: path.to_path_buf(),
      cur: Some(cur),
      stack: Vec::new()
    })
  }

  pub fn rewind(&mut self) -> io::Result<()> {
    self.cur = Some(try!(read_dir(self.root.as_path())));
    self.stack = Vec::new();
    Ok(())
  }
}

fn is_dir(path: &Path) -> bool {
  match metadata(path) {
    Ok(m) => m.is_dir(),
    Err(_) => false
  }
}

impl Iterator for Walker {
  type Item = io::Result<DirEntry>;

  fn next(&mut self) -> Option<io::Result<DirEntry>> {
    match self.cur {
      None => return None,
      Some(ref mut d) => {
        match d.next() {
          Some(Err(e)) => return Some(Err(e)),
          Some(Ok(entry)) => {
            let path = entry.path();
            if is_dir(path.clone().as_path()) {
              match read_dir(path) {
                Err(e) => return Some(Err(e)),
                Ok(next) => self.stack.push(next)
              };
            }
            
            return Some(Ok(entry))
          },
          None => {}
        }
      }
    }

    match self.stack.pop() {
      None => return None,
      Some(next) => {
        self.cur = Some(next);
        self.next()
      }
    }
  }
}

#[cfg(test)]
mod test {
  use std::env::temp_dir;
  use std::fs;
  use super::Walker;
  
  macro_rules! check {($e:expr) => (
    match $e {
      Ok(t) => t,
      Err(e) => panic!("{} failed with: {}", stringify!($e), e),
    }
  )}

  #[test]
  fn test_walker() {
    let mut tmp = temp_dir();
    tmp.push("walk_dir");
    let dir = tmp.as_path();
    check!(fs::create_dir(dir));

    let dir1 = &dir.join("01/02/03");
    check!(fs::create_dir_all(dir1));
    check!(fs::File::create(&dir1.join("04")));

    let dir2 = &dir.join("11/12/13");
    check!(fs::create_dir_all(dir2));
    check!(fs::File::create(&dir2.join("14")));

    let files = check!(Walker::new(dir));
    let mut cur = [0; 2];
    for f in files {
      let f = f.unwrap().path();
      let stem = f.file_stem().unwrap().to_str().unwrap();
      let root = stem.as_bytes()[0] - b'0';
      let name = stem.as_bytes()[1] - b'0';
      assert!(cur[root as usize] < name);
      cur[root as usize] = name;
    }

    check!(fs::remove_dir_all(dir));
  }
}