use crate::io::Buf;
use crate::postgres::database::Postgres;
use crate::postgres::protocol::{TypeFormat, TypeId};
use byteorder::NetworkEndian;
#[derive(Debug)]
pub(crate) struct RowDescription {
pub(crate) fields: Box<[Field]>,
}
#[derive(Debug)]
pub(crate) struct Field {
pub(crate) name: Option<Box<str>>,
pub(crate) table_id: Option<u32>,
pub(crate) column_id: i16,
pub(crate) type_id: TypeId,
pub(crate) type_size: i16,
pub(crate) type_mod: i32,
pub(crate) type_format: TypeFormat,
}
impl RowDescription {
pub(crate) fn read(mut buf: &[u8]) -> crate::Result<Self> {
let cnt = buf.get_u16::<NetworkEndian>()? as usize;
let mut fields = Vec::with_capacity(cnt);
for _ in 0..cnt {
let name = buf.get_str_nul()?;
let name = if name == "?column?" {
None
} else {
Some(name.to_owned().into_boxed_str())
};
let table_id = buf.get_u32::<NetworkEndian>()?;
fields.push(Field {
name,
table_id: if table_id > 0 { Some(table_id) } else { None },
column_id: buf.get_i16::<NetworkEndian>()?,
type_id: TypeId(buf.get_u32::<NetworkEndian>()?),
type_size: buf.get_i16::<NetworkEndian>()?,
type_mod: buf.get_i32::<NetworkEndian>()?,
type_format: buf.get_i16::<NetworkEndian>()?.into(),
});
}
Ok(Self {
fields: fields.into_boxed_slice(),
})
}
}
#[cfg(test)]
mod test {
use super::RowDescription;
#[test]
fn it_reads_row_description() {
#[rustfmt::skip]
let buf = bytes! {
0_u8, 2_u8,
b"user_id\0", 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8,
b"number_of_pages\0", 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 5_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8, 0_u8 };
let desc = RowDescription::read(&buf).unwrap();
assert_eq!(desc.fields.len(), 2);
assert_eq!(desc.fields[0].type_id.0, 0x0000_0000);
assert_eq!(desc.fields[1].type_id.0, 0x0000_0500);
}
#[test]
fn it_reads_empty_row_description() {
let buf = b"\x00\x00";
let desc = RowDescription::read(buf).unwrap();
assert_eq!(desc.fields.len(), 0);
}
}