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
use std::fs;
use std::error::Error;

/// A struct that holds
/// Query - Search query
/// Filename - The name of the file
pub struct Config {
    pub query: String,
    pub filename: String,
    case_sensitive : bool,
}

impl Config {
    /// The function that parses the args and returns a Result
    /// of either, a Config type or a Err.
    /// Args - std::env::Args
    /// Returns - ```Result<Config, Err>```
    pub fn parse_config<'a>(mut args : std::env::Args) -> Result<Config, &'a str> {

        // Skipping the first arg
        args.next();
        // Checking for everyone after

        let query = match args.next() {
            Some(arg) => arg,
            None => return Err("Query missing"),
        };

        let filename = match args.next() {
            Some(arg) => arg,
            None => return Err("Filename missing")
        };

        let case = match args.next() {
            Some(x) => if x == "-case" {false} else {true},
            None => false,
        };
        Ok(Config {query, filename, case_sensitive:case})
    }
}

/// The run function that controls the search utlity
/// Also worries about case senstivity
pub fn run(config: Config) -> Result<(), Box<dyn Error>> {
    let contents = fs::read_to_string(&config.filename)?;

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

    for (i, line) in results.into_iter().enumerate() {
        println!("{}: {}",i+1, &line);
    }

    Ok(())
}


fn search<'a> (query: &str, contents :&'a str) -> Vec<&'a str> {
    // .lines() returns an iterator
    contents
        .lines()
        .filter(|line| line.contains(query))
        .collect()
}


fn search_case_insensitive<'a> (query: &str, contents :&'a str) -> Vec<&'a str> {
    
    // .lines() returns an iterator
    contents
        .lines()
        .filter(|line| line.to_lowercase().contains(&query.to_lowercase()))
        .collect()
}

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

    #[test]
    fn with_case() {
        let query = "Rust:";
        let contents = "\
    Rust:
    safe, fast, productive.
    Pick three.";

        assert_eq!(vec!["Rust:"], search(query, contents));
    }

    #[test]
    fn no_case() {
        let query = "RuSt";
        let contents = "\
    Rust:
    safe, fast, productive.
    Pick three.";

        assert_eq!(vec!["Rust:"], search_case_insensitive(query, contents));
    }

    #[test]
    fn run_test() {

        let var = Config {query: "frog".to_string(), filename : "poem.txt".to_string(), case_sensitive:true};

        assert!(run(var).is_ok())
    }

}