mirror of
https://github.com/nushell/nushell
synced 2024-12-27 21:43:09 +00:00
Merge pull request #569 from est31/serde_instead_of_specialization
Remove use of ExtractType in deserialize_any
This commit is contained in:
commit
246c9c06dc
2 changed files with 116 additions and 68 deletions
|
@ -15,59 +15,6 @@ impl<T> ExtractType for T {
|
||||||
)))
|
)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: ExtractType> ExtractType for Vec<Tagged<T>> {
|
|
||||||
fn extract(value: &Tagged<Value>) -> Result<Self, ShellError> {
|
|
||||||
let name = std::any::type_name::<T>();
|
|
||||||
trace!("<Vec> Extracting {:?} for Vec<{}>", value, name);
|
|
||||||
|
|
||||||
match value.item() {
|
|
||||||
Value::List(items) => {
|
|
||||||
let mut out = vec![];
|
|
||||||
|
|
||||||
for item in items {
|
|
||||||
out.push(T::extract(item)?.tagged(item.tag()));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(out)
|
|
||||||
}
|
|
||||||
other => Err(ShellError::type_error(
|
|
||||||
"Vec",
|
|
||||||
other.type_name().tagged(value.tag()),
|
|
||||||
)),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: ExtractType, U: ExtractType> ExtractType for (T, U) {
|
|
||||||
fn extract(value: &Tagged<Value>) -> Result<(T, U), ShellError> {
|
|
||||||
let t_name = std::any::type_name::<T>();
|
|
||||||
let u_name = std::any::type_name::<U>();
|
|
||||||
|
|
||||||
trace!("Extracting {:?} for ({}, {})", value, t_name, u_name);
|
|
||||||
|
|
||||||
match value.item() {
|
|
||||||
Value::List(items) => {
|
|
||||||
if items.len() == 2 {
|
|
||||||
let first = &items[0];
|
|
||||||
let second = &items[1];
|
|
||||||
|
|
||||||
Ok((T::extract(first)?, U::extract(second)?))
|
|
||||||
} else {
|
|
||||||
Err(ShellError::type_error(
|
|
||||||
"two-element-tuple",
|
|
||||||
"not-two".tagged(value.tag()),
|
|
||||||
))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
other => Err(ShellError::type_error(
|
|
||||||
"two-element-tuple",
|
|
||||||
other.type_name().tagged(value.tag()),
|
|
||||||
)),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: ExtractType> ExtractType for Option<T> {
|
impl<T: ExtractType> ExtractType for Option<T> {
|
||||||
fn extract(value: &Tagged<Value>) -> Result<Option<T>, ShellError> {
|
fn extract(value: &Tagged<Value>) -> Result<Option<T>, ShellError> {
|
||||||
let name = std::any::type_name::<T>();
|
let name = std::any::type_name::<T>();
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
use crate::prelude::*;
|
use crate::prelude::*;
|
||||||
use log::trace;
|
use log::trace;
|
||||||
use serde::{de, forward_to_deserialize_any};
|
use serde::de;
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct DeserializerItem<'de> {
|
pub struct DeserializerItem<'de> {
|
||||||
key: String,
|
key_struct_field: Option<(String, &'de str)>,
|
||||||
struct_field: &'de str,
|
|
||||||
val: Tagged<Value>,
|
val: Tagged<Value>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -26,6 +25,13 @@ impl<'de> ConfigDeserializer<'de> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn push_val(&mut self, val: Tagged<Value>) {
|
||||||
|
self.stack.push(DeserializerItem {
|
||||||
|
key_struct_field: None,
|
||||||
|
val,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
pub fn push(&mut self, name: &'static str) -> Result<(), ShellError> {
|
pub fn push(&mut self, name: &'static str) -> Result<(), ShellError> {
|
||||||
let value: Option<Tagged<Value>> = if name == "rest" {
|
let value: Option<Tagged<Value>> = if name == "rest" {
|
||||||
let positional = self.call.args.slice_from(self.position);
|
let positional = self.call.args.slice_from(self.position);
|
||||||
|
@ -44,8 +50,7 @@ impl<'de> ConfigDeserializer<'de> {
|
||||||
trace!("pushing {:?}", value);
|
trace!("pushing {:?}", value);
|
||||||
|
|
||||||
self.stack.push(DeserializerItem {
|
self.stack.push(DeserializerItem {
|
||||||
key: name.to_string(),
|
key_struct_field: Some((name.to_string(), name)),
|
||||||
struct_field: name,
|
|
||||||
val: value.unwrap_or_else(|| {
|
val: value.unwrap_or_else(|| {
|
||||||
Value::nothing().tagged(Tag::unknown_origin(self.call.name_span))
|
Value::nothing().tagged(Tag::unknown_origin(self.call.name_span))
|
||||||
}),
|
}),
|
||||||
|
@ -54,6 +59,12 @@ impl<'de> ConfigDeserializer<'de> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn top(&mut self) -> &DeserializerItem {
|
||||||
|
let value = self.stack.last();
|
||||||
|
trace!("inspecting top value :: {:?}", value);
|
||||||
|
value.expect("Can't get top elemant of an empty stack")
|
||||||
|
}
|
||||||
|
|
||||||
pub fn pop(&mut self) -> DeserializerItem {
|
pub fn pop(&mut self) -> DeserializerItem {
|
||||||
let value = self.stack.pop();
|
let value = self.stack.pop();
|
||||||
trace!("popping value :: {:?}", value);
|
trace!("popping value :: {:?}", value);
|
||||||
|
@ -69,15 +80,27 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut ConfigDeserializer<'de> {
|
||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
let value = self.pop();
|
unimplemented!("deserialize_any")
|
||||||
let name = std::any::type_name::<V::Value>();
|
|
||||||
trace!("<Deserialize any> Extracting {:?}", name);
|
|
||||||
|
|
||||||
V::Value::extract(&value.val)
|
|
||||||
}
|
}
|
||||||
|
fn deserialize_bool<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
let value = self.pop();
|
||||||
|
trace!("Extracting {:?} for bool", value.val);
|
||||||
|
|
||||||
forward_to_deserialize_any! { bool option seq }
|
match &value.val {
|
||||||
|
Tagged {
|
||||||
|
item: Value::Primitive(Primitive::Boolean(b)),
|
||||||
|
..
|
||||||
|
} => visitor.visit_bool(*b),
|
||||||
|
Tagged {
|
||||||
|
item: Value::Primitive(Primitive::Nothing),
|
||||||
|
..
|
||||||
|
} => visitor.visit_bool(false),
|
||||||
|
other => Err(ShellError::type_error("Boolean", other.tagged_type_name())),
|
||||||
|
}
|
||||||
|
}
|
||||||
fn deserialize_i8<V>(self, _visitor: V) -> Result<V::Value, Self::Error>
|
fn deserialize_i8<V>(self, _visitor: V) -> Result<V::Value, Self::Error>
|
||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
|
@ -169,6 +192,19 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut ConfigDeserializer<'de> {
|
||||||
unimplemented!("deserialize_byte_buf")
|
unimplemented!("deserialize_byte_buf")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn deserialize_option<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
let value = self.top();
|
||||||
|
let name = std::any::type_name::<V::Value>();
|
||||||
|
trace!("<Option> Extracting {:?} for Option<{}>", value, name);
|
||||||
|
match value.val.item() {
|
||||||
|
Value::Primitive(Primitive::Nothing) => visitor.visit_none(),
|
||||||
|
_ => visitor.visit_some(self),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn deserialize_unit<V>(self, _visitor: V) -> Result<V::Value, Self::Error>
|
fn deserialize_unit<V>(self, _visitor: V) -> Result<V::Value, Self::Error>
|
||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
|
@ -195,12 +231,41 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut ConfigDeserializer<'de> {
|
||||||
{
|
{
|
||||||
unimplemented!("deserialize_newtype_struct")
|
unimplemented!("deserialize_newtype_struct")
|
||||||
}
|
}
|
||||||
|
fn deserialize_seq<V>(mut self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
fn deserialize_tuple<V>(self, _len: usize, _visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
unimplemented!("deserialize_tuple")
|
let value = self.pop();
|
||||||
|
trace!("<Vec> Extracting {:?} for vec", value.val);
|
||||||
|
|
||||||
|
match value.val.into_parts() {
|
||||||
|
(Value::List(items), _) => {
|
||||||
|
let de = SeqDeserializer::new(&mut self, items.into_iter());
|
||||||
|
visitor.visit_seq(de)
|
||||||
|
}
|
||||||
|
(other, tag) => Err(ShellError::type_error(
|
||||||
|
"Vec",
|
||||||
|
other.type_name().tagged(tag),
|
||||||
|
)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn deserialize_tuple<V>(mut self, len: usize, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
let value = self.pop();
|
||||||
|
trace!("<Tuple> Extracting {:?} for tuple with {} elements", value.val, len);
|
||||||
|
|
||||||
|
match value.val.into_parts() {
|
||||||
|
(Value::List(items), _) => {
|
||||||
|
let de = SeqDeserializer::new(&mut self, items.into_iter());
|
||||||
|
visitor.visit_seq(de)
|
||||||
|
}
|
||||||
|
(other, tag) => Err(ShellError::type_error(
|
||||||
|
"Tuple",
|
||||||
|
other.type_name().tagged(tag),
|
||||||
|
)),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
fn deserialize_tuple_struct<V>(
|
fn deserialize_tuple_struct<V>(
|
||||||
self,
|
self,
|
||||||
|
@ -270,6 +335,42 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut ConfigDeserializer<'de> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct SeqDeserializer<'a, 'de: 'a, I: Iterator<Item=Tagged<Value>>> {
|
||||||
|
de: &'a mut ConfigDeserializer<'de>,
|
||||||
|
vals: I,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, 'de: 'a, I: Iterator<Item=Tagged<Value>>> SeqDeserializer<'a, 'de, I> {
|
||||||
|
fn new(de: &'a mut ConfigDeserializer<'de>, vals: I) -> Self {
|
||||||
|
SeqDeserializer {
|
||||||
|
de,
|
||||||
|
vals,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, 'de: 'a, I: Iterator<Item=Tagged<Value>>> de::SeqAccess<'de> for SeqDeserializer<'a, 'de, I> {
|
||||||
|
type Error = ShellError;
|
||||||
|
|
||||||
|
fn next_element_seed<T>(&mut self, seed: T) -> Result<Option<T::Value>, Self::Error>
|
||||||
|
where
|
||||||
|
T: de::DeserializeSeed<'de>,
|
||||||
|
{
|
||||||
|
let next = if let Some(next) = self.vals.next() {
|
||||||
|
next
|
||||||
|
} else {
|
||||||
|
return Ok(None);
|
||||||
|
};
|
||||||
|
|
||||||
|
self.de.push_val(next);
|
||||||
|
seed.deserialize(&mut *self.de).map(Some)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn size_hint(&self) -> Option<usize> {
|
||||||
|
return self.vals.size_hint().1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
struct StructDeserializer<'a, 'de: 'a> {
|
struct StructDeserializer<'a, 'de: 'a> {
|
||||||
de: &'a mut ConfigDeserializer<'de>,
|
de: &'a mut ConfigDeserializer<'de>,
|
||||||
fields: &'static [&'static str],
|
fields: &'static [&'static str],
|
||||||
|
|
Loading…
Reference in a new issue