fd/src/main.rs

103 lines
2.8 KiB
Rust
Raw Normal View History

2017-05-12 11:50:03 +02:00
extern crate walkdir;
extern crate regex;
extern crate getopts;
2017-05-12 13:02:20 +02:00
extern crate ansi_term;
2017-05-12 11:50:03 +02:00
use std::env;
use std::path::Path;
use std::io::Write;
use std::process;
use std::error::Error;
use walkdir::{WalkDir, DirEntry, WalkDirIterator};
2017-05-12 12:02:25 +02:00
use regex::{Regex, RegexBuilder};
2017-05-12 11:50:03 +02:00
use getopts::Options;
2017-05-12 13:02:20 +02:00
use ansi_term::Colour;
/// Print a search result to the console.
fn print_entry(entry: &DirEntry, path_str: &str) {
let style = match entry {
e if e.path_is_symbolic_link() => Colour::Purple,
e if e.path().is_dir() => Colour::Cyan,
_ => Colour::White
};
println!("{}", style.paint(path_str));
}
2017-05-12 11:50:03 +02:00
2017-05-12 12:02:25 +02:00
/// Check if filename of entry starts with a dot.
2017-05-12 11:50:03 +02:00
fn is_hidden(entry: &DirEntry) -> bool {
entry.file_name()
.to_str()
.map(|s| s.starts_with("."))
.unwrap_or(false)
}
2017-05-12 12:02:25 +02:00
/// Recursively scan the given root path and search for pathnames matching the
/// pattern.
2017-05-12 11:50:03 +02:00
fn scan(root: &Path, pattern: &Regex) {
let walker = WalkDir::new(root).into_iter();
for entry in walker.filter_entry(|e| !is_hidden(e))
.filter_map(|e| e.ok()) {
if entry.path() == root {
continue;
}
2017-05-12 13:02:20 +02:00
let path_relative =
match entry.path().strip_prefix(root) {
Ok(r) => r,
Err(_) => continue
};
2017-05-12 13:02:20 +02:00
let path_str = match path_relative.to_str() {
Some(p) => p,
2017-05-12 11:50:03 +02:00
None => continue
};
pattern.find(path_str)
2017-05-12 13:02:20 +02:00
.map(|_| print_entry(&entry, path_str));
2017-05-12 11:50:03 +02:00
}
}
2017-05-12 12:02:25 +02:00
/// Print error message to stderr and exit with status `1`.
fn error(message: &str) -> ! {
writeln!(&mut std::io::stderr(), "{}", message)
2017-05-12 11:50:03 +02:00
.expect("Failed writing to stderr");
process::exit(1);
}
fn main() {
let args: Vec<String> = env::args().collect();
let mut opts = Options::new();
opts.optflag("h", "help", "print this help message");
2017-05-12 12:09:46 +02:00
opts.optflag("s", "sensitive", "case-sensitive search");
2017-05-12 11:50:03 +02:00
let matches = match opts.parse(&args[1..]) {
Ok(m) => m,
Err(e) => error(e.description())
2017-05-12 11:50:03 +02:00
};
if matches.opt_present("h") {
let brief = "Usage: fd [PATTERN]";
print!("{}", opts.usage(&brief));
process::exit(1);
}
2017-05-12 12:09:46 +02:00
let case_insensitive = !matches.opt_present("s");
2017-05-12 11:50:03 +02:00
let empty = String::new();
let pattern = matches.free.get(0).unwrap_or(&empty);
let current_dir_buf = match env::current_dir() {
Ok(cd) => cd,
Err(_) => error("Could not get current directory!")
};
2017-05-12 11:50:03 +02:00
let current_dir = current_dir_buf.as_path();
match RegexBuilder::new(pattern)
.case_insensitive(case_insensitive)
.build() {
Ok(re) => scan(current_dir, &re),
Err(err) => error(err.description())
2017-05-12 11:50:03 +02:00
}
}