use bytes;
use bytes::compress::*;
use bytes::varnum::*;
use escaped_wtf8;
use io::*;
use multipart::*;
use {CompressionTarget, TokenWriterError};
use binjs_shared::{FieldName, InterfaceName, SharedString};
use std;
use std::cell::RefCell;
use std::collections::{HashMap, HashSet};
use std::fmt::{Debug, Display, Formatter};
use std::hash::Hash;
use std::io::Write;
use std::ops::{Add, AddAssign, Deref};
use std::rc::Rc;
use vec_map;
use vec_map::*;
#[derive(Clone, Debug)]
pub struct Targets {
    pub grammar_table: CompressionTarget,
    pub strings_table: CompressionTarget,
    pub tree: CompressionTarget,
}
impl Targets {
    pub fn reset(&mut self) {
        self.grammar_table.reset();
        self.strings_table.reset();
        self.tree.reset();
    }
}
trait Serializable {
    
    fn write<W: Write>(&self, &mut W) -> Result<usize, std::io::Error>;
    
    fn write_with_compression<W: Write>(
        &self,
        out: &mut W,
        compression: &Compression,
    ) -> Result<CompressionResult, std::io::Error> {
        let mut uncompressed = Vec::with_capacity(2048);
        self.write(&mut uncompressed)?;
        compression.compress(&uncompressed, out)
    }
}
impl Serializable for Vec<u8> {
    fn write<W: Write>(&self, out: &mut W) -> Result<usize, std::io::Error> {
        out.write_all(&self)?;
        Ok(self.len())
    }
}
impl Serializable for SharedString {
    fn write<W: Write>(&self, out: &mut W) -> Result<usize, std::io::Error> {
        let mut total = 0;
        let bytes = self.deref().as_bytes();
        let unescaped = escaped_wtf8::unescape(bytes);
        total += out.write_varnum(unescaped.len() as u32)?;
        out.write_all(unescaped.deref())?;
        total += self.len();
        Ok(total)
    }
}
impl Serializable for Option<SharedString> {
    fn write<W: Write>(&self, out: &mut W) -> Result<usize, std::io::Error> {
        const EMPTY_STRING: [u8; 2] = [255, 0];
        let total = match *self {
            None => {
                let mut total = 0;
                total += out.write_varnum(EMPTY_STRING.len() as u32)?;
                out.write_all(&EMPTY_STRING)?;
                total += EMPTY_STRING.len();
                total
            }
            Some(ref data) => data.write(out)?,
        };
        Ok(total)
    }
}
struct TableEntry<T>
where
    T: Clone + std::fmt::Debug + Ord,
{
    
    
    instances: RefCell<u32>,
    
    data: T,
    
    index: TableIndex<T>,
}
impl<T> TableEntry<T>
where
    T: Clone + std::fmt::Debug + Ord,
{
    fn new(data: T) -> Self {
        TableEntry {
            instances: RefCell::new(1),
            index: TableIndex::new(&format!("{:?}", data)),
            data,
        }
    }
}
impl<T> PartialEq for TableEntry<T>
where
    T: Clone + std::fmt::Debug + Ord,
{
    fn eq(&self, other: &Self) -> bool {
        self.cmp(other) == std::cmp::Ordering::Equal
    }
}
impl<T> Eq for TableEntry<T> where T: Clone + std::fmt::Debug + Ord {}
impl<T> PartialOrd for TableEntry<T>
where
    T: Clone + std::fmt::Debug + Ord,
{
    fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
        Some(self.cmp(other))
    }
}
impl<T> Ord for TableEntry<T>
where
    T: Clone + std::fmt::Debug + Ord,
{
    fn cmp(&self, other: &Self) -> std::cmp::Ordering {
        u32::cmp(&*other.instances.borrow(), &*self.instances.borrow())
            .then_with(|| self.data.cmp(&other.data))
    }
}
struct WriterTable<Entry>
where
    Entry: Eq + Hash + Clone + Serializable + FormatInTable + Debug + Ord,
{
    map: HashMap<Entry, TableEntry<Entry>>,
}
impl<Entry> WriterTable<Entry>
where
    Entry: Eq + Hash + Clone + Serializable + FormatInTable + Debug + Ord,
{
    pub fn new() -> Self {
        WriterTable {
            map: HashMap::new(),
        }
    }
}
impl<Entry> WriterTable<Entry>
where
    Entry: Eq + Hash + Clone + Serializable + FormatInTable + Debug + Ord,
{
    
    
    
    fn get(&self, kind: &Entry) -> Option<&TableEntry<Entry>> {
        self.map.get(kind).map(|entry| {
            
            let mut borrow = entry.instances.borrow_mut();
            *borrow += 1;
            entry
        })
    }
    
    
    
    fn insert(&mut self, entry: Entry) -> TableIndex<Entry> {
        use std::collections::hash_map::Entry::*;
        match self.map.entry(entry) {
            Occupied(slot) => {
                *slot.get().instances.borrow_mut() += 1;
                slot.get().index.clone()
            }
            Vacant(slot) => {
                let entry = TableEntry::new(slot.key().clone());
                let index = entry.index.clone();
                slot.insert(entry);
                index
            }
        }
    }
}
impl<Entry> Serializable for WriterTable<Entry>
where
    Entry: Eq + Hash + Clone + Serializable + FormatInTable + Debug + Ord,
{
    fn write<W: Write>(&self, out: &mut W) -> Result<usize, std::io::Error> {
        let mut total = 0;
        
        let mut contents: Vec<_> = self.map.values().collect();
        contents.sort_unstable();
        
        for i in 0..contents.len() {
            let mut borrow = contents[i].index.index.borrow_mut();
            *borrow = Some(i as u32);
        }
        
        let mut serialized = Vec::with_capacity(contents.len());
        for entry in contents.drain(..) {
            let mut bytes = Vec::with_capacity(256);
            entry.data.write(&mut bytes)?;
            serialized.push(bytes);
        }
        
        total += out.write_varnum(serialized.len() as u32)?;
        if Entry::HAS_LENGTH_INDEX {
            
            for entry in &serialized {
                total += out.write_varnum(entry.len() as u32)?;
            }
        }
        
        for entry in &serialized {
            out.write_all(&entry)?;
            total += entry.len()
        }
        
        for entry in self.map.values() {
            debug_assert!(entry.index.index.borrow().is_some())
        }
        Ok(total)
    }
}
#[derive(PartialOrd, Ord, PartialEq, Eq, Clone, Hash, Debug)] 
pub struct NodeDescription {
    kind: InterfaceName,
}
impl Serializable for NodeDescription {
    fn write<W: Write>(&self, out: &mut W) -> Result<usize, std::io::Error> {
        let mut total = 0;
        total += self.kind.as_shared_string().write(out)?;
        Ok(total)
    }
}
impl FormatInTable for NodeDescription {
    const HAS_LENGTH_INDEX: bool = false;
}
enum UnresolvedTreeNode {
    
