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::datetime::DateTime;
use rbdc::Error;
use std::str::FromStr;
impl Encode for DateTime {
fn encode(self, buf: &mut Vec<u8>) -> Result<usize, Error> {
let datetime = self.0;
let datetime_size =
date_time_size_hint(datetime.hour, datetime.min, datetime.sec, datetime.nano);
buf.push(datetime_size as u8);
let date = fastdate::Date {
day: datetime.day,
mon: datetime.mon,
year: datetime.year,
};
let mut size = date.encode(buf)?;
buf.remove(buf.len() - (size + 1));
if datetime_size > 4 {
let time = fastdate::Time {
nano: datetime.nano,
sec: datetime.sec,
min: datetime.min,
hour: datetime.hour,
};
let size_time = time.encode(buf)?;
buf.remove(buf.len() - (size_time + 1));
size += size_time;
}
Ok(1 + size)
}
}
impl Decode for DateTime {
fn decode(value: MySqlValue) -> Result<Self, Error> {
Ok(match value.format() {
MySqlValueFormat::Text => Self(fastdate::DateTime::from_str(value.as_str()?).unwrap()),
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 {
nano: 0,
sec: 0,
min: 0,
hour: 0,
}
};
Self(fastdate::DateTime {
nano: time.nano,
sec: time.sec,
min: time.min,
hour: time.hour,
day: date.day,
mon: date.mon,
year: date.year,
})
}
})
}
}
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,
}
}