use anyhow::{anyhow, Result}; use base64::{decode, encode}; use std::io::{BufRead, BufReader}; use std::io::{Read, Write}; use quick_xml::events::{BytesEnd, BytesStart, Event}; use quick_xml::{Reader, Writer}; use crate::xml::*; //--------------------------------------- #[derive(Clone)] pub struct Superblock { pub uuid: String, pub block_size: u32, pub nr_cache_blocks: u32, pub policy: String, pub hint_width: u32, } #[derive(Clone)] pub struct Map { pub cblock: u32, pub oblock: u64, pub dirty: bool, } #[derive(Clone)] pub struct Hint { pub cblock: u32, pub data: Vec, } #[derive(Clone)] pub struct Discard { pub begin: u64, pub end: u64, } #[derive(Clone)] pub enum Visit { Continue, Stop, } pub trait MetadataVisitor { fn superblock_b(&mut self, sb: &Superblock) -> Result; fn superblock_e(&mut self) -> Result; fn mappings_b(&mut self) -> Result; fn mappings_e(&mut self) -> Result; fn mapping(&mut self, m: &Map) -> Result; fn hints_b(&mut self) -> Result; fn hints_e(&mut self) -> Result; fn hint(&mut self, h: &Hint) -> Result; fn discards_b(&mut self) -> Result; fn discards_e(&mut self) -> Result; fn discard(&mut self, d: &Discard) -> Result; fn eof(&mut self) -> Result; } pub struct XmlWriter { w: Writer, } impl XmlWriter { pub fn new(w: W) -> XmlWriter { XmlWriter { w: Writer::new_with_indent(w, 0x20, 2), } } } impl MetadataVisitor for XmlWriter { fn superblock_b(&mut self, sb: &Superblock) -> Result { let tag = b"superblock"; let mut elem = BytesStart::owned(tag.to_vec(), tag.len()); elem.push_attribute(mk_attr(b"uuid", sb.uuid.clone())); elem.push_attribute(mk_attr(b"block_size", sb.block_size)); elem.push_attribute(mk_attr(b"nr_cache_blocks", sb.nr_cache_blocks)); elem.push_attribute(mk_attr(b"policy", sb.policy.clone())); elem.push_attribute(mk_attr(b"hint_width", sb.hint_width)); self.w.write_event(Event::Start(elem))?; Ok(Visit::Continue) } fn superblock_e(&mut self) -> Result { self.w .write_event(Event::End(BytesEnd::borrowed(b"superblock")))?; Ok(Visit::Continue) } fn mappings_b(&mut self) -> Result { let tag = b"mappings"; let elem = BytesStart::owned(tag.to_vec(), tag.len()); self.w.write_event(Event::Start(elem))?; Ok(Visit::Continue) } fn mappings_e(&mut self) -> Result { self.w .write_event(Event::End(BytesEnd::borrowed(b"mappings")))?; Ok(Visit::Continue) } fn mapping(&mut self, m: &Map) -> Result { let tag = b"mapping"; let mut elem = BytesStart::owned(tag.to_vec(), tag.len()); elem.push_attribute(mk_attr(b"cache_block", m.cblock)); elem.push_attribute(mk_attr(b"origin_block", m.oblock)); elem.push_attribute(mk_attr(b"dirty", m.dirty)); self.w.write_event(Event::Empty(elem))?; Ok(Visit::Continue) } fn hints_b(&mut self) -> Result { let tag = b"hints"; let elem = BytesStart::owned(tag.to_vec(), tag.len()); self.w.write_event(Event::Start(elem))?; Ok(Visit::Continue) } fn hints_e(&mut self) -> Result { self.w .write_event(Event::End(BytesEnd::borrowed(b"hints")))?; Ok(Visit::Continue) } fn hint(&mut self, h: &Hint) -> Result { let tag = b"hint"; let mut elem = BytesStart::owned(tag.to_vec(), tag.len()); elem.push_attribute(mk_attr(b"cache_block", h.cblock)); elem.push_attribute(mk_attr(b"data", encode(&h.data[0..]))); self.w.write_event(Event::Empty(elem))?; Ok(Visit::Continue) } fn discards_b(&mut self) -> Result { let tag = b"discards"; let elem = BytesStart::owned(tag.to_vec(), tag.len()); self.w.write_event(Event::Start(elem))?; Ok(Visit::Continue) } fn discards_e(&mut self) -> Result { self.w .write_event(Event::End(BytesEnd::borrowed(b"discards")))?; Ok(Visit::Continue) } fn discard(&mut self, d: &Discard) -> Result { let tag = b"discard"; let mut elem = BytesStart::owned(tag.to_vec(), tag.len()); elem.push_attribute(mk_attr(b"dbegin", d.begin)); elem.push_attribute(mk_attr(b"dend", d.end)); self.w.write_event(Event::Empty(elem))?; Ok(Visit::Continue) } fn eof(&mut self) -> Result { Ok(Visit::Continue) } } //------------------------------------------ fn parse_superblock(e: &BytesStart) -> Result { let mut uuid: Option = None; let mut block_size: Option = None; let mut nr_cache_blocks: Option = None; let mut policy: Option = None; let mut hint_width: Option = None; for a in e.attributes() { let kv = a.unwrap(); match kv.key { b"uuid" => uuid = Some(string_val(&kv)), b"block_size" => block_size = Some(u32_val(&kv)?), b"nr_cache_blocks" => nr_cache_blocks = Some(u32_val(&kv)?), b"policy" => policy = Some(string_val(&kv)), b"hint_width" => hint_width = Some(u32_val(&kv)?), _ => return bad_attr("superblock", kv.key), } } let tag = "cache"; Ok(Superblock { uuid: check_attr(tag, "uuid", uuid)?, block_size: check_attr(tag, "block_size", block_size)?, nr_cache_blocks: check_attr(tag, "nr_cache_blocks", nr_cache_blocks)?, policy: check_attr(tag, "policy", policy)?, hint_width: check_attr(tag, "hint_width", hint_width)?, }) } fn parse_mapping(e: &BytesStart) -> Result { let mut cblock: Option = None; let mut oblock: Option = None; let mut dirty: Option = None; for a in e.attributes() { let kv = a.unwrap(); match kv.key { b"cache_block" => cblock = Some(u32_val(&kv)?), b"origin_block" => oblock = Some(u64_val(&kv)?), b"dirty" => dirty = Some(bool_val(&kv)?), _ => return bad_attr("mapping", kv.key), } } let tag = "mapping"; Ok(Map { cblock: check_attr(tag, "cache_block", cblock)?, oblock: check_attr(tag, "origin_block", oblock)?, dirty: check_attr(tag, "dirty", dirty)?, }) } fn parse_hint(e: &BytesStart) -> Result { let mut cblock: Option = None; let mut data: Option> = None; for a in e.attributes() { let kv = a.unwrap(); match kv.key { b"cache_block" => cblock = Some(u32_val(&kv)?), b"data" => data = Some(decode(bytes_val(&kv))?), _ => return bad_attr("mapping", kv.key), } } let tag = "hint"; Ok(Hint { cblock: check_attr(tag, "cache_block", cblock)?, data: check_attr(tag, "data", data)?, }) } fn handle_event(reader: &mut Reader, buf: &mut Vec, visitor: &mut M) -> Result where R: Read + BufRead, M: MetadataVisitor, { match reader.read_event(buf) { Ok(Event::Start(ref e)) => match e.name() { b"superblock" => visitor.superblock_b(&parse_superblock(e)?), b"mappings" => visitor.mappings_b(), b"hints" => visitor.hints_b(), _ => todo!(), }, Ok(Event::End(ref e)) => match e.name() { b"superblock" => visitor.superblock_e(), b"mappings" => visitor.mappings_e(), b"hints" => visitor.hints_e(), _ => todo!(), }, Ok(Event::Empty(ref e)) => match e.name() { b"mapping" => visitor.mapping(&parse_mapping(e)?), b"hint" => visitor.hint(&parse_hint(e)?), _ => todo!(), }, Ok(Event::Text(_)) => Ok(Visit::Continue), Ok(Event::Comment(_)) => Ok(Visit::Continue), Ok(Event::Eof) => { visitor.eof()?; Ok(Visit::Stop) } Ok(_) => todo!(), Err(e) => Err(anyhow!("{:?}", e)), } } pub fn read(input: R, visitor: &mut M) -> Result<()> where R: Read, M: MetadataVisitor, { let input = BufReader::new(input); let mut reader = Reader::from_reader(input); reader.trim_text(true); let mut buf = Vec::new(); while let Visit::Continue = handle_event(&mut reader, &mut buf, visitor)? {} Ok(()) }