    UnresolvedStringIndex(TableIndex<Option<SharedString>>),
    
    UnresolvedNodeIndex(TableIndex<NodeDescription>),
    
    UnresolvedOffset(Option<Box<UnresolvedTree>>),
    Tuple(Vec<Rc<UnresolvedTree>>),
    Encoded(Vec<u8>),
}
struct UnresolvedTree {
    nature: Nature,
    data: UnresolvedTreeNode,
}
enum ResolvedTree {
    Tuple(Vec<ResolvedTree>),
    Encoded(Vec<u8>),
}
impl ResolvedTree {
    fn write<W: Write>(&self, out: &mut W) -> Result<usize, std::io::Error> {
        match *self {
            ResolvedTree::Encoded(ref buf) => {
                out.write_all(&*buf)?;
                Ok(buf.len())
            }
            ResolvedTree::Tuple(ref items) => {
                let mut total = 0;
                for item in items {
                    total += item.write(out)?;
                }
                Ok(total)
            }
        }
    }
}
impl UnresolvedTree {
    fn resolve(self, stats: &mut Statistics) -> (u32, ResolvedTree) {
        use self::UnresolvedTreeNode::*;
        let (total_bytes, own_bytes, tree) = match self.data {
            UnresolvedStringIndex(index) => {
                let index = index
                    .index()
                    .expect("String index should have been resolved by now.");
                let mut buf = Vec::with_capacity(4);
                let byte_len: usize = buf.write_varnum(index).unwrap(); 
                (byte_len as u32, byte_len as u32, ResolvedTree::Encoded(buf))
            }
            UnresolvedNodeIndex(index) => {
                debug!(target: "multipart", "Rewriting node '{}'", index.description);
                let index = index
                    .index()
                    .expect("Node index should have been resolved by now.");
                let mut buf = Vec::with_capacity(4);
                let byte_len: usize = buf.write_varnum(index).unwrap(); 
                (byte_len as u32, byte_len as u32, ResolvedTree::Encoded(buf))
            }
            UnresolvedOffset(None) => {
                panic!("UnresolvedOffset should have children");
            }
            UnresolvedOffset(Some(child)) => {
                let (sub_byte_len, sub_resolved) = child.resolve(stats);
                let mut buf = Vec::with_capacity(4);
                let offset_byte_len = buf.write_varnum(sub_byte_len).unwrap(); 
                let offset_resolved = ResolvedTree::Encoded(buf);
                (
                    sub_byte_len + offset_byte_len as u32,
                    offset_byte_len as u32,
                    ResolvedTree::Tuple(vec![
                        offset_resolved,
                        ResolvedTree::Tuple(vec![sub_resolved]),
                    ]),
                )
            }
            Tuple(mut subtrees) => {
                let mut byte_len = 0;
                let mut resolved = Vec::with_capacity(subtrees.len());
                for tree in subtrees.drain(..) {
                    let tree = std::rc::Rc::try_unwrap(tree).unwrap_or_else(|e| {
                        panic!(
                            "Could not unwrap tree, it still has {} consumers",
                            std::rc::Rc::strong_count(&e)
                        )
                    });
                    let (sub_byte_len, sub_resolved) = tree.resolve(stats);
                    byte_len += sub_byte_len;
                    resolved.push(sub_resolved);
                }
                (
                    byte_len as u32,
                    byte_len as u32,
                    ResolvedTree::Tuple(resolved),
                )
            }
            Encoded(vec) => {
                let byte_len = vec.len() as u32;
                (byte_len, byte_len, ResolvedTree::Encoded(vec))
            }
        };
        let total = total_bytes as usize;
        let own = own_bytes as usize;
        match self.nature {
            Nature::String(_) => {
                stats.string.entries += 1;
                stats.string.own_bytes += own;
                stats.string.total_bytes += total;
            }
            Nature::TaggedTuple(index) => {
                stats.tagged_tuple.entries += 1;
                for stat in &mut [&mut stats.tagged_tuple, &mut stats.tagged_header] {
                    stat.own_bytes += own;
                    stat.total_bytes += total;
                }
                let key = index
                    .index
                    .borrow()
                    .expect("TableIndex hasn't been resolved");
                debug!(target: "multipart", "Updating table for {}", key);
                match stats.per_kind_index.entry(key as usize) {
                    vec_map::Entry::Occupied(mut entry) => {
                        let borrow = entry.get_mut();
                        borrow.entries += 1;
                        borrow.total_bytes += total;
                        borrow.shallow_bytes += own;
                        borrow.own_bytes += own;
                    }
                    vec_map::Entry::Vacant(entry) => {
                        entry.insert(NodeStatistics {
                            entries: 1,
                            max_entries: 1,
                            shallow_bytes: own,
                            total_bytes: total,
                            own_bytes: own,
                        });
                    }
                }
            }
            Nature::TaggedTupleHeader(index) => {
                stats.tagged_header.entries += 1;
                stats.tagged_header.own_bytes += own;
                stats.tagged_header.total_bytes += total;
                let key = index
                    .index
                    .borrow()
                    .expect("TableIndex hasn't been resolved");
                debug!(target: "multipart", "Updating table for {}", key);
                match stats.per_kind_index.entry(key as usize) {
                    vec_map::Entry::Occupied(mut entry) => {
                        let borrow = entry.get_mut();
                        borrow.entries += 1;
                        borrow.total_bytes += total;
                        borrow.shallow_bytes += own;
                        borrow.own_bytes += own;
                    }
                    vec_map::Entry::Vacant(entry) => {
                        entry.insert(NodeStatistics {
                            entries: 1,
                            max_entries: 1,
                            shallow_bytes: own,
                            total_bytes: total,
                            own_bytes: own,
                        });
                    }
                }
            }
            Nature::List => {
                stats.list.entries += 1;
                stats.list.total_bytes += total;
            }
            Nature::ListHeader => {
                stats.list.own_bytes += total;
            }
            Nature::Bool => {
                stats.bool.entries += 1;
                stats.bool.own_bytes += own;
                stats.bool.total_bytes += total;
                stats.bool.shallow_bytes += own;
            }
            Nature::Float => {
                stats.float.entries += 1;
                stats.float.own_bytes += own;
                stats.float.total_bytes += total;
                stats.float.shallow_bytes += own;
            }
            Nature::UnsignedLong => {
                stats.unsigned_long.entries += 1;
                stats.unsigned_long.own_bytes += own;
                stats.unsigned_long.total_bytes += total;
                stats.unsigned_long.shallow_bytes += own;
            }
            Nature::Offset => {
                stats.offset.entries += 1;
                stats.bool.own_bytes += own;
                stats.bool.total_bytes += total;
                stats.bool.shallow_bytes += own;
            }
            _ => {}
        }
        (total_bytes, tree)
    }
}
#[derive(Debug)]
enum Nature {
    List,
    ListHeader,
    UntaggedTuple,
    TaggedTuple(TableIndex<NodeDescription>),
    TaggedTupleHeader(TableIndex<NodeDescription>),
    Float,
    UnsignedLong,
    Bool,
    String(TableIndex<Option<SharedString>>),
    
