evtx 0.1.1

A Fast (and safe) parser for the Windows XML Event Log (EVTX) format
Documentation
use crate::model::xml::XmlElement;
use core::borrow::Borrow;
use log::trace;

use std::io::Write;

use xml::common::XmlVersion;
use xml::{writer::XmlEvent, EmitterConfig, EventWriter};

use failure::{format_err, Error};

pub trait BinXMLOutput<'a, W: Write> {
    fn with_writer(target: W) -> Self;
    fn into_writer(self) -> Result<W, Error>;

    fn visit_end_of_stream(&mut self) -> ();
    fn visit_open_start_element(&mut self, open_start_element: &XmlElement<'a>) -> ();
    fn visit_close_element(&mut self) -> ();
    fn visit_characters(&mut self, value: &str) -> ();
    fn visit_cdata_section(&mut self) -> ();
    fn visit_entity_reference(&mut self) -> ();
    fn visit_processing_instruction_target(&mut self) -> ();
    fn visit_processing_instruction_data(&mut self) -> ();
    fn visit_start_of_stream(&mut self) -> ();
}

pub struct XMLOutput<W: Write> {
    writer: EventWriter<W>,
    eof_reached: bool,
}

/// Adapter between binxml XmlModel type and rust-xml output stream.
impl<'a, W: Write> BinXMLOutput<'a, W> for XMLOutput<W> {
    fn with_writer(target: W) -> Self {
        let writer = EmitterConfig::new()
            .line_separator("\r\n")
            .perform_indent(true)
            .normalize_empty_elements(false)
            .create_writer(target);

        XMLOutput {
            writer,
            eof_reached: false,
        }
    }

    fn into_writer(self) -> Result<W, Error> {
        if self.eof_reached {
            Ok(self.writer.into_inner())
        } else {
            Err(format_err!(
                "Tried to return writer before EOF marked, incomplete output."
            ))
        }
    }

    fn visit_end_of_stream(&mut self) {
        trace!("visit_end_of_stream");
        self.eof_reached = true
    }

    fn visit_open_start_element(&mut self, element: &XmlElement) {
        trace!("visit_open_start_element: {:?}", element);
        if self.eof_reached {
            return;
        }

        let mut event_builder = XmlEvent::start_element(element.name.borrow());

        for attr in element.attributes.iter() {
            event_builder = event_builder.attr(attr.name.borrow(), &attr.value.borrow());
        }

        self.writer.write(event_builder).unwrap();
    }

    fn visit_close_element(&mut self) {
        trace!("visit_close_element");
        if self.eof_reached {
            return;
        }

        self.writer.write(XmlEvent::end_element()).unwrap();
    }

    fn visit_characters(&mut self, value: &str) {
        trace!("visit_chars");
        if self.eof_reached {
            return;
        }
        self.writer.write(XmlEvent::characters(value)).unwrap();
    }

    fn visit_cdata_section(&mut self) {
        unimplemented!("visit_cdata_section");
    }

    fn visit_entity_reference(&mut self) {
        unimplemented!("visit_entity_reference");
    }

    fn visit_processing_instruction_target(&mut self) {
        unimplemented!("visit_processing_instruction_target");
    }

    fn visit_processing_instruction_data(&mut self) {
        unimplemented!("visit_processing_instruction_data");
    }

    fn visit_start_of_stream(&mut self) {
        trace!("visit_start_of_stream");
        if self.eof_reached {
            return;
        }
        self.writer
            .write(XmlEvent::StartDocument {
                version: XmlVersion::Version10,
                encoding: None,
                standalone: None,
            })
            .unwrap();
    }
}