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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
pub mod read;
use read::{read_byte, read_int, read_string};
use std::{fs, io::Result, mem::transmute};
#[repr(i8)]
#[derive(Debug)]
pub enum Gamemode {
Standart = 0,
Taiko = 1,
CatchTheBeat = 2,
Mania = 3,
}
impl Gamemode {
fn from_byte(b: u8) -> Gamemode {
unsafe { transmute(b) }
}
}
impl Default for Gamemode {
fn default() -> Self {
Gamemode::Standart
}
}
#[allow(dead_code)]
#[derive(Debug, Default)]
pub struct Replay {
gamemode: Gamemode,
version: u32,
beatmap_md5: String,
username: String,
replay_md5: String,
n300: u16,
n100: u16,
n50: u16,
geki: u16,
katu: u16,
misses: u16,
score: u32,
combo: u16,
perfect: u8,
mods: u32,
life_bar: String,
time_stamp: usize,
replay_length: u32,
replay_data: Vec<u8>,
score_id: usize,
mod_info: Option<f64>,
raw: Vec<u8>,
}
impl Replay {
pub fn new() -> Self {
Replay::default()
}
pub fn read(&mut self, path: &str) -> Result<()> {
let mut content = fs::read(path)?;
let mut p = 0;
let p_ref = &mut p;
self.gamemode = Gamemode::from_byte(read_byte(p_ref, &content));
self.version = read_int!(u32, p_ref, &content);
self.beatmap_md5 =
read_string(p_ref, &mut content).unwrap_or("Can't read beatmap md5!".to_string());
self.username =
read_string(p_ref, &mut content).unwrap_or("Can't read username!".to_string());
self.replay_md5 =
read_string(p_ref, &mut content).unwrap_or("Can't read replay md5!".to_string());
self.n300 = read_int!(u16, p_ref, &content);
self.n100 = read_int!(u16, p_ref, &content);
self.n50 = read_int!(u16, p_ref, &content);
self.geki = read_int!(u16, p_ref, &content);
self.katu = read_int!(u16, p_ref, &content);
self.misses = read_int!(u16, p_ref, &content);
self.score = read_int!(u32, p_ref, &content);
self.combo = read_int!(u16, p_ref, &content);
self.perfect = read_byte(p_ref, &content);
self.mods = read_int!(u32, p_ref, &content);
self.life_bar = read_string(p_ref, &mut content).unwrap_or("".to_string());
self.time_stamp = read_int!(usize, p_ref, &content);
self.replay_length = read_int!(u32, p_ref, &content);
self.replay_data = content[*p_ref..(self.replay_length as usize)].to_vec();
*p_ref += self.replay_length as usize;
self.score_id = read_int!(usize, p_ref, &content);
if *p_ref != content.len() {
self.mod_info = Some(read_int!(usize, p_ref, &content) as f64);
}
self.raw = content;
Ok(())
}
}
#[cfg(test)]
mod tests {
use std::time::Instant;
use super::*;
#[test]
fn test_read_replay() {
let now = Instant::now();
let mut replay = Replay::new();
replay.read("./replay.osr").unwrap();
println!("{:?} - {:?}", replay.n300, now.elapsed());
}
}