    Offset,
}
#[derive(Clone)]
pub struct Tree(Rc<UnresolvedTree>);
#[derive(Debug)]
struct TableIndex<T> {
    phantom: std::marker::PhantomData<T>,
    description: SharedString, 
    index: Rc<RefCell<Option<u32>>>,
}
impl<T> Clone for TableIndex<T> {
    fn clone(&self) -> Self {
        TableIndex {
            phantom: std::marker::PhantomData,
            description: self.description.clone(),
            index: self.index.clone(),
        }
    }
}
impl<T> TableIndex<T> {
    fn new(description: &str) -> Self {
        TableIndex {
            phantom: std::marker::PhantomData,
            description: SharedString::from_string(description.to_string()),
            index: Rc::new(RefCell::new(None)),
        }
    }
    pub fn index(&self) -> Option<u32> {
        match *self.index.borrow() {
            None => None,
            Some(ref value) => Some(value.clone()),
        }
    }
}
impl<T> Serializable for TableIndex<T> {
    fn write<W: Write>(&self, out: &mut W) -> Result<usize, std::io::Error> {
        if let Some(ref i) = *self.index.borrow() {
            out.write_varnum(*i)
        } else {
            panic!("Attempting to serialize a TableIndex whose index is None");
        }
    }
}
impl TreeTokenWriter {
    pub fn new(mut targets: Targets) -> Self {
        targets.reset();
        TreeTokenWriter {
            grammar_table: WriterTable::new(),
            strings_table: WriterTable::new(),
            root: None,
            data: Vec::with_capacity(1024),
            targets,
            statistics: Statistics::default(),
        }
    }
    fn register(&mut self, data: UnresolvedTree) -> Tree {
        let result = Rc::new(data);
        self.root = Some(Tree(result.clone()));
        Tree(result)
    }
    pub fn done(mut self) -> Result<Box<[u8]>, TokenWriterError> {
        const MAGIC_HEADER: &[u8; 5] = b"BINJS";
        
        self.data
            .write_all(MAGIC_HEADER)
            .map_err(TokenWriterError::WriteError)?;
        self.statistics.uncompressed_bytes += MAGIC_HEADER.len();
        const FORMAT_VERSION: u32 = 1;
        self.data
            .write_varnum(FORMAT_VERSION)
            .map_err(TokenWriterError::WriteError)?;
        self.statistics.uncompressed_bytes += std::mem::size_of_val(&FORMAT_VERSION);
        
        self.data
            .write_all(HEADER_GRAMMAR_TABLE.as_bytes())
            .map_err(TokenWriterError::WriteError)?;
        self.statistics.uncompressed_bytes += HEADER_GRAMMAR_TABLE.len();
        {
            self.grammar_table
                .write(&mut self.targets.grammar_table)
                .map_err(TokenWriterError::WriteError)?;
            let (data, compression) = self
                .targets
                .grammar_table
                .done()
                .map_err(TokenWriterError::WriteError)?;
            self.data
                .write_all(data.as_ref())
                .map_err(TokenWriterError::WriteError)?;
            self.statistics.grammar_table.entries = self.grammar_table.map.len();
            self.statistics.grammar_table.max_entries = self.grammar_table.map.len();
            self.statistics.grammar_table.compression = compression;
        }
        
        self.data
            .write_all(HEADER_STRINGS_TABLE.as_bytes())
            .map_err(TokenWriterError::WriteError)?;
        {
            self.strings_table
                .write(&mut self.targets.strings_table)
                .map_err(TokenWriterError::WriteError)?;
            let (data, compression) = self
                .targets
                .strings_table
                .done()
                .map_err(TokenWriterError::WriteError)?;
            self.data
                .write_all(data.as_ref())
                .map_err(TokenWriterError::WriteError)?;
            self.statistics.strings_table.entries = self.strings_table.map.len();
            self.statistics.strings_table.max_entries = self.strings_table.map.len();
            self.statistics.strings_table.compression = compression;
        }
        
        for (key, value) in &self.strings_table.map {
            let len = match *key {
                None => 0,
                Some(ref s) => s.len(),
            };
            match self.statistics.string_lengths.entry(len) {
                vec_map::Entry::Occupied(mut entry) => {
                    let borrow = entry.get_mut();
                    *borrow += 1;
                }
                vec_map::Entry::Vacant(entry) => {
                    entry.insert(1);
                }
            }
            match self
                .statistics
                .string_usage
                .entry(value.instances.borrow().clone() as usize)
            {
                vec_map::Entry::Occupied(mut entry) => {
                    let borrow = entry.get_mut();
                    *borrow += 1;
                }
                vec_map::Entry::Vacant(entry) => {
                    entry.insert(1);
                }
            }
        }
        
        if let Some(root) = self.root {
            let mut tree_buf = Vec::with_capacity(2048);
            let root = std::rc::Rc::try_unwrap(root.0).unwrap_or_else(|e| {
                panic!(
                    "Could not unwrap tree, it still has {} consumers",
                    std::rc::Rc::strong_count(&e)
                )
            });
            let (_, resolved) = root.resolve(&mut self.statistics);
            resolved
                .write(&mut tree_buf)
                .map_err(TokenWriterError::WriteError)?;
            self.data
                .write_all(HEADER_TREE.as_bytes())
                .map_err(TokenWriterError::WriteError)?;
            {
                tree_buf
                    .write(&mut self.targets.tree)
                    .map_err(TokenWriterError::WriteError)?;
                let (data, compression) = self
                    .targets
                    .tree
                    .done()
                    .map_err(TokenWriterError::WriteError)?;
                self.data
                    .write_all(data.as_ref())
                    .map_err(TokenWriterError::WriteError)?;
                self.statistics.tree.entries = 1;
                self.statistics.tree.max_entries = 1;
                self.statistics.tree.compression = compression;
            }
        }
        
        for (key, value) in self.grammar_table.map {
            let index = value
                .index
                .index
                .borrow()
                .expect("Table index hasn't been resolved yet");
            match self.statistics.per_kind_index.get(index as usize) {
                None => continue,
                Some(stats) => {
                    match self.statistics.per_kind_name.entry(key.kind.clone()) {
                        std::collections::hash_map::Entry::Occupied(mut entry) => {
                            let borrow = entry.get_mut();
                            *borrow += stats.clone();
                        }
                        std::collections::hash_map::Entry::Vacant(entry) => {
                            entry.insert(stats.clone());
                        }
                    }
                    self.statistics.per_description.insert(key, stats.clone());
                }
            }
        }
        self.statistics.number_of_files = 1;
        self.statistics.compressed_bytes = self.data.len();
        self.statistics.uncompressed_bytes +=
            self.statistics.grammar_table.compression.before_bytes
                + self.statistics.strings_table.compression.before_bytes
                + self.statistics.tree.compression.before_bytes;
        Ok(self.data.clone().into_boxed_slice())
    }
}
impl TokenWriterWithTree for TreeTokenWriter {
    type Tree = Tree;
    type Data = Box<[u8]>;
    fn done(self) -> Result<Self::Data, TokenWriterError> {
        (self as TreeTokenWriter).done()
    }
    fn float(&mut self, value: Option<f64>) -> Result<Self::Tree, TokenWriterError> {
        let bytes: Vec<_> = bytes::float::bytes_of_float(value)
            .iter()
            .cloned()
            .collect();
        debug!(target: "multipart", "writing float {:?} => {:?}", value, bytes);
        Ok(self.register(UnresolvedTree {
            nature: Nature::Float,
            data: UnresolvedTreeNode::Encoded(bytes),
        }))
    }
    fn unsigned_long(&mut self, value: u32) -> Result<Self::Tree, TokenWriterError> {
        let mut bytes = Vec::with_capacity(4);
        bytes
            .write_varnum(value as u32)
            .map_err(TokenWriterError::WriteError)?;
        debug!(target: "multipart", "writing unsigned_long {:?} => {:?}", value, bytes);
        Ok(self.register(UnresolvedTree {
            nature: Nature::UnsignedLong,
            data: UnresolvedTreeNode::Encoded(bytes),
        }))
    }
    fn bool(&mut self, data: Option<bool>) -> Result<Self::Tree, TokenWriterError> {
        let bytes = bytes::bool::bytes_of_bool(data).iter().cloned().collect();
        debug!(target: "multipart", "writing bool {:?} => {:?}", data, bytes);
        Ok(self.register(UnresolvedTree {
            nature: Nature::Bool,
            data: UnresolvedTreeNode::Encoded(bytes),
        }))
    }
    fn offset(&mut self) -> Result<Self::Tree, TokenWriterError> {
        Ok(self.register(UnresolvedTree {
            data: UnresolvedTreeNode::UnresolvedOffset(None), 
            nature: Nature::Offset,
        }))
    }
    fn string(&mut self, data: Option<&SharedString>) -> Result<Self::Tree, TokenWriterError> {
        let key = data.map(Clone::clone);
        let index = self
            .strings_table
            .get(&key)
            .map(|entry| entry.index.clone());
        if let Some(index) = index {
            return Ok(self.register(UnresolvedTree {
                data: UnresolvedTreeNode::UnresolvedStringIndex(index.clone()),
                nature: Nature::String(index),
            }));
        }
        let index = self.strings_table.insert(key);
        debug!(target: "multipart", "writing string {:?} => {:?}", data, index);
        Ok(self.register(UnresolvedTree {
            data: UnresolvedTreeNode::UnresolvedStringIndex(index.clone()),
            nature: Nature::String(index),
        }))
    }
    fn list(&mut self, mut children: Vec<Self::Tree>) -> Result<Self::Tree, TokenWriterError> {
        let mut items = Vec::with_capacity(children.len() + 1);
        
        let mut encoded_number_of_items = Vec::with_capacity(4);
        encoded_number_of_items
            .write_varnum(children.len() as u32)
            .map_err(TokenWriterError::WriteError)?;
        items.push(Rc::new(UnresolvedTree {
            data: UnresolvedTreeNode::Encoded(encoded_number_of_items),
            nature: Nature::ListHeader,
        }));
        let len = children.len();
        
        let children: Vec<_> = children.drain(..).map(|tree| tree.0.clone()).collect();
        items.extend(children);
        debug!(target: "multipart", "writing list with {} => {} items", len, items.len());
        Ok(self.register(UnresolvedTree {
            data: UnresolvedTreeNode::Tuple(items),
            nature: Nature::List,
        }))
    }
    
    
    
    
    
    
    
