summaryrefslogtreecommitdiffstats
path: root/src/args.rs
diff options
context:
space:
mode:
authorAndrew Gallant <jamslam@gmail.com>2016-11-05 21:44:15 -0400
committerAndrew Gallant <jamslam@gmail.com>2016-11-05 21:45:55 -0400
commitb272be25fa7b616689384d2e4bec3e01715e2477 (patch)
tree8743d1ec86294f81da800c024f094a4846158527 /src/args.rs
parent1aeae3e22da4e9ffa78ece485369e7a24744b2d4 (diff)
Add parallel recursive directory iterator.
This adds a new walk type in the `ignore` crate, `WalkParallel`, which provides a way for recursively iterating over a set of paths in parallel while respecting various ignore rules. The API is a bit strange, as a closure producing a closure isn't something one often sees, but it does seem to work well. This also allowed us to simplify much of the worker logic in ripgrep proper, where MultiWorker is now gone.
Diffstat (limited to 'src/args.rs')
-rw-r--r--src/args.rs105
1 files changed, 27 insertions, 78 deletions
diff --git a/src/args.rs b/src/args.rs
index 9d2923b8..017ade4c 100644
--- a/src/args.rs
+++ b/src/args.rs
@@ -1,4 +1,3 @@
-use std::cmp;
use std::env;
use std::io;
use std::path::{Path, PathBuf};
@@ -21,10 +20,9 @@ use ignore::types::{FileTypeDef, Types, TypesBuilder};
use ignore;
use out::{Out, ColoredTerminal};
use printer::Printer;
-use search_buffer::BufferSearcher;
-use search_stream::{InputBuffer, Searcher};
#[cfg(windows)]
use terminal_win::WindowsBuffer;
+use worker::{Worker, WorkerBuilder};
use Result;
@@ -364,7 +362,7 @@ impl RawArgs {
};
let threads =
if self.flag_threads == 0 {
- cmp::min(8, num_cpus::get())
+ num_cpus::get()
} else {
self.flag_threads
};
@@ -576,18 +574,6 @@ impl Args {
self.grep.clone()
}
- /// Creates a new input buffer that is used in searching.
- pub fn input_buffer(&self) -> InputBuffer {
- let mut inp = InputBuffer::new();
- inp.eol(self.eol);
- inp
- }
-
- /// Whether we should prefer memory maps for searching or not.
- pub fn mmap(&self) -> bool {
- self.mmap
- }
-
/// Whether ripgrep should be quiet or not.
pub fn quiet(&self) -> bool {
self.quiet
@@ -662,47 +648,27 @@ impl Args {
&self.paths
}
- /// Create a new line based searcher whose configuration is taken from the
- /// command line. This searcher supports a dizzying array of features:
- /// inverted matching, line counting, context control and more.
- pub fn searcher<'a, R: io::Read, W: Send + Terminal>(
- &self,
- inp: &'a mut InputBuffer,
- printer: &'a mut Printer<W>,
- grep: &'a Grep,
- path: &'a Path,
- rdr: R,
- ) -> Searcher<'a, R, W> {
- Searcher::new(inp, printer, grep, path, rdr)
- .after_context(self.after_context)
- .before_context(self.before_context)
- .count(self.count)
- .files_with_matches(self.files_with_matches)
- .eol(self.eol)
- .line_number(self.line_number)
- .invert_match(self.invert_match)
- .quiet(self.quiet)
- .text(self.text)
+ /// Returns true if there is exactly one file path given to search.
+ pub fn is_one_path(&self) -> bool {
+ self.paths.len() == 1
+ && (self.paths[0] == Path::new("-") || self.paths[0].is_file())
}
- /// Create a new line based searcher whose configuration is taken from the
- /// command line. This search operates on an entire file all once (which
- /// may have been memory mapped).
- pub fn searcher_buffer<'a, W: Send + Terminal>(
- &self,
- printer: &'a mut Printer<W>,
- grep: &'a Grep,
- path: &'a Path,
- buf: &'a [u8],
- ) -> BufferSearcher<'a, W> {
- BufferSearcher::new(printer, grep, path, buf)
+ /// Create a worker whose configuration is taken from the
+ /// command line.
+ pub fn worker(&self) -> Worker {
+ WorkerBuilder::new(self.grep())
+ .after_context(self.after_context)
+ .before_context(self.before_context)
.count(self.count)
.files_with_matches(self.files_with_matches)
.eol(self.eol)
.line_number(self.line_number)
.invert_match(self.invert_match)
+ .mmap(self.mmap)
.quiet(self.quiet)
.text(self.text)
+ .build()
}
/// Returns the number of worker search threads that should be used.
@@ -722,7 +688,17 @@ impl Args {
}
/// Create a new recursive directory iterator over the paths in argv.
- pub fn walker(&self) -> Walk {
+ pub fn walker(&self) -> ignore::Walk {
+ self.walker_builder().build()
+ }
+
+ /// Create a new parallel recursive directory iterator over the paths
+ /// in argv.
+ pub fn walker_parallel(&self) -> ignore::WalkParallel {
+ self.walker_builder().build_parallel()
+ }
+
+ fn walker_builder(&self) -> ignore::WalkBuilder {
let paths = self.paths();
let mut wd = ignore::WalkBuilder::new(&paths[0]);
for path in &paths[1..] {
@@ -744,7 +720,8 @@ impl Args {
wd.git_exclude(!self.no_ignore && !self.no_ignore_vcs);
wd.ignore(!self.no_ignore);
wd.parents(!self.no_ignore_parent);
- Walk(wd.build())
+ wd.threads(self.threads());
+ wd
}
}
@@ -761,34 +738,6 @@ fn version() -> String {
}
}
-/// A simple wrapper around the ignore::Walk iterator. This will
-/// automatically emit error messages to stderr and will skip directories.
-pub struct Walk(ignore::Walk);
-
-impl Iterator for Walk {
- type Item = ignore::DirEntry;
-
- fn next(&mut self) -> Option<ignore::DirEntry> {
- while let Some(result) = self.0.next() {
- match result {
- Ok(dent) => {
- if let Some(err) = dent.error() {
- eprintln!("{}", err);
- }
- if dent.file_type().map_or(false, |x| x.is_dir()) {
- continue;
- }
- return Some(dent);
- }
- Err(err) => {
- eprintln!("{}", err);
- }
- }
- }
- None
- }
-}
-
/// A single state in the state machine used by `unescape`.
#[derive(Clone, Copy, Eq, PartialEq)]
enum State {