extern crate ctrlc; extern crate docopt; extern crate env_logger; extern crate grep; extern crate ignore; #[cfg(windows)] extern crate kernel32; #[macro_use] extern crate lazy_static; extern crate libc; #[macro_use] extern crate log; extern crate memchr; extern crate memmap; extern crate num_cpus; extern crate regex; extern crate rustc_serialize; extern crate term; #[cfg(windows)] extern crate winapi; use std::error::Error; use std::io; use std::io::Write; use std::process; use std::result; use std::sync::{Arc, Mutex}; use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering}; use std::sync::mpsc; use std::thread; use std::cmp; use term::Terminal; use args::Args; use worker::Work; macro_rules! errored { ($($tt:tt)*) => { return Err(From::from(format!($($tt)*))); } } macro_rules! eprintln { ($($tt:tt)*) => {{ use std::io::Write; let _ = writeln!(&mut ::std::io::stderr(), $($tt)*); }} } mod args; mod atty; mod out; mod pathutil; mod printer; mod search_buffer; mod search_stream; #[cfg(windows)] mod terminal_win; mod worker; pub type Result = result::Result>; fn main() { match Args::parse().map(Arc::new).and_then(run) { Ok(count) if count == 0 => process::exit(1), Ok(_) => process::exit(0), Err(err) => { eprintln!("{}", err); process::exit(1); } } } fn run(args: Arc) -> Result { if args.never_match() { return Ok(0); } { let args = args.clone(); ctrlc::set_handler(move || { let stdout = io::stdout(); let mut stdout = stdout.lock(); let _ = args.stdout().reset(); let _ = stdout.flush(); process::exit(1); }); } let threads = cmp::max(1, args.threads() - 1); if args.files() { if threads == 1 || args.is_one_path() { run_files_one_thread(args) } else { run_files_parallel(args) } } else if args.type_list() { run_types(args) } else if threads == 1 || args.is_one_path() { run_one_thread(args) } else { run_parallel(args) } } fn run_parallel(args: Arc) -> Result { let out = Arc::new(Mutex::new(args.out())); let quiet_matched = QuietMatched::new(args.quiet()); let paths_searched = Arc::new(AtomicUsize::new(0)); let match_count = Arc::new(AtomicUsize::new(0)); args.walker_parallel().run(|| { let args = args.clone(); let quiet_matched = quiet_matched.clone(); let paths_searched = paths_searched.clone(); let match_count = match_count.clone(); let out = out.clone(); let mut outbuf = args.outbuf(); let mut worker = args.worker(); Box::new(move |result| { use ignore::WalkState::*; if quiet_matched.has_match() { return Quit; } let dent = match get_or_log_dir_entry(result, args.no_messages()) { None => return Continue, Some(dent) => dent, }; paths_searched.fetch_add(1, Ordering::SeqCst); outbuf.clear(); { // This block actually executes the search and prints the // results into outbuf. let mut printer = args.printer(&mut outbuf); let count = if dent.is_stdin() { worker.run(&mut printer, Work::Stdin) } else { worker.run(&mut printer, Work::DirEntry(dent)) }; match_count.fetch_add(count as usize, Ordering::SeqCst); if quiet_matched.set_match(count > 0) { return Quit; } } if !outbuf.get_ref().is_empty() { // This should be the only mutex in all of ripgrep. Since the // common case is to report a small number of matches relative // to the corpus, this really shouldn't matter much. // // Still, it'd be nice to send this on a channel, but then we'd // need to manage a pool of outbufs, which would complicate the // code. let mut out = out.lock().unwrap(); out.write(&outbuf); } Continue }) }); if !args.paths().is_empty() && paths_searched.load(Ordering::SeqCst) == 0 { if !args.no_messages() { eprint_nothing_searched(); } } Ok(match_count.load(Ordering::SeqCst) as u64) } fn run_one_thread(args: Arc) -> Result { let mut worker = args.worker(); let mut term = args.stdout(); let mut paths_searched: u64 = 0; let mut match_count = 0; for result in args.walker() { let dent = match get_or_log_dir_entry(result, args.no_messages()) { None => continue, Some(dent) => dent, }; let mut printer = args.printer(&mut term); if match_count > 0 { if args.quiet() { break; } if let Some(sep) = args.file_separator() { printer = printer.file_separator(sep); } } paths_searched += 1; match_count += if dent.is_stdin() { worker.run(&mut printer, Work::Stdin) } else { worker.run(&mut printer, Work::DirEntry(dent)) }; } if !args.paths().is_empty() && paths_searched == 0 { if !args.no_messages() { eprint_nothing_searched(); } } Ok(match_count) } fn run_files_parallel(args: Arc) -> Result { let print_args = args.clone(); let (tx, rx) = mpsc::channel::(); let print_thread = thread::spawn(move || { let term = print_args.stdout(); let mut printer = print_args.printer(term); let mut file_count = 0; for dent in rx.iter() { printer.path(dent.path()); file_count += 1; } file_count }); let no_messages = args.no_messages(); args.walker_parallel().run(move || { let tx = tx.clone(); Box::new(move |result| { if let Some(dent) = get_or_log_dir_entry(result, no_messages) { tx.send(dent).unwrap(); } ignore::WalkState::Continue }) }); Ok(print_thread.join().unwrap()) } fn run_files_one_thread(args: Arc) -> Result { let term = args.stdout(); let mut printer = args.printer(term); let mut file_count = 0; for result in args.walker() { let dent = match get_or_log_dir_entry(result, args.no_messages()) { None => continue, Some(dent) => dent, }; printer.path(dent.path()); file_count += 1; } Ok(file_count) } fn run_types(args: Arc) -> Result { let term = args.stdout(); let mut printer = args.printer(term); let mut ty_count = 0; for def in args.type_defs() { printer.type_def(def); ty_count += 1; } Ok(ty_count) } fn get_or_log_dir_entry( result: result::Result, no_messages: bool, ) -> Option { match result { Err(err) => { if !no_messages { eprintln!("{}", err); } None } Ok(dent) => { if let Some(err) = dent.error() { if !no_messages { eprintln!("{}", err); } } // A depth of 0 means the user gave the path explicitly, so we // should always try to search it. if dent.depth() == 0 { return Some(dent); } let ft = match dent.file_type() { None => return Some(dent), // entry is stdin Some(ft) => ft, }; if ft.is_file() { Some(dent) } else { None } } } } fn eprint_nothing_searched() { eprintln!("No files were searched, which means ripgrep probably \ applied a filter you didn't expect. \ Try running again with --debug."); } /// A simple thread safe abstraction for determining whether a search should /// stop if the user has requested quiet mode. #[derive(Clone, Debug)] pub struct QuietMatched(Arc>); impl QuietMatched { /// Create a new QuietMatched value. /// /// If quiet is true, then set_match and has_match will reflect whether /// a search should quit or not because it found a match. /// /// If quiet is false, then set_match is always a no-op and has_match /// always returns false. pub fn new(quiet: bool) -> QuietMatched { let atomic = if quiet { Some(AtomicBool::new(false)) } else { None }; QuietMatched(Arc::new(atomic)) } /// Returns true if and only if quiet mode is enabled and a match has /// occurred. pub fn has_match(&self) -> bool { match *self.0 { None => false, Some(ref matched) => matched.load(Ordering::SeqCst), } } /// Sets whether a match has occurred or not. /// /// If quiet mode is disabled, then this is a no-op. pub fn set_match(&self, yes: bool) -> bool { match *self.0 { None => false, Some(_) if !yes => false, Some(ref m) => { m.store(true, Ordering::SeqCst); true } } } }