    fn tagged_tuple(
        &mut self,
        name: &InterfaceName,
        children: &[(&FieldName, Self::Tree)],
    ) -> Result<Self::Tree, TokenWriterError> {
        let data;
        let description = NodeDescription { kind: name.clone() };
        debug!(target: "multipart", "writing tagged tuple {:?} with {} children as {:?}",
            name,
            children.len(),
            description,
        );
        let index: TableIndex<_> = self.grammar_table.insert(description);
        {
            debug!(target: "multipart", "tagged tuple index: {index:?}",
                index = index);
            
            let mut children_data = vec![];
            
            
            
            
            
            
            
            
            
            
            
            
            
            
            
            let mut is_lazy_field = false;
            for child in children.iter() {
                let child_data = (child.1).0.clone();
                match *child_data {
                    UnresolvedTree {
                        data: UnresolvedTreeNode::UnresolvedOffset(ref v),
                        nature: Nature::Offset,
                    } => {
                        assert!(!is_lazy_field);
                        assert!(v.is_none());
                        
                        is_lazy_field = true;
                    }
                    _ => {
                        if is_lazy_field {
                            let offset_child = UnresolvedTree {
                                data: UnresolvedTreeNode::Tuple(vec![child_data.clone()]),
                                nature: Nature::UntaggedTuple,
                            };
                            children_data.push(Rc::new(UnresolvedTree {
                                data: UnresolvedTreeNode::UnresolvedOffset(Some(Box::new(
                                    offset_child,
                                ))),
                                nature: Nature::Offset,
                            }));
                            is_lazy_field = false;
                        } else {
                            children_data.push(child_data.clone());
                        }
                    }
                }
            }
            assert!(!is_lazy_field);
            let children = UnresolvedTree {
                data: UnresolvedTreeNode::Tuple(children_data),
                nature: Nature::UntaggedTuple,
            };
            let prefix = Rc::new(UnresolvedTree {
                data: UnresolvedTreeNode::UnresolvedNodeIndex(index.clone()),
                nature: Nature::TaggedTupleHeader(index.clone()),
            });
            data = vec![prefix, Rc::new(children)];
        }
        Ok(self.register(UnresolvedTree {
            data: UnresolvedTreeNode::Tuple(data),
            nature: Nature::TaggedTuple(index),
        }))
    }
}
pub struct TreeTokenWriter {
    
    
    grammar_table: WriterTable<NodeDescription>,
    
