aboutsummaryrefslogtreecommitdiff
path: root/src/history.rs
blob: 32385435bbb1634f7e29ec0ed8ee9294b51356ad (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
86
87
88
89
/* ************************************************************************** */
/*                                                                            */
/*                                                            .               */
/*   history.rs                                              / \              */
/*                                                          /   \             */
/*   By: charles <charles.cabergs@gmail.com>               /o  o \            */
/*                                                        /  v    \           */
/*   Created: 2020/06/25 13:24:10 by charles             /    _    \          */
/*   Updated: 2020/06/25 13:24:12 by charles            '-----------'         */
/*                                                                            */
/* ************************************************************************** */

use std::time::Duration;
use std::str;
use std::num;
use std::fmt;

use csv;
use chrono;
use chrono::prelude::*;

use super::scramble::Scramble;


pub struct SolveTime(Duration);

impl str::FromStr for SolveTime {
    type Err = num::ParseIntError;
    // fmt: mm:ss.lll  (l = millisecond)
    fn from_str(s: &str) -> Result<Self, Self::Err> {
        let minute: u64 = s[..2].parse()?;
        let second: u64 = s[3..5].parse()?;
        let millis: u64 = s[6..].parse()?;
        Ok(SolveTime(Duration::from_secs(minute * 60_u64 + second) + Duration::from_millis(millis)))
    }
}

impl fmt::Display for SolveTime {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "{:0>2}:{:0>2}.{:0>3}",
               self.0.as_secs() / 60,
               self.0.as_secs() % 60,
               self.0.as_millis() % 1000)
    }
}

struct Entry {
    time: SolveTime,
    scramble: Scramble,
    date: chrono::DateTime<Utc>,
}

pub struct History(Vec<Entry>);

const VEC_START_SIZE: usize = 200;

impl History {
    pub fn from_csv(file_path: &str) -> History {
        let mut history = History(Vec::with_capacity(VEC_START_SIZE));
        let mut reader = csv::Reader::from_path(file_path).unwrap();
        for result in reader.records() {
            if let Ok(record) = result {
                history.0.push(Entry{
                    time:     record[0].parse::<SolveTime>().unwrap(),
                    scramble: record[1].parse::<Scramble>().unwrap(),
                    date:     record[2].parse::<chrono::DateTime<Utc>>().unwrap()
                })
            }
        }
        history
    }

    pub fn save_csv(&self, file_path: &str) {
        let mut writter = csv::Writer::from_path(file_path).unwrap();
        writter.write_record(&["time", "scramble", "date"]).unwrap();
        for entry in &self.0 {
            writter.write_record(&[
                entry.time.to_string(),
                entry.scramble.to_string(),
                entry.date.to_string()
            ]).unwrap();
        }
        writter.flush().unwrap();
    }

    pub fn summarize(&self, n: usize) -> Vec<String> {
        self.0.iter().skip(self.0.len() - n).map(|Entry{time, ..}| time.to_string()).collect()
    }
}