use std::error::Error;
use std::fmt;
use std::mem;
use yaml_rust2 as yaml;
use crate::format;
use crate::map::Map;
use crate::value::{Value, ValueKind};
pub(crate) fn parse(
uri: Option<&String>,
text: &str,
) -> Result<Map<String, Value>, Box<dyn Error + Send + Sync>> {
let mut docs = yaml::YamlLoader::load_from_str(text)?;
let root = match docs.len() {
0 => yaml::Yaml::Hash(yaml::yaml::Hash::new()),
1 => mem::replace(&mut docs[0], yaml::Yaml::Null),
n => {
return Err(Box::new(MultipleDocumentsError(n)));
}
};
let value = from_yaml_value(uri, &root)?;
format::extract_root_table(uri, value)
}
fn from_yaml_value(
uri: Option<&String>,
value: &yaml::Yaml,
) -> Result<Value, Box<dyn Error + Send + Sync>> {
match *value {
yaml::Yaml::String(ref value) => Ok(Value::new(uri, ValueKind::String(value.clone()))),
yaml::Yaml::Real(ref value) => {
value
.parse::<f64>()
.map_err(|_| {
Box::new(FloatParsingError(value.clone())) as Box<dyn Error + Send + Sync>
})
.map(ValueKind::Float)
.map(|f| Value::new(uri, f))
}
yaml::Yaml::Integer(value) => Ok(Value::new(uri, ValueKind::I64(value))),
yaml::Yaml::Boolean(value) => Ok(Value::new(uri, ValueKind::Boolean(value))),
yaml::Yaml::Hash(ref table) => {
let mut m = Map::new();
for (key, value) in table {
match key {
yaml::Yaml::String(k) => m.insert(k.to_owned(), from_yaml_value(uri, value)?),
yaml::Yaml::Integer(k) => m.insert(k.to_string(), from_yaml_value(uri, value)?),
yaml::Yaml::Boolean(k) => m.insert(k.to_string(), from_yaml_value(uri, value)?),
yaml::Yaml::Real(k) => m.insert(k.to_owned(), from_yaml_value(uri, value)?),
other => Err(Box::new(UnsupportedHashKeyError(format!("{other:?}"))))?,
};
}
Ok(Value::new(uri, ValueKind::Table(m)))
}
yaml::Yaml::Array(ref array) => {
let mut l = Vec::new();
for value in array {
l.push(from_yaml_value(uri, value)?);
}
Ok(Value::new(uri, ValueKind::Array(l)))
}
_ => Ok(Value::new(uri, ValueKind::Nil)),
}
}
#[derive(Debug, Copy, Clone)]
struct MultipleDocumentsError(usize);
impl fmt::Display for MultipleDocumentsError {
fn fmt(&self, format: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(format, "Got {} YAML documents, expected 1", self.0)
}
}
impl Error for MultipleDocumentsError {
fn description(&self) -> &str {
"More than one YAML document provided"
}
}
#[derive(Debug, Clone)]
struct FloatParsingError(String);
impl fmt::Display for FloatParsingError {
fn fmt(&self, format: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(format, "Parsing {} as floating point number failed", self.0)
}
}
impl Error for FloatParsingError {
fn description(&self) -> &str {
"Floating point number parsing failed"
}
}
#[derive(Debug, Clone)]
struct UnsupportedHashKeyError(String);
impl fmt::Display for UnsupportedHashKeyError {
fn fmt(&self, format: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
format,
"Cannot parse {} because it is an unsupported hash key type",
self.0
)
}
}
impl Error for UnsupportedHashKeyError {
fn description(&self) -> &str {
"Unsupported yaml hash key found"
}
}