    strings_table: WriterTable<Option<SharedString>>,
    root: Option<Tree>,
    data: Vec<u8>,
    targets: Targets,
    statistics: Statistics,
}
#[derive(Clone, Debug)]
pub struct SectionStatistics {
    
    pub entries: usize,
    
    
    pub max_entries: usize,
    pub compression: CompressionResult,
}
impl Default for SectionStatistics {
    fn default() -> Self {
        SectionStatistics {
            entries: 0,
            max_entries: 0,
            compression: CompressionResult {
                before_bytes: 0,
                after_bytes: 0,
                algorithms: HashSet::new(),
            },
        }
    }
}
impl AddAssign for CompressionResult {
    fn add_assign(&mut self, mut rhs: Self) {
        self.before_bytes += rhs.before_bytes;
        self.after_bytes += rhs.after_bytes;
        self.algorithms.extend(rhs.algorithms.drain());
    }
}
impl AddAssign for SectionStatistics {
    fn add_assign(&mut self, rhs: Self) {
        self.entries += rhs.entries;
        if rhs.max_entries > self.max_entries {
            self.max_entries = rhs.max_entries;
        }
        self.compression += rhs.compression;
    }
}
#[derive(Clone, Debug, Default)]
pub struct NodeStatistics {
    
    pub entries: usize,
    
    
    pub max_entries: usize,
    
    
    pub own_bytes: usize,
    
    
    pub shallow_bytes: usize,
    
