diff --git a/.idea/misc.xml b/.idea/misc.xml
index fee09bd..89e54d4 100644
--- a/.idea/misc.xml
+++ b/.idea/misc.xml
@@ -20,6 +20,7 @@
+
diff --git a/.idea/rust.iml b/.idea/rust.iml
index fca239b..eb9d073 100644
--- a/.idea/rust.iml
+++ b/.idea/rust.iml
@@ -78,6 +78,10 @@
+
+
+
+
@@ -89,6 +93,7 @@
+
diff --git a/.idea/workspace.xml b/.idea/workspace.xml
index fed1dab..54a3b77 100644
--- a/.idea/workspace.xml
+++ b/.idea/workspace.xml
@@ -14,45 +14,29 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+
-
-
+
+
-
+
-
-
+
+
+
+
+
+
+
+
+
+
+
@@ -110,6 +94,9 @@
+
+
+
@@ -143,20 +130,13 @@
-
+
-
-
-
-
-
-
-
-
-
+
+
@@ -189,7 +169,7 @@
-
+
@@ -327,8 +307,8 @@
-
-
+
+
@@ -344,7 +324,7 @@
-
+
@@ -589,9 +569,23 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
-
+
@@ -612,17 +606,24 @@
-
+
-
-
+
+
-
+
-
-
+
+
+
+
+
+
+
+
+
diff --git a/minigrep/Cargo.toml b/minigrep/Cargo.toml
new file mode 100644
index 0000000..de54896
--- /dev/null
+++ b/minigrep/Cargo.toml
@@ -0,0 +1,7 @@
+[package]
+name = "minigrep"
+version = "0.1.0"
+authors = ["Timothy Warren "]
+edition = "2018"
+
+[dependencies]
diff --git a/minigrep/poem.txt b/minigrep/poem.txt
new file mode 100644
index 0000000..8707527
--- /dev/null
+++ b/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/minigrep/src/lib.rs b/minigrep/src/lib.rs
new file mode 100644
index 0000000..64ed7a9
--- /dev/null
+++ b/minigrep/src/lib.rs
@@ -0,0 +1,101 @@
+use std::env;
+use std::error::Error;
+use std::fs;
+
+pub struct Config {
+ pub query: String,
+ pub filename: String,
+ pub case_sensitive: bool,
+}
+
+impl Config {
+ pub fn new(args: &[String]) -> Result {
+ if args.len() < 3 {
+ return Err("not enough arguments");
+ }
+
+ let query = args[1].clone();
+ let filename = args[2].clone();
+
+ let case_sensitive = env::var("CASE_INSENSITIVE").is_err();
+
+ Ok(Config { query, filename, case_sensitive })
+ }
+}
+
+pub fn run(config: Config) -> Result<(), Box> {
+ // The ? operator passes the error back to the caller.
+ 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 line in results {
+ println!("{}", line);
+ }
+
+ Ok(())
+}
+
+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
+}
+
+fn search_case_insensitive<'a>(query: &str, contents: &'a str) -> Vec<&'a str> {
+ let query = query.to_lowercase();
+ let mut results = Vec::new();
+
+ 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/minigrep/src/main.rs b/minigrep/src/main.rs
new file mode 100644
index 0000000..8302095
--- /dev/null
+++ b/minigrep/src/main.rs
@@ -0,0 +1,20 @@
+use std::env;
+use std::process;
+
+use minigrep;
+use minigrep::Config;
+
+fn main() {
+ // collect turns an iterator into a collection
+ let args: Vec = env::args().collect();
+
+ let config = Config::new(&args).unwrap_or_else(|err| {
+ eprintln!("Problem parsing arguments: {}", err);
+ process::exit(1);
+ });
+
+ if let Err(e) = minigrep::run(config) {
+ eprintln!("Application error: {}", e);
+ process::exit(1);
+ }
+}