use crate::{
bytes_buffer::BytesBuffer,
dns::WireFormat,
lib::{Cow, Vec},
lib::Write,
Name,
};
use super::RR;
#[derive(Debug, PartialEq, Eq, Hash, Clone)]
pub struct NSEC<'a> {
pub next_name: Name<'a>,
pub type_bit_maps: Vec<NsecTypeBitMap<'a>>,
}
#[derive(Debug, PartialEq, Eq, Hash, Clone)]
pub struct NsecTypeBitMap<'a> {
pub window_block: u8,
pub bitmap: Cow<'a, [u8]>,
}
impl RR for NSEC<'_> {
const TYPE_CODE: u16 = 47;
}
impl<'a> WireFormat<'a> for NSEC<'a> {
const MINIMUM_LEN: usize = 0;
fn parse(data: &mut BytesBuffer<'a>) -> crate::Result<Self>
where
Self: Sized,
{
let next_name = Name::parse(data)?;
let mut type_bit_maps = Vec::new();
let mut prev_window_block = None;
while data.has_remaining() {
let window_block = data.get_u8()?;
if let Some(prev_window_block) = prev_window_block {
if window_block <= prev_window_block {
return Err(crate::SimpleDnsError::InvalidDnsPacket);
}
}
prev_window_block = Some(window_block);
let bitmap_length = data.get_u8()? as usize;
if bitmap_length > 32 {
return Err(crate::SimpleDnsError::InvalidDnsPacket);
}
let bitmap = data.get_slice(bitmap_length)?;
type_bit_maps.push(NsecTypeBitMap {
window_block,
bitmap: Cow::Borrowed(bitmap),
});
}
Ok(Self {
next_name,
type_bit_maps,
})
}
fn write_to<T: Write>(&self, out: &mut T) -> crate::Result<()> {
self.next_name.write_to(out)?;
let mut sorted = self.type_bit_maps.clone();
sorted.sort_by(|a, b| a.window_block.cmp(&b.window_block));
for record in sorted.iter() {
out.write_all(&[record.window_block])?;
out.write_all(&[record.bitmap.len() as u8])?;
out.write_all(&record.bitmap)?;
}
Ok(())
}
fn len(&self) -> usize {
self.next_name.len()
}
}
impl NSEC<'_> {
pub fn into_owned<'b>(self) -> NSEC<'b> {
let type_bit_maps = self
.type_bit_maps
.into_iter()
.map(|x| NsecTypeBitMap {
window_block: x.window_block,
bitmap: x.bitmap.into_owned().into(),
})
.collect();
NSEC {
next_name: self.next_name.into_owned(),
type_bit_maps,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::lib::vec;
#[test]
fn parse_and_write_nsec() {
let nsec = NSEC {
next_name: Name::new("host.example.com.").unwrap(),
type_bit_maps: vec![NsecTypeBitMap {
window_block: 0,
bitmap: vec![64, 1, 0, 0, 0, 1].into(),
}],
};
let mut data = Vec::new();
nsec.write_to(&mut data).unwrap();
let nsec = NSEC::parse(&mut data[..].into()).unwrap();
assert_eq!(nsec.next_name, Name::new("host.example.com.").unwrap());
assert_eq!(nsec.type_bit_maps.len(), 1);
assert_eq!(nsec.type_bit_maps[0].window_block, 0);
assert_eq!(nsec.type_bit_maps[0].bitmap, vec![64, 1, 0, 0, 0, 1]);
}
#[test]
#[cfg(feature = "std")]
fn parse_sample() -> Result<(), Box<dyn std::error::Error>> {
use crate::{rdata::RData, ResourceRecord};
let sample_file = std::fs::read("samples/zonefile/NSEC.sample")?;
let sample_rdata = match ResourceRecord::parse(&mut sample_file[..].into())?.rdata {
RData::NSEC(rdata) => rdata,
_ => unreachable!(),
};
assert_eq!(
sample_rdata.next_name,
Name::new("host.example.com.").unwrap()
);
assert_eq!(sample_rdata.type_bit_maps.len(), 1);
assert_eq!(sample_rdata.type_bit_maps[0].window_block, 0);
assert_eq!(
sample_rdata.type_bit_maps[0].bitmap,
vec![64, 1, 0, 0, 0, 1]
);
Ok(())
}
}