use std::io::Write;
use byteorder_slice::{BigEndian, LittleEndian};
use super::blocks::block_common::{Block, PcapNgBlock};
use super::blocks::interface_description::InterfaceDescriptionBlock;
use super::blocks::section_header::SectionHeaderBlock;
use super::{PcapNgState, RawBlock};
use crate::{Endianness, PcapError, PcapResult};
pub struct PcapNgWriter<W: Write> {
state: PcapNgState,
writer: W,
}
impl<W: Write> PcapNgWriter<W> {
pub fn new(writer: W) -> PcapResult<Self> {
Self::with_endianness(writer, Endianness::native())
}
pub fn with_endianness(writer: W, endianness: Endianness) -> PcapResult<Self> {
let section = SectionHeaderBlock { endianness, ..Default::default() };
Self::with_section_header(writer, section)
}
pub fn with_section_header(mut writer: W, section: SectionHeaderBlock<'_>) -> PcapResult<Self> {
let mut state = PcapNgState::default();
let endianness = section.endianness;
let block = section
.into_owned()
.into_block();
state.update_from_block(&block)?;
match endianness {
Endianness::Big => block.write_to::<BigEndian, _>(&state, &mut writer)?,
Endianness::Little => block.write_to::<LittleEndian, _>(&state, &mut writer)?,
};
Ok(Self { state, writer })
}
pub fn write_block(&mut self, block: &Block) -> PcapResult<usize> {
match block {
Block::InterfaceStatistics(blk) => {
if blk.interface_id as usize >= self.state.interfaces.len() {
return Err(PcapError::InvalidInterfaceId(blk.interface_id));
}
},
Block::EnhancedPacket(blk) => {
if blk.interface_id as usize >= self.state.interfaces.len() {
return Err(PcapError::InvalidInterfaceId(blk.interface_id));
}
},
_ => (),
}
self.state.update_from_block(block)?;
match self.state.section.endianness {
Endianness::Big => block.write_to::<BigEndian, _>(&self.state, &mut self.writer),
Endianness::Little => block.write_to::<LittleEndian, _>(&self.state, &mut self.writer),
}
}
pub fn write_pcapng_block<'a, B: PcapNgBlock<'a>>(&mut self, block: B) -> PcapResult<usize> {
self.write_block(&block.into_block())
}
pub fn write_raw_block(&mut self, block: &RawBlock) -> PcapResult<usize> {
match self.state.section.endianness {
Endianness::Big => {
let written = block.write_to::<BigEndian, _>(&mut self.writer)?;
self.state.update_from_raw_block::<BigEndian>(block)?;
Ok(written)
},
Endianness::Little => {
let written = block.write_to::<LittleEndian, _>(&mut self.writer)?;
self.state.update_from_raw_block::<LittleEndian>(block)?;
Ok(written)
}
}
}
pub fn into_inner(self) -> W {
self.writer
}
pub fn get_ref(&self) -> &W {
&self.writer
}
pub fn get_mut(&mut self) -> &mut W {
&mut self.writer
}
pub fn state(&self) -> &PcapNgState {
&self.state
}
pub fn section(&self) -> &SectionHeaderBlock<'static> {
&self.state.section
}
pub fn interfaces(&self) -> &[InterfaceDescriptionBlock<'static>] {
&self.state.interfaces
}
}