1
0
mirror of https://github.com/jht5945/finding.git synced 2025-12-27 21:20:05 +08:00
Files
finding/src/main.rs

223 lines
6.8 KiB
Rust

extern crate argparse;
extern crate term;
extern crate term_size;
extern crate rust_util;
use std::{
fs::File,
path::Path,
io::prelude::*,
};
use argparse::{ArgumentParser, StoreTrue, Store};
use rust_util::*;
const VERSION: &str = "0.1";
fn print_version() {
print!(r#"finding {}
Copyright (C) 2019 Hatter Jiang.
License MIT <https://opensource.org/licenses/MIT>
Written by Hatter Jiang
"#, VERSION);
}
fn get_term_width() -> Option<usize> {
match term_size::dimensions() {
None => None,
Some((w, _h)) => Some(w),
}
}
fn get_term_width_message(message: &str, left: usize) -> String {
match get_term_width() {
None => message.to_string(),
Some(w) => {
let len = message.len();
if w > len {
return message.to_string();
}
let mut s = String::new();
s.push_str(&message[0..w-10-5-left]);
s.push_str("[...]");
s.push_str(&message[len-10..]);
s
},
}
}
fn find_huge_files(huge_file_size: &String, dir_path: &Path) {
let huge_file_size_bytes = match parse_size(&huge_file_size) {
Err(err) => {
print_message(MessageType::ERROR, &format!("Parse size failed: {}", err));
return;
},
Ok(bytes) => bytes as u64,
};
walk_dir(&dir_path, &|_, _| (/* do not process error */), &|p| {
match p.metadata() {
Err(_) => (),
Ok(metadata) => {
let len = metadata.len();
if len >= huge_file_size_bytes {
match p.to_str() {
None => (),
Some(p_str) => {
print_lastline("");
print_message(MessageType::OK, &format!("{} [{}]", p_str, get_display_size(len as i64)));
},
}
}
},
}
}, &|p| {
match p.to_str() {
None => (),
Some(p_str) => print_lastline(&get_term_width_message(&format!("Scanning: {}", p_str), 10)),
}
true
}).unwrap_or(());
print_lastline("");
}
fn read_file_content(file: &Path, large_file_len: u64) -> XResult<String> {
if ! file.exists() {
return Err(new_box_error(&format!("File not exists: {:?}", file)));
}
if ! file.is_file() {
return Err(new_box_error(&format!("File is not file: {:?}", file)));
}
let file_len = file.metadata()?.len();
if file_len > large_file_len {
return Err(new_box_error(&format!("File too large: {:?}, len: {}", file, file_len)));
}
let mut f = File::open(file)?;
let mut content = String::new();
f.read_to_string(&mut content)?;
Ok(content)
}
#[derive(Debug)]
struct MatchLine {
line_number: usize,
line_string: String,
}
impl MatchLine {
fn new(l_no: usize, l_str: String) -> MatchLine {
MatchLine {
line_number: l_no,
line_string: l_str,
}
}
}
fn match_lines(tag: &str, content: &String, search_text: &String) {
let lines = content.lines();
let mut match_lines_vec = vec![];
let mut l_no = 0usize;
for ln in lines {
if ln.contains(search_text) {
match_lines_vec.push(MatchLine::new(l_no, ln.to_string()));
}
l_no += 1;
}
if match_lines_vec.len() > 0 {
print_lastline("");
print_message(MessageType::OK, &format!("Find in {}:", tag));
for i in 0..match_lines_vec.len() {
print!("{}: ", match_lines_vec[i].line_number + 1);
let ss: Vec<&str> = match_lines_vec[i].line_string.split(search_text).collect();
for j in 0..ss.len() {
print!("{}", ss[j]);
if j < ss.len() -1 {
print_color(Some(term::color::RED), true, search_text);
}
}
println!();
}
}
}
fn find_text_files(large_text_file_size: &String, dir_path: &Path, search_text: &String) {
if search_text.len() < 1 {
print_message(MessageType::ERROR, "Param search_text cannot be empty.");
return;
}
let large_text_file_size_bytes = match parse_size(&large_text_file_size) {
Err(err) => {
print_message(MessageType::ERROR, &format!("Parse size failed: {}", err));
return;
},
Ok(bytes) => bytes as u64,
};
walk_dir(&dir_path, &|_, _| (/* do not process error */), &|p| {
let p_str = match p.to_str() {
None => return,
Some(s) => s,
};
let file_content = match read_file_content(p, large_text_file_size_bytes) {
Err(_err) => {
// TODO ... print_message(MessageType::ERROR, &format!("Read file {} failed: {}", p_str, err));
return;
},
Ok(c) => c,
};
match_lines(p_str, &file_content, search_text);
}, &|p| {
match p.to_str() {
None => (),
Some(p_str) => {
if p_str.ends_with("/.git") {
return false;
}
print_lastline(&get_term_width_message(&format!("Scanning: {}", p_str), 10))
},
}
true
}).unwrap_or(());
print_lastline("");
}
fn main() {
let mut version = false;
let mut target = String::from("text");
let mut huge_file_size = String::from("100M");
let mut large_text_file_size = String::from("10M");
let mut dir = String::from(".");
let mut search_text = String::new();
{
let mut ap = ArgumentParser::new();
ap.set_description("finding - command line find tool.");
ap.refer(&mut target).add_option(&["-t", "--target"], Store, "Target, text, huge[file], default text");
ap.refer(&mut dir).add_option(&["-d", "--dir"], Store, "Target directory, default current dir(.)");
ap.refer(&mut huge_file_size).add_option(&["--huge-file"], Store, "Huge file size, default 100M");
ap.refer(&mut large_text_file_size).add_option(&["--large-text-file"], Store, "Large text file, default 10M");
ap.refer(&mut version).add_option(&["-v", "--version"], StoreTrue, "Print version");
ap.refer(&mut search_text).add_argument("SEARCH TEXT", Store, "Search text");
ap.parse_args_or_exit();
}
if version {
print_version();
return;
}
let dir_path = match get_absolute_path(&dir) {
None => {
print_message(MessageType::ERROR, &format!("Cannot find dir: {}", dir));
return;
},
Some(path) => path,
};
match target.as_str() {
"huge" | "hugefile" => find_huge_files(&huge_file_size, &dir_path),
"text" => find_text_files(&large_text_file_size, &dir_path, &search_text),
unknown => print_message(MessageType::ERROR, &format!("Unknown command: {}", unknown)),
}
}