mirror of
https://github.com/nushell/nushell
synced 2025-01-14 22:24:54 +00:00
Add Value::recurse_mut()
to save duplicated code in PluginCustomValue
(#12218)
# Description We do a lot of visiting contained values in the serialization / validity functions within `PluginCustomValue` utils. This adds `Value::recurse_mut()` which wraps up most of that logic into something that can be reused.
This commit is contained in:
parent
c7e0d4b1e5
commit
1cb5221f01
2 changed files with 146 additions and 202 deletions
|
@ -1,4 +1,4 @@
|
||||||
use std::{cmp::Ordering, sync::Arc};
|
use std::{cmp::Ordering, convert::Infallible, sync::Arc};
|
||||||
|
|
||||||
use nu_protocol::{ast::Operator, CustomValue, IntoSpanned, ShellError, Span, Value};
|
use nu_protocol::{ast::Operator, CustomValue, IntoSpanned, ShellError, Span, Value};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
@ -238,57 +238,27 @@ impl PluginCustomValue {
|
||||||
|
|
||||||
/// Add a [`PluginSource`] to all [`PluginCustomValue`]s within a value, recursively.
|
/// Add a [`PluginSource`] to all [`PluginCustomValue`]s within a value, recursively.
|
||||||
pub(crate) fn add_source(value: &mut Value, source: &Arc<PluginSource>) {
|
pub(crate) fn add_source(value: &mut Value, source: &Arc<PluginSource>) {
|
||||||
let span = value.span();
|
// This can't cause an error.
|
||||||
match value {
|
let _: Result<(), Infallible> = value.recurse_mut(&mut |value| {
|
||||||
// Set source on custom value
|
let span = value.span();
|
||||||
Value::CustomValue { ref val, .. } => {
|
match value {
|
||||||
if let Some(custom_value) = val.as_any().downcast_ref::<PluginCustomValue>() {
|
// Set source on custom value
|
||||||
// Since there's no `as_mut_any()`, we have to copy the whole thing
|
Value::CustomValue { ref val, .. } => {
|
||||||
let mut custom_value = custom_value.clone();
|
if let Some(custom_value) = val.as_any().downcast_ref::<PluginCustomValue>() {
|
||||||
custom_value.source = Some(source.clone());
|
// Since there's no `as_mut_any()`, we have to copy the whole thing
|
||||||
*value = Value::custom_value(Box::new(custom_value), span);
|
let mut custom_value = custom_value.clone();
|
||||||
|
custom_value.source = Some(source.clone());
|
||||||
|
*value = Value::custom_value(Box::new(custom_value), span);
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
// LazyRecord could generate other values, but we shouldn't be receiving it anyway
|
||||||
|
//
|
||||||
|
// It's better to handle this as a bug
|
||||||
|
Value::LazyRecord { .. } => unimplemented!("add_source for LazyRecord"),
|
||||||
|
_ => Ok(()),
|
||||||
}
|
}
|
||||||
// Any values that can contain other values need to be handled recursively
|
});
|
||||||
Value::Range { ref mut val, .. } => {
|
|
||||||
Self::add_source(&mut val.from, source);
|
|
||||||
Self::add_source(&mut val.to, source);
|
|
||||||
Self::add_source(&mut val.incr, source);
|
|
||||||
}
|
|
||||||
Value::Record { ref mut val, .. } => {
|
|
||||||
for (_, rec_value) in val.iter_mut() {
|
|
||||||
Self::add_source(rec_value, source);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Value::List { ref mut vals, .. } => {
|
|
||||||
for list_value in vals.iter_mut() {
|
|
||||||
Self::add_source(list_value, source);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Value::Closure { ref mut val, .. } => {
|
|
||||||
for (_, captured_value) in val.captures.iter_mut() {
|
|
||||||
Self::add_source(captured_value, source);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// All of these don't contain other values
|
|
||||||
Value::Bool { .. }
|
|
||||||
| Value::Int { .. }
|
|
||||||
| Value::Float { .. }
|
|
||||||
| Value::Filesize { .. }
|
|
||||||
| Value::Duration { .. }
|
|
||||||
| Value::Date { .. }
|
|
||||||
| Value::String { .. }
|
|
||||||
| Value::Glob { .. }
|
|
||||||
| Value::Block { .. }
|
|
||||||
| Value::Nothing { .. }
|
|
||||||
| Value::Error { .. }
|
|
||||||
| Value::Binary { .. }
|
|
||||||
| Value::CellPath { .. } => (),
|
|
||||||
// LazyRecord could generate other values, but we shouldn't be receiving it anyway
|
|
||||||
//
|
|
||||||
// It's better to handle this as a bug
|
|
||||||
Value::LazyRecord { .. } => unimplemented!("add_source for LazyRecord"),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Check that all [`CustomValue`]s present within the `value` are [`PluginCustomValue`]s that
|
/// Check that all [`CustomValue`]s present within the `value` are [`PluginCustomValue`]s that
|
||||||
|
@ -300,182 +270,102 @@ impl PluginCustomValue {
|
||||||
value: &mut Value,
|
value: &mut Value,
|
||||||
source: &PluginSource,
|
source: &PluginSource,
|
||||||
) -> Result<(), ShellError> {
|
) -> Result<(), ShellError> {
|
||||||
let span = value.span();
|
value.recurse_mut(&mut |value| {
|
||||||
match value {
|
let span = value.span();
|
||||||
// Set source on custom value
|
match value {
|
||||||
Value::CustomValue { val, .. } => {
|
// Set source on custom value
|
||||||
if let Some(custom_value) = val.as_any().downcast_ref::<PluginCustomValue>() {
|
Value::CustomValue { val, .. } => {
|
||||||
if custom_value
|
if let Some(custom_value) = val.as_any().downcast_ref::<PluginCustomValue>() {
|
||||||
.source
|
if custom_value
|
||||||
.as_ref()
|
.source
|
||||||
.map(|s| s.is_compatible(source))
|
.as_ref()
|
||||||
.unwrap_or(false)
|
.map(|s| s.is_compatible(source))
|
||||||
{
|
.unwrap_or(false)
|
||||||
Ok(())
|
{
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(ShellError::CustomValueIncorrectForPlugin {
|
||||||
|
name: custom_value.name().to_owned(),
|
||||||
|
span,
|
||||||
|
dest_plugin: source.name().to_owned(),
|
||||||
|
src_plugin: custom_value
|
||||||
|
.source
|
||||||
|
.as_ref()
|
||||||
|
.map(|s| s.name().to_owned()),
|
||||||
|
})
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
|
// Only PluginCustomValues can be sent
|
||||||
Err(ShellError::CustomValueIncorrectForPlugin {
|
Err(ShellError::CustomValueIncorrectForPlugin {
|
||||||
name: custom_value.name().to_owned(),
|
name: val.value_string(),
|
||||||
span,
|
span,
|
||||||
dest_plugin: source.name().to_owned(),
|
dest_plugin: source.name().to_owned(),
|
||||||
src_plugin: custom_value.source.as_ref().map(|s| s.name().to_owned()),
|
src_plugin: None,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
// Only PluginCustomValues can be sent
|
|
||||||
Err(ShellError::CustomValueIncorrectForPlugin {
|
|
||||||
name: val.value_string(),
|
|
||||||
span,
|
|
||||||
dest_plugin: source.name().to_owned(),
|
|
||||||
src_plugin: None,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
// LazyRecord would be a problem for us, since it could return something else the
|
||||||
|
// next time, and we have to collect it anyway to serialize it. Collect it in place,
|
||||||
|
// and then verify the source of the result
|
||||||
|
Value::LazyRecord { val, .. } => {
|
||||||
|
*value = val.collect()?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
_ => Ok(()),
|
||||||
}
|
}
|
||||||
// Any values that can contain other values need to be handled recursively
|
})
|
||||||
Value::Range { val, .. } => {
|
|
||||||
Self::verify_source(&mut val.from, source)?;
|
|
||||||
Self::verify_source(&mut val.to, source)?;
|
|
||||||
Self::verify_source(&mut val.incr, source)
|
|
||||||
}
|
|
||||||
Value::Record { ref mut val, .. } => val
|
|
||||||
.iter_mut()
|
|
||||||
.try_for_each(|(_, rec_value)| Self::verify_source(rec_value, source)),
|
|
||||||
Value::List { ref mut vals, .. } => vals
|
|
||||||
.iter_mut()
|
|
||||||
.try_for_each(|list_value| Self::verify_source(list_value, source)),
|
|
||||||
Value::Closure { ref mut val, .. } => val
|
|
||||||
.captures
|
|
||||||
.iter_mut()
|
|
||||||
.try_for_each(|(_, captured_value)| Self::verify_source(captured_value, source)),
|
|
||||||
// All of these don't contain other values
|
|
||||||
Value::Bool { .. }
|
|
||||||
| Value::Int { .. }
|
|
||||||
| Value::Float { .. }
|
|
||||||
| Value::Filesize { .. }
|
|
||||||
| Value::Duration { .. }
|
|
||||||
| Value::Date { .. }
|
|
||||||
| Value::String { .. }
|
|
||||||
| Value::Glob { .. }
|
|
||||||
| Value::Block { .. }
|
|
||||||
| Value::Nothing { .. }
|
|
||||||
| Value::Error { .. }
|
|
||||||
| Value::Binary { .. }
|
|
||||||
| Value::CellPath { .. } => Ok(()),
|
|
||||||
// LazyRecord would be a problem for us, since it could return something else the next
|
|
||||||
// time, and we have to collect it anyway to serialize it. Collect it in place, and then
|
|
||||||
// verify the source of the result
|
|
||||||
Value::LazyRecord { val, .. } => {
|
|
||||||
*value = val.collect()?;
|
|
||||||
Self::verify_source(value, source)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Convert all plugin-native custom values to [`PluginCustomValue`] within the given `value`,
|
/// Convert all plugin-native custom values to [`PluginCustomValue`] within the given `value`,
|
||||||
/// recursively. This should only be done on the plugin side.
|
/// recursively. This should only be done on the plugin side.
|
||||||
pub(crate) fn serialize_custom_values_in(value: &mut Value) -> Result<(), ShellError> {
|
pub(crate) fn serialize_custom_values_in(value: &mut Value) -> Result<(), ShellError> {
|
||||||
let span = value.span();
|
value.recurse_mut(&mut |value| {
|
||||||
match value {
|
let span = value.span();
|
||||||
Value::CustomValue { ref val, .. } => {
|
match value {
|
||||||
if val.as_any().downcast_ref::<PluginCustomValue>().is_some() {
|
Value::CustomValue { ref val, .. } => {
|
||||||
// Already a PluginCustomValue
|
if val.as_any().downcast_ref::<PluginCustomValue>().is_some() {
|
||||||
Ok(())
|
// Already a PluginCustomValue
|
||||||
} else {
|
Ok(())
|
||||||
let serialized = Self::serialize_from_custom_value(&**val, span)?;
|
} else {
|
||||||
*value = Value::custom_value(Box::new(serialized), span);
|
let serialized = Self::serialize_from_custom_value(&**val, span)?;
|
||||||
|
*value = Value::custom_value(Box::new(serialized), span);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Collect LazyRecord before proceeding
|
||||||
|
Value::LazyRecord { ref val, .. } => {
|
||||||
|
*value = val.collect()?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
_ => Ok(()),
|
||||||
}
|
}
|
||||||
// Any values that can contain other values need to be handled recursively
|
})
|
||||||
Value::Range { ref mut val, .. } => {
|
|
||||||
Self::serialize_custom_values_in(&mut val.from)?;
|
|
||||||
Self::serialize_custom_values_in(&mut val.to)?;
|
|
||||||
Self::serialize_custom_values_in(&mut val.incr)
|
|
||||||
}
|
|
||||||
Value::Record { ref mut val, .. } => val
|
|
||||||
.iter_mut()
|
|
||||||
.try_for_each(|(_, rec_value)| Self::serialize_custom_values_in(rec_value)),
|
|
||||||
Value::List { ref mut vals, .. } => vals
|
|
||||||
.iter_mut()
|
|
||||||
.try_for_each(Self::serialize_custom_values_in),
|
|
||||||
Value::Closure { ref mut val, .. } => val
|
|
||||||
.captures
|
|
||||||
.iter_mut()
|
|
||||||
.map(|(_, captured_value)| captured_value)
|
|
||||||
.try_for_each(Self::serialize_custom_values_in),
|
|
||||||
// All of these don't contain other values
|
|
||||||
Value::Bool { .. }
|
|
||||||
| Value::Int { .. }
|
|
||||||
| Value::Float { .. }
|
|
||||||
| Value::Filesize { .. }
|
|
||||||
| Value::Duration { .. }
|
|
||||||
| Value::Date { .. }
|
|
||||||
| Value::String { .. }
|
|
||||||
| Value::Glob { .. }
|
|
||||||
| Value::Block { .. }
|
|
||||||
| Value::Nothing { .. }
|
|
||||||
| Value::Error { .. }
|
|
||||||
| Value::Binary { .. }
|
|
||||||
| Value::CellPath { .. } => Ok(()),
|
|
||||||
// Collect any lazy records that exist and try again
|
|
||||||
Value::LazyRecord { val, .. } => {
|
|
||||||
*value = val.collect()?;
|
|
||||||
Self::serialize_custom_values_in(value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Convert all [`PluginCustomValue`]s to plugin-native custom values within the given `value`,
|
/// Convert all [`PluginCustomValue`]s to plugin-native custom values within the given `value`,
|
||||||
/// recursively. This should only be done on the plugin side.
|
/// recursively. This should only be done on the plugin side.
|
||||||
pub(crate) fn deserialize_custom_values_in(value: &mut Value) -> Result<(), ShellError> {
|
pub(crate) fn deserialize_custom_values_in(value: &mut Value) -> Result<(), ShellError> {
|
||||||
let span = value.span();
|
value.recurse_mut(&mut |value| {
|
||||||
match value {
|
let span = value.span();
|
||||||
Value::CustomValue { ref val, .. } => {
|
match value {
|
||||||
if let Some(val) = val.as_any().downcast_ref::<PluginCustomValue>() {
|
Value::CustomValue { ref val, .. } => {
|
||||||
let deserialized = val.deserialize_to_custom_value(span)?;
|
if let Some(val) = val.as_any().downcast_ref::<PluginCustomValue>() {
|
||||||
*value = Value::custom_value(deserialized, span);
|
let deserialized = val.deserialize_to_custom_value(span)?;
|
||||||
Ok(())
|
*value = Value::custom_value(deserialized, span);
|
||||||
} else {
|
Ok(())
|
||||||
// Already not a PluginCustomValue
|
} else {
|
||||||
|
// Already not a PluginCustomValue
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Collect LazyRecord before proceeding
|
||||||
|
Value::LazyRecord { ref val, .. } => {
|
||||||
|
*value = val.collect()?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
_ => Ok(()),
|
||||||
}
|
}
|
||||||
// Any values that can contain other values need to be handled recursively
|
})
|
||||||
Value::Range { ref mut val, .. } => {
|
|
||||||
Self::deserialize_custom_values_in(&mut val.from)?;
|
|
||||||
Self::deserialize_custom_values_in(&mut val.to)?;
|
|
||||||
Self::deserialize_custom_values_in(&mut val.incr)
|
|
||||||
}
|
|
||||||
Value::Record { ref mut val, .. } => val
|
|
||||||
.iter_mut()
|
|
||||||
.try_for_each(|(_, rec_value)| Self::deserialize_custom_values_in(rec_value)),
|
|
||||||
Value::List { ref mut vals, .. } => vals
|
|
||||||
.iter_mut()
|
|
||||||
.try_for_each(Self::deserialize_custom_values_in),
|
|
||||||
Value::Closure { ref mut val, .. } => val
|
|
||||||
.captures
|
|
||||||
.iter_mut()
|
|
||||||
.map(|(_, captured_value)| captured_value)
|
|
||||||
.try_for_each(Self::deserialize_custom_values_in),
|
|
||||||
// All of these don't contain other values
|
|
||||||
Value::Bool { .. }
|
|
||||||
| Value::Int { .. }
|
|
||||||
| Value::Float { .. }
|
|
||||||
| Value::Filesize { .. }
|
|
||||||
| Value::Duration { .. }
|
|
||||||
| Value::Date { .. }
|
|
||||||
| Value::String { .. }
|
|
||||||
| Value::Glob { .. }
|
|
||||||
| Value::Block { .. }
|
|
||||||
| Value::Nothing { .. }
|
|
||||||
| Value::Error { .. }
|
|
||||||
| Value::Binary { .. }
|
|
||||||
| Value::CellPath { .. } => Ok(()),
|
|
||||||
// Collect any lazy records that exist and try again
|
|
||||||
Value::LazyRecord { val, .. } => {
|
|
||||||
*value = val.collect()?;
|
|
||||||
Self::deserialize_custom_values_in(value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1828,6 +1828,60 @@ impl Value {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Visits all values contained within the value (including this value) with a mutable reference
|
||||||
|
/// given to the closure.
|
||||||
|
///
|
||||||
|
/// If the closure returns `Err`, the traversal will stop.
|
||||||
|
///
|
||||||
|
/// If collecting lazy records to check them as well is desirable, make sure to do it in your
|
||||||
|
/// closure. The traversal continues on whatever modifications you make during the closure.
|
||||||
|
/// Captures of closure values are currently visited, as they are values owned by the closure.
|
||||||
|
pub fn recurse_mut<E>(
|
||||||
|
&mut self,
|
||||||
|
f: &mut impl FnMut(&mut Value) -> Result<(), E>,
|
||||||
|
) -> Result<(), E> {
|
||||||
|
// Visit this value
|
||||||
|
f(self)?;
|
||||||
|
// Check for contained values
|
||||||
|
match self {
|
||||||
|
// Any values that can contain other values need to be handled recursively
|
||||||
|
Value::Range { ref mut val, .. } => {
|
||||||
|
val.from.recurse_mut(f)?;
|
||||||
|
val.to.recurse_mut(f)?;
|
||||||
|
val.incr.recurse_mut(f)
|
||||||
|
}
|
||||||
|
Value::Record { ref mut val, .. } => val
|
||||||
|
.iter_mut()
|
||||||
|
.try_for_each(|(_, rec_value)| rec_value.recurse_mut(f)),
|
||||||
|
Value::List { ref mut vals, .. } => vals
|
||||||
|
.iter_mut()
|
||||||
|
.try_for_each(|list_value| list_value.recurse_mut(f)),
|
||||||
|
// Closure captures are visited. Maybe these don't have to be if they are changed to
|
||||||
|
// more opaque references.
|
||||||
|
Value::Closure { ref mut val, .. } => val
|
||||||
|
.captures
|
||||||
|
.iter_mut()
|
||||||
|
.map(|(_, captured_value)| captured_value)
|
||||||
|
.try_for_each(|captured_value| captured_value.recurse_mut(f)),
|
||||||
|
// All of these don't contain other values
|
||||||
|
Value::Bool { .. }
|
||||||
|
| Value::Int { .. }
|
||||||
|
| Value::Float { .. }
|
||||||
|
| Value::Filesize { .. }
|
||||||
|
| Value::Duration { .. }
|
||||||
|
| Value::Date { .. }
|
||||||
|
| Value::String { .. }
|
||||||
|
| Value::Glob { .. }
|
||||||
|
| Value::Block { .. }
|
||||||
|
| Value::Nothing { .. }
|
||||||
|
| Value::Error { .. }
|
||||||
|
| Value::Binary { .. }
|
||||||
|
| Value::CellPath { .. } => Ok(()),
|
||||||
|
// These could potentially contain values, but we expect the closure to handle them
|
||||||
|
Value::LazyRecord { .. } | Value::CustomValue { .. } => Ok(()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Check if the content is empty
|
/// Check if the content is empty
|
||||||
pub fn is_empty(&self) -> bool {
|
pub fn is_empty(&self) -> bool {
|
||||||
match self {
|
match self {
|
||||||
|
|
Loading…
Reference in a new issue