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
|
use crate::{model::deck::Entry, util::serialization};
use anyhow::{Result, Error};
use std::fs::File;
use std::io::{prelude::*, BufReader};
use std::fmt;
#[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: String) -> 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 translation = line[1..].trim().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)
}
|