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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
use std::error::Error;
use std::fs;
use std::env;

pub fn run(config: Config) -> Result<(), Box<dyn Error>> {
    let contents = fs::read_to_string(config.filename)?;
    // println!("with text:\n{}", contents);

    let results = if config.case_sensitive {
        search(&config.query, &contents)
    } else {
        search_case_insensitive(&config.query, &contents)
    };

    for line in results {
        println!("Has found in line {}: {}", line.1, line.0);
    }
    Ok(())
}

// default search method
fn search<'a>(query: &str, contents: &'a str) -> Vec<(&'a str, i32)> {
    // zero-cost abstraction
    contents.lines().zip(1..).filter(|(x, _)| {
        x.contains(&query)
    }).collect()

    // to be improved
    // let mut i = 0;
    // contents.lines().filter(|x| x.contains(query)).map(|x| {
    //     i += 1;
    //     (x, i)
    // }).collect()

    // let mut results = Vec::new();
    // let mut i = 0;
    // for line in contents.lines() {
    //     i += 1;
    //     if line.contains(query) {
    //         results.push((line, i));
    //     }
    // }
    // results
}

// case_insensitive search method
fn search_case_insensitive<'a>(query: &str, contents: &'a str) -> Vec<(&'a str, i32)> {
    contents.lines().zip(1..).filter(|(x, _)| {
        x.to_lowercase().contains(&query.to_lowercase())
    }).collect()

    // contents.lines().enumerate().filter(|(_, x)| {
    //     x.to_lowercase().contains(&query.to_lowercase())
    // }).map(|(i , x)|
    //     (x , i as i32 + 1)
    // ).collect()

    // let mut results = Vec::new();
    // let query = query.to_lowercase();
    // let mut i = 0;
    // for line in contents.lines() {
    //     i += 1;
    //     if line.to_lowercase().contains(&query) {
    //         results.push((line, i));
    //     }
    // }
    // results

    // let mut i = 0;
    // contents.lines().filter_map(|x| {
    //     i += 1;
    //     if x.to_lowercase().contains(&query.to_lowercase()) {
    //         Some((x, i))
    //     } else {
    //         None
    //     }
    // }).collect()
}

pub struct Config {
    pub query: String,
    pub filename: String,
    pub case_sensitive: bool,
    pub case_skip_space: bool,
}

impl Config {
    pub fn new(mut args: env::Args) -> Result<Config, &'static str> {
        args.next();
        let query = match args.next() {
            Some(arg) => arg,
            None => return Err("Didn't get a query string!"),
        };
        let filename = match args.next() {
            Some(arg) => arg,
            None => return Err("Didn't get a file name!")
        };

        // match the environment variables
        let case_sensitive = match env::var("CASE_INSENSITIVE") {
            Ok(string) => {
                if string == "0" {
                    true
                } else {
                    false
                }
            }
            Err(_) => true,
        };
        let case_skip_space = match env::var("CASE_SKIP_SPACE") {
            Ok(string) => {
                if string == "0" {
                    false
                } else {
                    true
                }
            }
            Err(_) => false,
        };
        Ok(Config {
            query,
            filename,
            case_sensitive,
            case_skip_space,
        })
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn case_sensitive() {
        let query = "duct";
        let contents = "
Rust: 
safe, fast, productive.
Pick three.
Duct tape.";
        assert_eq!("safe, fast, productive.", search(query, contents)[0].0)
    }

    #[test]
    fn case_insensitive() {
        let query = "RuSt";
        let contents = "\
Rust:
safe, fast, productive
Trust me.";
        assert_eq!(
            vec![("Rust:", 1), ("Trust me.", 3)],
            search_case_insensitive(query, contents),
        );
    }
}