-
Notifications
You must be signed in to change notification settings - Fork 221
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Required switching to `serde_yaml`. - Like with `yaml-rust`, requires special handling for table keys. Signed-off-by: Brennan Kinney <[email protected]>
- Loading branch information
1 parent
0120ea5
commit cfabdba
Showing
6 changed files
with
50 additions
and
100 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,105 +1,14 @@ | ||
use std::error::Error; | ||
use std::fmt; | ||
use std::mem; | ||
|
||
use yaml_rust as yaml; | ||
|
||
use crate::format; | ||
use crate::map::Map; | ||
use crate::value::{Value, ValueKind}; | ||
use crate::value::Value; | ||
|
||
pub fn parse( | ||
uri: Option<&String>, | ||
text: &str, | ||
) -> Result<Map<String, Value>, Box<dyn Error + Send + Sync>> { | ||
// Parse a YAML object from file | ||
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)?; | ||
// Parse a YAML input from the provided text | ||
let value = format::from_parsed_value(uri, serde_yaml::from_str(text)?); | ||
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) => { | ||
// TODO: Figure out in what cases this can panic? | ||
value | ||
.parse::<f64>() | ||
.map_err(|_| { | ||
Box::new(FloatParsingError(value.to_string())) 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)?), | ||
_ => unreachable!(), | ||
}; | ||
} | ||
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))) | ||
} | ||
|
||
// 1. Yaml NULL | ||
// 2. BadValue – It shouldn't be possible to hit BadValue as this only happens when | ||
// using the index trait badly or on a type error but we send back nil. | ||
// 3. Alias – No idea what to do with this and there is a note in the lib that its | ||
// not fully supported yet anyway | ||
_ => 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" | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters