mirror of
https://github.com/nushell/nushell
synced 2025-01-13 21:55:07 +00:00
Merge pull request #579 from est31/serde_instead_of_specialization
Use serde instead of specialization
This commit is contained in:
commit
1464feaab7
3 changed files with 70 additions and 52 deletions
|
@ -1,5 +1,4 @@
|
|||
#![feature(generators)]
|
||||
#![feature(specialization)]
|
||||
#![feature(proc_macro_hygiene)]
|
||||
|
||||
#[macro_use]
|
||||
|
|
|
@ -1,4 +1,3 @@
|
|||
use crate::object::base as value;
|
||||
use crate::prelude::*;
|
||||
use log::trace;
|
||||
|
||||
|
@ -6,29 +5,6 @@ pub trait ExtractType: Sized {
|
|||
fn extract(value: &Tagged<Value>) -> Result<Self, ShellError>;
|
||||
}
|
||||
|
||||
impl<T> ExtractType for T {
|
||||
default fn extract(_value: &Tagged<Value>) -> Result<T, ShellError> {
|
||||
let name = std::any::type_name::<T>();
|
||||
Err(ShellError::unimplemented(format!(
|
||||
"<T> ExtractType for {}",
|
||||
name
|
||||
)))
|
||||
}
|
||||
}
|
||||
impl<T: ExtractType> ExtractType for Option<T> {
|
||||
fn extract(value: &Tagged<Value>) -> Result<Option<T>, ShellError> {
|
||||
let name = std::any::type_name::<T>();
|
||||
trace!("<Option> Extracting {:?} for Option<{}>", value, name);
|
||||
|
||||
let result = match value.item() {
|
||||
Value::Primitive(Primitive::Nothing) => None,
|
||||
_ => Some(T::extract(value)?),
|
||||
};
|
||||
|
||||
Ok(result)
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: ExtractType> ExtractType for Tagged<T> {
|
||||
fn extract(value: &Tagged<Value>) -> Result<Tagged<T>, ShellError> {
|
||||
let name = std::any::type_name::<T>();
|
||||
|
@ -38,14 +14,6 @@ impl<T: ExtractType> ExtractType for Tagged<T> {
|
|||
}
|
||||
}
|
||||
|
||||
impl ExtractType for Value {
|
||||
fn extract(value: &Tagged<Value>) -> Result<Value, ShellError> {
|
||||
trace!("<Tagged> Extracting {:?} for Value", value);
|
||||
|
||||
Ok(value.item().clone())
|
||||
}
|
||||
}
|
||||
|
||||
impl ExtractType for bool {
|
||||
fn extract(value: &Tagged<Value>) -> Result<bool, ShellError> {
|
||||
trace!("Extracting {:?} for bool", value);
|
||||
|
@ -119,15 +87,3 @@ impl ExtractType for String {
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl ExtractType for value::Block {
|
||||
fn extract(value: &Tagged<Value>) -> Result<value::Block, ShellError> {
|
||||
match value {
|
||||
Tagged {
|
||||
item: Value::Block(block),
|
||||
..
|
||||
} => Ok(block.clone()),
|
||||
other => Err(ShellError::type_error("Block", other.tagged_type_name())),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
use crate::prelude::*;
|
||||
use log::trace;
|
||||
use serde::de;
|
||||
use std::path::PathBuf;
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct DeserializerItem<'de> {
|
||||
|
@ -293,6 +294,22 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut ConfigDeserializer<'de> {
|
|||
where
|
||||
V: Visitor<'de>,
|
||||
{
|
||||
fn visit<'de, T, V>(
|
||||
val: T,
|
||||
name: &'static str,
|
||||
fields: &'static [&'static str],
|
||||
visitor: V
|
||||
) -> Result<V::Value, ShellError>
|
||||
where
|
||||
T: serde::Serialize,
|
||||
V: Visitor<'de>,
|
||||
{
|
||||
let json = serde_json::to_string(&val)?;
|
||||
let json_cursor = std::io::Cursor::new(json.into_bytes());
|
||||
let mut json_de = serde_json::Deserializer::from_reader(json_cursor);
|
||||
let r = json_de.deserialize_struct(name, fields, visitor)?;
|
||||
return Ok(r);
|
||||
}
|
||||
trace!(
|
||||
"deserializing struct {:?} {:?} (stack={:?})",
|
||||
name,
|
||||
|
@ -300,14 +317,60 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut ConfigDeserializer<'de> {
|
|||
self.stack
|
||||
);
|
||||
|
||||
if self.saw_root {
|
||||
let value = self.pop();
|
||||
let name = std::any::type_name::<V::Value>();
|
||||
trace!("Extracting {:?} for {:?}", value.val, name);
|
||||
V::Value::extract(&value.val)
|
||||
} else {
|
||||
if !self.saw_root {
|
||||
self.saw_root = true;
|
||||
visitor.visit_seq(StructDeserializer::new(&mut self, fields))
|
||||
return visitor.visit_seq(StructDeserializer::new(&mut self, fields));
|
||||
}
|
||||
|
||||
let value = self.pop();
|
||||
|
||||
let type_name = std::any::type_name::<V::Value>();
|
||||
let tagged_val_name = std::any::type_name::<Tagged<Value>>();
|
||||
|
||||
if name == tagged_val_name {
|
||||
return visit::<Tagged<Value>, _>(value.val, name, fields, visitor);
|
||||
}
|
||||
|
||||
if name == "Block" {
|
||||
let block = match value.val {
|
||||
Tagged {
|
||||
item: Value::Block(block),
|
||||
..
|
||||
} => block,
|
||||
other => return Err(ShellError::type_error("Block", other.tagged_type_name())),
|
||||
};
|
||||
return visit::<value::Block, _>(block, name, fields, visitor);
|
||||
}
|
||||
|
||||
trace!("Extracting {:?} for {:?}", value.val, type_name);
|
||||
|
||||
let tag = value.val.tag();
|
||||
match value.val {
|
||||
Tagged {
|
||||
item: Value::Primitive(Primitive::Boolean(b)),
|
||||
..
|
||||
} => visit::<Tagged<bool>, _>(b.tagged(tag), name, fields, visitor),
|
||||
Tagged {
|
||||
item: Value::Primitive(Primitive::Nothing),
|
||||
..
|
||||
} => visit::<Tagged<bool>, _>(false.tagged(tag), name, fields, visitor),
|
||||
Tagged {
|
||||
item: Value::Primitive(Primitive::Path(p)),
|
||||
..
|
||||
} => visit::<Tagged<PathBuf>, _>(p.clone().tagged(tag), name, fields, visitor),
|
||||
Tagged {
|
||||
item: Value::Primitive(Primitive::Int(int)),
|
||||
..
|
||||
} => {
|
||||
let i: i64 = int.tagged(value.val.tag).coerce_into("converting to i64")?;
|
||||
visit::<Tagged<i64>, _>(i.tagged(tag), name, fields, visitor)
|
||||
},
|
||||
Tagged {
|
||||
item: Value::Primitive(Primitive::String(string)),
|
||||
..
|
||||
} => visit::<Tagged<String>, _>(string.tagged(tag), name, fields, visitor),
|
||||
|
||||
other => return Err(ShellError::type_error(name, other.tagged_type_name())),
|
||||
}
|
||||
}
|
||||
fn deserialize_enum<V>(
|
||||
|
|
Loading…
Reference in a new issue