    pub total_bytes: usize,
}
impl AddAssign for NodeStatistics {
    fn add_assign(&mut self, rhs: Self) {
        self.entries += rhs.entries;
        if rhs.max_entries > self.max_entries {
            self.max_entries = rhs.max_entries;
        }
        self.own_bytes += rhs.own_bytes;
        self.shallow_bytes += rhs.shallow_bytes;
        self.total_bytes += rhs.total_bytes;
    }
}
#[derive(Clone, Debug, Default)]
pub struct Statistics {
    pub grammar_table: SectionStatistics,
    pub strings_table: SectionStatistics,
    pub tree: SectionStatistics,
    pub per_kind_index: VecMap<NodeStatistics>,
    pub per_kind_name: HashMap<InterfaceName, NodeStatistics>,
    pub per_description: HashMap<NodeDescription, NodeStatistics>,
    
    pub list_lengths: VecMap<usize>,
    
    pub string_lengths: VecMap<usize>,
    
    pub string_usage: VecMap<usize>,
    pub bool: NodeStatistics,
    pub float: NodeStatistics,
    pub unsigned_long: NodeStatistics,
    pub string: NodeStatistics,
    pub list: NodeStatistics,
    pub offset: NodeStatistics,
    pub list_header: NodeStatistics,
    pub tagged_header: NodeStatistics,
    pub tagged_tuple: NodeStatistics,
    pub number_of_files: usize,
    pub uncompressed_bytes: usize,
    pub compressed_bytes: usize,
    pub source_bytes: Option<usize>,
}
impl AddAssign for Statistics {
    fn add_assign(&mut self, rhs: Self) {
        let copy = self.clone() + rhs; 
        *self = copy;
    }
}
impl Add for Statistics {
    type Output = Self;
    fn add(mut self, mut rhs: Self) -> Self {
        self.grammar_table += rhs.grammar_table;
        self.strings_table += rhs.strings_table;
        self.tree += rhs.tree;
        for (key, value) in rhs.per_kind_index.drain() {
            use vec_map::Entry::*;
            match self.per_kind_index.entry(key) {
                Occupied(mut entry) => {
                    *entry.get_mut() += value;
                }
                Vacant(entry) => {
                    entry.insert(value);
                }
            }
        }
        for (key, value) in rhs.per_description.drain() {
            use std::collections::hash_map::Entry::*;
            match self.per_description.entry(key) {
                Occupied(mut entry) => {
                    *entry.get_mut() += value;
                }
                Vacant(entry) => {
                    entry.insert(value);
                }
            }
        }
        for (key, value) in rhs.per_kind_name.drain() {
            use std::collections::hash_map::Entry::*;
            match self.per_kind_name.entry(key) {
                Occupied(mut entry) => {
                    *entry.get_mut() += value;
                }
                Vacant(entry) => {
                    entry.insert(value);
                }
            }
        }
        for (key, value) in rhs.list_lengths.drain() {
            use vec_map::Entry::*;
            match self.list_lengths.entry(key) {
                Occupied(mut entry) => {
                    *entry.get_mut() += value;
                }
                Vacant(entry) => {
                    entry.insert(value);
                }
            }
        }
        for (key, value) in rhs.string_lengths.drain() {
            use vec_map::Entry::*;
            match self.string_lengths.entry(key) {
                Occupied(mut entry) => {
                    *entry.get_mut() += value;
                }
                Vacant(entry) => {
                    entry.insert(value);
                }
            }
        }
        for (key, value) in rhs.string_usage.drain() {
            use vec_map::Entry::*;
            match self.string_usage.entry(key) {
                Occupied(mut entry) => {
                    *entry.get_mut() += value;
                }
                Vacant(entry) => {
                    entry.insert(value);
                }
            }
        }
        self.bool += rhs.bool;
        self.float += rhs.float;
        self.unsigned_long += rhs.unsigned_long;
        self.string += rhs.string;
        self.list += rhs.list;
        self.list_header += rhs.list_header;
        self.tagged_header += rhs.tagged_header;
        self.tagged_tuple += rhs.tagged_tuple;
        self.number_of_files += rhs.number_of_files;
        self.compressed_bytes += rhs.compressed_bytes;
        self.uncompressed_bytes += rhs.uncompressed_bytes;
        self.source_bytes = match (self.source_bytes, rhs.source_bytes) {
            (Some(x), Some(y)) => Some(x + y),
            _ => None,
        };
        self
    }
}
impl Statistics {
    pub fn with_source_bytes(mut self, source_bytes: usize) -> Self {
        self.source_bytes = Some(source_bytes);
        self
    }
}
struct SectionAndStatistics<'a> {
    total_compressed_bytes: usize,
    total_uncompressed_bytes: usize,
    section: &'a SectionStatistics,
}
impl<'a> Display for SectionAndStatistics<'a> {
    fn fmt(&self, f: &mut Formatter) -> Result<(), std::fmt::Error> {
        write!(f, "\t\t\tEntries: {}\n", self.section.entries)?;
        if self.section.max_entries != self.section.entries {
            write!(
                f,
                "\t\t\tMax entries per file: {}\n",
                self.section.max_entries
            )?;
        }
        write!(f, "\t\t\tCompression: [")?;
        let mut first = true;
        for item in &self.section.compression.algorithms {
            if !first {
                write!(f, ", ")?;
            }
            first = false;
            write!(f, "{}", item.name())?;
        }
        write!(f, "]\n")?;
        write!(
            f,
            "\t\t\tUncompressed bytes: {} ({:.2}%)\n",
            self.section.compression.before_bytes,
            100. * (self.section.compression.before_bytes as f64)
                / (self.total_uncompressed_bytes as f64)
        )?;
        write!(
            f,
            "\t\t\tCompressed bytes: {} ({:.2}%)\n",
            self.section.compression.after_bytes,
            100. * (self.section.compression.after_bytes as f64)
                / (self.total_compressed_bytes as f64)
        )?;
        Ok(())
    }
}
struct NodeAndStatistics<'a> {
    name: &'a str,
    stats: &'a NodeStatistics,
    header_bytes: usize,
    total_uncompressed_bytes: usize,
    total_number_of_entries: usize,
}
impl<'a> Display for NodeAndStatistics<'a> {
    fn fmt(&self, f: &mut Formatter) -> Result<(), std::fmt::Error> {
        write!(f, "\t\t{}:\n", self.name)?;
        write!(
            f,
            "\t\t\tEntries: {} ({:.2}%)\n",
            self.stats.entries,
            100. * (self.stats.entries as f64) / (self.total_number_of_entries as f64)
        )?;
        if self.stats.max_entries != self.stats.entries {
            write!(
                f,
                "\t\t\tMax entries per file: {}\n",
                self.stats.max_entries
            )?;
        }
        write!(
            f,
            "\t\t\tOwn bytes: {} ({:.2}%)\n",
            self.stats.own_bytes,
            100. * (self.stats.own_bytes as f64) / (self.total_uncompressed_bytes as f64)
        )?;
        if self.header_bytes != 0 {
            write!(
                f,
                "\t\t\tHeader bytes: {} ({:.2}%)\n",
                self.header_bytes,
                100. * (self.header_bytes as f64) / (self.total_uncompressed_bytes as f64)
            )?;
        }
        if self.stats.shallow_bytes != self.stats.own_bytes && self.stats.shallow_bytes != 0 {
            write!(
                f,
                "\t\t\tShallow bytes: {} ({:.2}%)\n",
                self.stats.shallow_bytes,
                100. * (self.stats.shallow_bytes as f64) / (self.total_uncompressed_bytes as f64)
            )?;
        }
        if self.stats.total_bytes != self.stats.own_bytes {
            write!(
                f,
                "\t\t\tTotal bytes: {} ({:.2}%)\n",
                self.stats.total_bytes,
                100. * (self.stats.total_bytes as f64) / (self.total_uncompressed_bytes as f64)
            )?;
        }
        Ok(())
    }
}
struct NodeNameAndStatistics {
    total_uncompressed_bytes: usize,
    nodes: Vec<(InterfaceName, NodeStatistics)>,
}
impl Display for NodeNameAndStatistics {
    fn fmt(&self, f: &mut Formatter) -> Result<(), std::fmt::Error> {
        let total_number_of_entries: usize = self.nodes.iter().map(|node| node.1.entries).sum();
        
        for &(ref name, ref stats) in &self.nodes {
            let for_display = NodeAndStatistics {
                header_bytes: 0,
                total_uncompressed_bytes: self.total_uncompressed_bytes,
                total_number_of_entries,
                name: name.as_str(),
                stats,
            };
            write!(f, "{}", for_display)?;
        }
        Ok(())
    }
}
struct ListLengthsAndNumber(Vec<(usize, usize)>, String);
impl Display for ListLengthsAndNumber {
    fn fmt(&self, f: &mut Formatter) -> Result<(), std::fmt::Error> {
        
        let total_number_of_entries: usize = self.0.iter().map(|&(_, number)| number).sum();
        for &(ref length, ref number) in &self.0 {
            write!(
                f,
                "\t\t{} {} x {} ({:.2}%)\n",
                self.1,
                length,
                number,
                100. * (*number as f64) / (total_number_of_entries as f64)
            )?;
        }
        Ok(())
    }
}
impl Display for Statistics {
    fn fmt(&self, f: &mut Formatter) -> Result<(), std::fmt::Error> {
        
        let mut per_kind: Vec<_> = self
            .per_kind_name
            .iter()
            .map(|(a, b)| (a.clone(), b.clone()))
            .collect();
        per_kind.sort_unstable_by(|a, b| usize::cmp(&b.1.entries, &a.1.entries));
        
        let mut per_description: Vec<_> = self
            .per_description
            .iter()
            .map(|(a, b)| (a.clone(), b.clone()))
            .collect();
        per_description.sort_unstable_by(|a, b| usize::cmp(&b.1.entries, &a.1.entries));
        let mut list_per_size: Vec<_> = self
            .list_lengths
            .iter()
            .map(|(a, b)| (a.clone(), b.clone()))
            .collect();
        list_per_size.sort_unstable_by(|a, b| usize::cmp(&b.1, &a.1));
        let mut strings_per_size: Vec<_> = self
            .string_lengths
            .iter()
            .map(|(a, b)| (a.clone(), b.clone()))
            .collect();
        strings_per_size.sort_unstable_by(|a, b| usize::cmp(&b.1, &a.1));
        let mut strings_per_usage: Vec<_> = self
            .string_usage
            .iter()
            .map(|(a, b)| (a.clone(), b.clone()))
            .collect();
        strings_per_usage.sort_unstable_by(|a, b| usize::cmp(&b.1, &a.1));
        let total_number_of_tokens = self.bool.entries
            + self.float.entries
            + self.unsigned_long.entries
            + self.string.entries
            + self.list.entries
            + self.tagged_tuple.entries;
        write!(
            f,
            "
Statistics
\tFiles:
\t\tNumber: {number_of_files}
\t\tTotal source bytes: {total_source_bytes}
\t\tTotal uncompressed bytes: {total_uncompressed_bytes}
\t\tTotal compressed bytes: {total_compressed_bytes}
\t\tRatio: {compression_ratio}
\tSections:
\t\tGrammar:
{section_grammar}
\t\tStrings:
{section_strings}
\t\tTree:
{section_tree}
\tNodes:
{collapsed_nodes}
\tTokens:
{token_bool}
{token_float}
{token_unsigned_long}
{token_offset}
{token_string}
{token_list}
{token_tagged_tuple}
\tLists per size:
{lists_per_size}
\tStrings per size:
{strings_per_size}
\tStrings per usage:
{strings_per_usage}
",
            number_of_files = self.number_of_files,
            total_source_bytes = match self.source_bytes {
                None => "<not available>".to_string(),
                Some(ref bytes) => format!("{}", bytes),
            },
            total_uncompressed_bytes = self.uncompressed_bytes,
            total_compressed_bytes = self.compressed_bytes,
            compression_ratio = match self.source_bytes {
                None => "<not available>".to_string(),
                Some(ref bytes) => {
                    format!("{:.2}", (self.compressed_bytes as f64) / (*bytes as f64))
                }
            },
            lists_per_size = ListLengthsAndNumber(list_per_size, "length".to_string()),
            strings_per_size = ListLengthsAndNumber(strings_per_size, "length".to_string()),
            strings_per_usage = ListLengthsAndNumber(strings_per_usage, "occurrences".to_string()),
            section_grammar = SectionAndStatistics {
                total_uncompressed_bytes: self.uncompressed_bytes,
                total_compressed_bytes: self.compressed_bytes,
                section: &self.grammar_table,
            },
            section_strings = SectionAndStatistics {
                total_uncompressed_bytes: self.uncompressed_bytes,
                total_compressed_bytes: self.compressed_bytes,
                section: &self.strings_table,
            },
            section_tree = SectionAndStatistics {
                total_uncompressed_bytes: self.uncompressed_bytes,
                total_compressed_bytes: self.compressed_bytes,
                section: &self.tree,
            },
            collapsed_nodes = NodeNameAndStatistics {
                total_uncompressed_bytes: self.uncompressed_bytes,
                nodes: per_kind
            },
            token_bool = NodeAndStatistics {
                name: "Bool",
                stats: &self.bool,
                total_number_of_entries: total_number_of_tokens,
                total_uncompressed_bytes: self.uncompressed_bytes,
                header_bytes: 0,
            },
            token_float = NodeAndStatistics {
                name: "Float",
                stats: &self.float,
                total_number_of_entries: total_number_of_tokens,
                total_uncompressed_bytes: self.uncompressed_bytes,
                header_bytes: 0,
            },
            token_unsigned_long = NodeAndStatistics {
                name: "UnsignedLong",
                stats: &self.unsigned_long,
                total_number_of_entries: total_number_of_tokens,
                total_uncompressed_bytes: self.uncompressed_bytes,
                header_bytes: 0,
            },
            token_offset = NodeAndStatistics {
                name: "Offset",
                stats: &self.offset,
                total_number_of_entries: total_number_of_tokens,
                total_uncompressed_bytes: self.uncompressed_bytes,
                header_bytes: 0,
            },
            token_string = NodeAndStatistics {
                name: "String indices",
                stats: &self.string,
                total_number_of_entries: total_number_of_tokens,
                total_uncompressed_bytes: self.uncompressed_bytes,
                header_bytes: 0,
            },
            token_list = NodeAndStatistics {
                name: "List",
                stats: &self.list,
                total_number_of_entries: total_number_of_tokens,
                total_uncompressed_bytes: self.uncompressed_bytes,
                header_bytes: self.list_header.own_bytes,
            },
            token_tagged_tuple = NodeAndStatistics {
                name: "Tagged Tuple",
                stats: &self.tagged_tuple,
                total_number_of_entries: total_number_of_tokens,
                total_uncompressed_bytes: self.uncompressed_bytes,
                header_bytes: self.tagged_header.own_bytes,
            },
        )
    }
}