320x100
320x100

grep 만들기

cargo new minigrep
cd minigrep
// src/main.rs
use std::env; // 커맨드 라인 인수로 전달된 값들과 환경변수를 읽을 수 있기 위한 러스트 표준 라이브러리
use std::process;

use minigrep::Config;

fn main() {
    // env 라이브러리로부터 args 함수를 가져와 collect 함수를 실행하여 인수를 담은 벡터 생성
    // 유니코드를 포함하는 인수를 받을수도 있다면 args_os를 대신 사용할 것
    let args: Vec<String> = env::args().collect();
    
    // 커스터마이징된 에러처리 정의 (Results를 반환)
    let config = Config::build(&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);
    }
}
// src/lib.rs
use std::error::Error;
use std::fs;
use std::env;

// 테스트 모듈. search 함수가 가져야할 동작을 지정
#[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)
        );
    }
}

// 텍스트 안에 일치하는 문자열이 있는지 검색
// search 함수에 의해 반환된 데이터가 오래 남음을 표기
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 query = query.to_lowercase();
    let mut results = Vec::new();
    
    for line in contents.lines() {
        if line.to_lowercase().contains(&query) {
            results.push(line);
        }
    }
    results
}

pub struct Config {
    pub query: String;
    pub file_path: String,
    pub ignore_case: bool,
}

impl Config {
    pub fn build(args: &[String]) -> Result<Config, &'static str> {
        if args.len() < 3 {
            return Err("not enough arguments");
        }
        
        let query = args[1].clone();
        let file_path = args[2].clone();
        
        // IGNORE_CASE 환경변수가 설정되어 있으면 true, 아니면 false
        let ignore_case = env::var("IGNORE_CASE").is_ok();
        
        Ok(Config { 
            query, 
            file_path,
            ignore_case,
        })     
    }
}

// Box<dyn Error> Error 트레이트를 구현한 동적 타입을 반환 
pub fn run(config: Config) -> Result<(), Box<dyn Error>> {
    let contents = fs::read_to_string(config.file_path)?; // panic! 대신 에러 값을 반환
    
    let results = if config.ignore_case {
        search_case_insensitive(&config.query, &contents)
    } else {
        search(&config.query, &contents)
    };
    
    for line in results {
        println!("{line}");
    }
    
    Ok(())
}

 

 

300x250
728x90

'Programming > Rust' 카테고리의 다른 글

러스트 카고와 crates.io  (0) 2025.10.09
러스트 반복자와 클로저  (0) 2025.10.08
러스트 자동화 테스트  (0) 2025.09.11
러스트 - 제네릭 타입, 트레이트, 라이프타임  (1) 2025.09.08
러스트 에러 처리  (0) 2025.09.08