aboutsummaryrefslogtreecommitdiff
path: root/src/deck.rs
blob: 82566bd0c482d94cd78d2e4d31ea473249583005 (plain)
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
use crate::{model::entry::Entry, util::serialization};
use anyhow::{Error, Result};
use std::fmt;
use std::fs::File;
use std::io::{prelude::*, BufReader};
use std::path::Path;

#[derive(Debug, Clone)]
struct ParseError {
    line: usize,
    message: String,
}

impl fmt::Display for ParseError {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        write!(f, "{} (parsing line {})", self.message, self.line)
    }
}

impl std::error::Error for ParseError {
    fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
        None
    }
}

pub fn read(deck: &str) -> Result<Vec<Entry>> {
    let file = File::open(deck)?;
    let reader = BufReader::new(file);
    let mut entries: Vec<Entry> = Vec::new();

    for (index, line) in reader.lines().enumerate() {
        let line = line?;
        let line = line.trim();

        if !line.starts_with('#') && !line.is_empty() {
            if !line.starts_with('-') {
                return Err(Error::from(ParseError {
                    line: index + 1,
                    message: "an entry should starts with “-”.".to_string(),
                }));
            } else {
                let without_minus = line.split('-').skip(1).collect::<Vec<&str>>().join("-");
                let without_comment = without_minus.split('#').collect::<Vec<&str>>()[0].trim();
                let translation = without_comment.split(':').collect::<Vec<&str>>();
                if translation.len() != 2 {
                    return Err(Error::from(ParseError {
                        line: index + 1,
                        message: "an entry should contain two parts separated by “:”.".to_string(),
                    }));
                } else {
                    let t1 = translation[0].trim();
                    let t2 = translation[1].trim();
                    if t1.is_empty() || t2.is_empty() {
                        return Err(Error::from(ParseError {
                            line: index + 1,
                            message: "an entry should contain two parts separated by “:”."
                                .to_string(),
                        }));
                    } else {
                        entries.push(Entry {
                            part_1: serialization::line_to_words(&t1.to_string()),
                            part_2: serialization::line_to_words(&t2.to_string()),
                        })
                    }
                }
            }
        }
    }

    Ok(entries)
}

pub fn pp_from_path(path: &str) -> Option<String> {
    Some(capitalize(
        Path::new(&path).with_extension("").file_name()?.to_str()?,
    ))
}

fn capitalize(s: &str) -> String {
    let mut c = s.chars();
    match c.next() {
        None => String::new(),
        Some(f) => f.to_uppercase().collect::<String>() + c.as_str(),
    }
}