summaryrefslogtreecommitdiffstats
path: root/bin/core/imag-grep/src/main.rs
blob: 223849c914b6bb8932464e1e2acaed54be636e99 (plain)
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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
//
// imag - the personal information management suite for the commandline
// Copyright (C) 2015-2018 Matthias Beyer <mail@beyermatthias.de> and contributors
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; version
// 2.1 of the License.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
//

#![deny(
    non_camel_case_types,
    non_snake_case,
    path_statements,
    trivial_numeric_casts,
    unstable_features,
    unused_allocation,
    unused_import_braces,
    unused_imports,
    unused_must_use,
    unused_mut,
    unused_qualifications,
    while_true,
)]

#[macro_use] extern crate log;
extern crate clap;
extern crate regex;

extern crate libimagstore;
#[macro_use] extern crate libimagrt;
extern crate libimagerror;

use std::io::Write;

use regex::Regex;

use libimagrt::runtime::Runtime;
use libimagrt::setup::generate_runtime_setup;
use libimagstore::store::Entry;
use libimagerror::trace::MapErrTrace;
use libimagerror::exit::ExitUnwrap;
use libimagerror::io::ToExitCode;

mod ui;

struct Options {
    files_with_matches: bool,
    count: bool,
}

fn main() {
    let version = make_imag_version!();
    let rt = generate_runtime_setup("imag-grep",
                                    &version,
                                    "grep through entries text",
                                    ui::build_ui);

    let opts = Options {
        files_with_matches    : rt.cli().is_present("files-with-matches"),
        count                 : rt.cli().is_present("count"),
    };

    let mut count : usize = 0;

    let pattern = rt
        .cli()
        .value_of("pattern")
        .map(Regex::new)
        .unwrap() // ensured by clap
        .unwrap_or_else(|e| {
            error!("Regex building error: {:?}", e);
            ::std::process::exit(1)
        });

    let overall_count = rt
        .store()
        .entries()
        .map_err_trace_exit_unwrap(1)
        .into_get_iter()
        .filter_map(|res| res.map_err_trace_exit_unwrap(1))
        .filter(|entry| pattern.is_match(entry.get_content()))
        .map(|entry| show(&rt, &entry, &pattern, &opts, &mut count))
        .count();

    if opts.count {
        let _ = writeln!(rt.stdout(), "{}", count).to_exit_code().unwrap_or_exit();
    } else if !opts.files_with_matches {
        let _ = writeln!(rt.stdout(), "Processed {} files, {} matches, {} nonmatches",
                 overall_count,
                 count,
                 overall_count - count)
            .to_exit_code()
            .unwrap_or_exit();
    }
}

fn show(rt: &Runtime, e: &Entry, re: &Regex, opts: &Options, count: &mut usize) {
    if opts.files_with_matches {
        let _ = writeln!(rt.stdout(), "{}", e.get_location()).to_exit_code().unwrap_or_exit();
    } else if opts.count {
        *count += 1;
    } else {
        let _ = writeln!(rt.stdout(), "{}:", e.get_location()).to_exit_code().unwrap_or_exit();
        for capture in re.captures_iter(e.get_content()) {
            for mtch in capture.iter() {
                if let Some(m) = mtch {
                    let _ = writeln!(rt.stdout(), " '{}'", m.as_str()).to_exit_code().unwrap_or_exit();
                }
            }
        }

        let _ = writeln!(rt.stdout(), "").to_exit_code().unwrap_or_exit();
        *count += 1;
    }
}