From afc53f5a38f9963e0a95491acf5e74817caa7366 Mon Sep 17 00:00:00 2001 From: Jasper Ras Date: Mon, 23 Feb 2026 21:11:25 +0100 Subject: [PATCH] finished ch12 io project --- ch12-io-project/minigrep/Cargo.lock | 7 ++++ ch12-io-project/minigrep/Cargo.toml | 6 +++ ch12-io-project/minigrep/output.txt | 1 + ch12-io-project/minigrep/poem.txt | 9 +++++ ch12-io-project/minigrep/src/lib.rs | 55 ++++++++++++++++++++++++++++ ch12-io-project/minigrep/src/main.rs | 55 ++++++++++++++++++++++++++++ 6 files changed, 133 insertions(+) create mode 100644 ch12-io-project/minigrep/Cargo.lock create mode 100644 ch12-io-project/minigrep/Cargo.toml create mode 100644 ch12-io-project/minigrep/output.txt create mode 100644 ch12-io-project/minigrep/poem.txt create mode 100644 ch12-io-project/minigrep/src/lib.rs create mode 100644 ch12-io-project/minigrep/src/main.rs diff --git a/ch12-io-project/minigrep/Cargo.lock b/ch12-io-project/minigrep/Cargo.lock new file mode 100644 index 0000000..1511dd5 --- /dev/null +++ b/ch12-io-project/minigrep/Cargo.lock @@ -0,0 +1,7 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 4 + +[[package]] +name = "minigrep" +version = "0.1.0" diff --git a/ch12-io-project/minigrep/Cargo.toml b/ch12-io-project/minigrep/Cargo.toml new file mode 100644 index 0000000..b8302ae --- /dev/null +++ b/ch12-io-project/minigrep/Cargo.toml @@ -0,0 +1,6 @@ +[package] +name = "minigrep" +version = "0.1.0" +edition = "2024" + +[dependencies] diff --git a/ch12-io-project/minigrep/output.txt b/ch12-io-project/minigrep/output.txt new file mode 100644 index 0000000..5b34572 --- /dev/null +++ b/ch12-io-project/minigrep/output.txt @@ -0,0 +1 @@ +To tell your name the livelong day diff --git a/ch12-io-project/minigrep/poem.txt b/ch12-io-project/minigrep/poem.txt new file mode 100644 index 0000000..8707527 --- /dev/null +++ b/ch12-io-project/minigrep/poem.txt @@ -0,0 +1,9 @@ +I'm nobody! Who are you? +Are you nobody, too? +Then there's a pair of us - don't tell! +They'd banish us, you know. + +How dreary to be somebody! +How public, like a frog +To tell your name the livelong day +To an admiring bog! diff --git a/ch12-io-project/minigrep/src/lib.rs b/ch12-io-project/minigrep/src/lib.rs new file mode 100644 index 0000000..f43d5c7 --- /dev/null +++ b/ch12-io-project/minigrep/src/lib.rs @@ -0,0 +1,55 @@ +pub fn search<'a>(query: &str, contents: &'a str) -> Vec<&'a str> { + let mut results = Vec::new(); + + for line in contents.lines() { + if line.contains(query) { + results.push(line); + } + } + + results +} + +pub fn search_case_insensitive<'a>(query: &str, contents: &'a str) -> Vec<&'a str> { + let mut results = Vec::new(); + let query = query.to_lowercase(); + + for line in contents.lines() { + if line.to_lowercase().contains(&query) { + results.push(line); + } + } + + results +} + +#[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!(vec!["safe, fast, productive."], search(query, contents)); + } + + #[test] + fn case_insensitive() { + let query = "rUsT"; + let contents = "\ +Rust: +safe, fast, productive. +Pick three. +Trust me. + "; + + assert_eq!(vec!["Rust:", "Trust me."], search_case_insensitive(query, contents)); + } +} diff --git a/ch12-io-project/minigrep/src/main.rs b/ch12-io-project/minigrep/src/main.rs new file mode 100644 index 0000000..cade4a3 --- /dev/null +++ b/ch12-io-project/minigrep/src/main.rs @@ -0,0 +1,55 @@ +use std::env; +use std::fs; +use std::process; +use std::error::Error; + +use minigrep::{search, search_case_insensitive}; + +fn main() { + let args: Vec = env::args().collect(); + let config = Config::build(&args).unwrap_or_else(|err| { + eprintln!("problem parsing arguments: {}", err); + process::exit(1); + }); + + if let Err(e) = run(config) { + eprintln!("problem running program: {}", e); + process::exit(1); + } +} + +struct Config { + query: String, + file_path: String, + ignore_case: bool, +} + +impl Config { + fn build(args: &[String]) -> Result { + if args.len() < 3 { + return Err("not enough arguments"); + } + + let query = args[1].clone(); + let file_path = args[2].clone(); + let ignore_case = env::var("IGNORE_CASE").is_ok(); + + Ok(Config{ query, file_path, ignore_case }) + } +} + +fn run(config: Config) -> Result<(), Box> { + let contents = fs::read_to_string(config.file_path)?; + + let results = if config.ignore_case { + search_case_insensitive(&config.query, &contents) + } else { + search(&config.query, &contents) + }; + + for line in results { + println!("{line}"); + } + + Ok(()) +}