use crate::Error;
use crate::SymbolValue;
use crate::TypedValueHandle;
use crate::Value;
use crate::ValueArena;
use crate::ValueHandle;
use crate::MAJOR_VERSION;
use crate::MINOR_VERSION;
use crate::VALUE_KIND_ARRAY;
use crate::VALUE_KIND_FALSE;
use crate::VALUE_KIND_FIXNUM;
use crate::VALUE_KIND_HASH;
use crate::VALUE_KIND_HASH_DEFAULT;
use crate::VALUE_KIND_INSTANCE_VARIABLES;
use crate::VALUE_KIND_NIL;
use crate::VALUE_KIND_OBJECT;
use crate::VALUE_KIND_OBJECT_LINK;
use crate::VALUE_KIND_STRING;
use crate::VALUE_KIND_SYMBOL;
use crate::VALUE_KIND_SYMBOL_LINK;
use crate::VALUE_KIND_TRUE;
use crate::VALUE_KIND_USER_DEFINED;
use indexmap::IndexSet;
use std::io::Write;
pub struct Dumper<'a, W> {
writer: W,
arena: &'a ValueArena,
symbol_links: IndexSet<TypedValueHandle<SymbolValue>>,
object_links: IndexSet<ValueHandle>,
}
impl<'a, W> Dumper<'a, W> {
fn new(writer: W, arena: &'a ValueArena) -> Self {
Self {
writer,
arena,
symbol_links: IndexSet::new(),
object_links: IndexSet::new(),
}
}
}
impl<W> Dumper<'_, W>
where
W: Write,
{
fn write_header(&mut self) -> Result<(), Error> {
self.writer.write_all(&[MAJOR_VERSION, MINOR_VERSION])?;
Ok(())
}
fn write_byte(&mut self, byte: u8) -> Result<(), Error> {
self.writer.write_all(std::slice::from_ref(&byte))?;
Ok(())
}
fn write_byte_string(&mut self, value: &[u8]) -> Result<(), Error> {
let len =
i32::try_from(value.len()).map_err(|error| Error::USizeInvalidFixnum { error })?;
self.write_fixnum(len)?;
self.writer.write_all(value)?;
Ok(())
}
fn write_fixnum(&mut self, mut value: i32) -> Result<(), Error> {
if value == 0 {
self.writer.write_all(&[0])?;
return Ok(());
}
if value > 0 && value < 123 {
let value = u8::try_from(value).unwrap();
self.writer.write_all(&[value + 5])?;
return Ok(());
}
if value < 0 && value > -124 {
let value = u8::try_from((value - 5) & 0xFF).unwrap();
self.writer.write_all(&[value])?;
return Ok(());
}
let mut buffer = [0; std::mem::size_of::<i32>() + 1];
let mut buffer_size = 0;
for i in 1..(std::mem::size_of::<i32>() + 1) {
buffer[i] = u8::try_from(value & 0xFF).unwrap();
buffer_size = i + 1;
value >>= 8;
if value == 0 {
buffer[0] = u8::try_from(i).unwrap();
break;
}
if value == -1 {
buffer[0] = (-i8::try_from(i).unwrap()) as u8;
break;
}
}
self.writer.write_all(&buffer[..buffer_size])?;
Ok(())
}
fn try_write_value_object_link(&mut self, handle: ValueHandle) -> Result<bool, Error> {
let (index, inserted) = self.object_links.insert_full(handle);
if !inserted {
self.write_value_object_link(index)?;
return Ok(true);
}
Ok(false)
}
fn write_value_symbol_like(
&mut self,
handle: TypedValueHandle<SymbolValue>,
value: &SymbolValue,
) -> Result<(), Error> {
match self.symbol_links.get_index_of(&handle) {
Some(index) => {
let index =
i32::try_from(index).map_err(|error| Error::USizeInvalidFixnum { error })?;
self.write_byte(VALUE_KIND_SYMBOL_LINK)?;
self.write_fixnum(index)?;
}
None => {
self.symbol_links.insert(handle);
self.write_byte(VALUE_KIND_SYMBOL)?;
self.write_byte_string(value.value())?;
}
}
Ok(())
}
fn write_value_object_link(&mut self, index: usize) -> Result<(), Error> {
let index = i32::try_from(index).map_err(|error| Error::USizeInvalidFixnum { error })?;
self.write_byte(VALUE_KIND_OBJECT_LINK)?;
self.write_fixnum(index)?;
Ok(())
}
fn write_instance_variables(
&mut self,
instance_variables: &[(TypedValueHandle<SymbolValue>, ValueHandle)],
) -> Result<(), Error> {
let num_vars = i32::try_from(instance_variables.len())
.map_err(|error| Error::USizeInvalidFixnum { error })?;
self.write_fixnum(num_vars)?;
for (name, value) in instance_variables.iter() {
self.write_value((*name).into())?;
self.write_value(*value)?;
}
Ok(())
}
fn write_value(&mut self, handle: ValueHandle) -> Result<(), Error> {
let value = self
.arena
.get(handle)
.ok_or(Error::InvalidValueHandle { handle })?;
match value {
Value::Nil(_) => self.write_byte(VALUE_KIND_NIL)?,
Value::Bool(value) => {
if value.value() {
self.write_byte(VALUE_KIND_TRUE)?
} else {
self.write_byte(VALUE_KIND_FALSE)?
}
}
Value::Fixnum(value) => {
self.write_byte(VALUE_KIND_FIXNUM)?;
self.write_fixnum(value.value())?;
}
Value::Symbol(value) => {
let handle = TypedValueHandle::new_unchecked(handle);
self.write_value_symbol_like(handle, value)?;
}
Value::Array(value) => {
if self.try_write_value_object_link(handle)? {
return Ok(());
}
let len = i32::try_from(value.len())
.map_err(|error| Error::USizeInvalidFixnum { error })?;
self.write_byte(VALUE_KIND_ARRAY)?;
self.write_fixnum(len)?;
for value in value.value().iter() {
self.write_value(*value)?;
}
}
Value::Hash(value) => {
if self.try_write_value_object_link(handle)? {
return Ok(());
}
let default_value = value.default_value();
let value = value.value();
if default_value.is_some() {
self.write_byte(VALUE_KIND_HASH_DEFAULT)?;
} else {
self.write_byte(VALUE_KIND_HASH)?;
}
let num_vars = i32::try_from(value.len())
.map_err(|error| Error::USizeInvalidFixnum { error })?;
self.write_fixnum(num_vars)?;
for (key, value) in value.iter() {
self.write_value(*key)?;
self.write_value(*value)?;
}
if let Some(default_value) = default_value {
self.write_value(default_value)?;
}
}
Value::Object(value) => {
if self.try_write_value_object_link(handle)? {
return Ok(());
}
self.write_byte(VALUE_KIND_OBJECT)?;
self.write_value(value.name().into())?;
self.write_instance_variables(value.instance_variables())?;
}
Value::String(value) => {
if self.try_write_value_object_link(handle)? {
return Ok(());
}
match value.instance_variables() {
Some(instance_variables) => {
self.write_byte(VALUE_KIND_INSTANCE_VARIABLES)?;
self.write_byte(VALUE_KIND_STRING)?;
self.write_byte_string(value.value())?;
self.write_instance_variables(instance_variables)?;
}
None => {
self.write_byte(VALUE_KIND_STRING)?;
self.write_byte_string(value.value())?;
}
}
}
Value::UserDefined(value) => {
if self.try_write_value_object_link(handle)? {
return Ok(());
}
match value.instance_variables() {
Some(instance_variables) => {
self.write_byte(VALUE_KIND_INSTANCE_VARIABLES)?;
self.write_byte(VALUE_KIND_USER_DEFINED)?;
self.write_value(value.name().into())?;
self.write_byte_string(value.value())?;
self.write_instance_variables(instance_variables)?;
}
None => {
self.write_byte(VALUE_KIND_USER_DEFINED)?;
self.write_value(value.name().into())?;
self.write_byte_string(value.value())?;
}
}
}
}
Ok(())
}
fn dump(&mut self) -> Result<(), Error> {
self.write_header()?;
self.write_value(self.arena.root())?;
Ok(())
}
}
pub fn dump<W>(writer: W, value_arena: &ValueArena) -> Result<(), Error>
where
W: Write,
{
let mut dumper = Dumper::new(writer, value_arena);
dumper.dump()?;
Ok(())
}