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
use std::str::FromStr;
use crate::types::date::decode_date_buf;
use crate::types::time::decode_time;
use crate::types::{Decode, Encode};
use crate::value::{MySqlValue, MySqlValueFormat};
use rbdc::timestamp::Timestamp;
use rbdc::Error;
impl Encode for Timestamp {
fn encode(self, buf: &mut Vec<u8>) -> Result<usize, Error> {
let datetime = fastdate::DateTime::from_timestamp_millis(self.0 as i64);
let size = date_time_size_hint(datetime.hour, datetime.min, datetime.sec, datetime.micro);
buf.push(size as u8);
let date = fastdate::Date {
day: datetime.day,
mon: datetime.mon,
year: datetime.year,
};
let size_date = date.encode(buf)?;
buf.remove(buf.len() - 1 - size_date);
let mut size_time = 0;
if (size + size_date) > 4 {
let time = fastdate::Time {
micro: datetime.micro,
sec: datetime.sec,
min: datetime.min,
hour: datetime.hour,
};
size_time = time.encode(buf)?;
buf.remove(buf.len() - 1 - size_time);
}
Ok(size + size_date + size_time)
}
}
impl Decode for Timestamp {
fn decode(value: MySqlValue) -> Result<Self, Error> {
Ok(match value.format() {
MySqlValueFormat::Text => Self(
fastdate::DateTime::from_str(value.as_str()?)
.unwrap()
.unix_timestamp_millis() as u64,
),
MySqlValueFormat::Binary => {
let buf = value.as_bytes()?;
let len = buf[0];
let date = decode_date_buf(&buf[1..])?;
let time = if len > 4 {
decode_time(len - 4, &buf[5..])
} else {
fastdate::Time {
micro: 0,
sec: 0,
min: 0,
hour: 0,
}
};
Self(
fastdate::DateTime {
micro: time.micro,
sec: time.sec,
min: time.min,
hour: time.hour,
day: date.day,
mon: date.mon,
year: date.year,
}
.unix_timestamp_millis() as u64,
)
}
})
}
}
fn date_time_size_hint(hour: u8, min: u8, sec: u8, nano: u32) -> usize {
match (hour, min, sec, nano) {
(0, 0, 0, 0) => 4,
(_, _, _, 0) => 7,
(_, _, _, _) => 11,
}
}