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:
Devyn Cairns 2024-03-16 07:54:42 -07:00 committed by GitHub
parent c7e0d4b1e5
commit 1cb5221f01
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 146 additions and 202 deletions

View file

@ -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,6 +238,8 @@ 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>) {
// This can't cause an error.
let _: Result<(), Infallible> = value.recurse_mut(&mut |value| {
let span = value.span(); let span = value.span();
match value { match value {
// Set source on custom value // Set source on custom value
@ -248,47 +250,15 @@ impl PluginCustomValue {
custom_value.source = Some(source.clone()); custom_value.source = Some(source.clone());
*value = Value::custom_value(Box::new(custom_value), span); *value = Value::custom_value(Box::new(custom_value), span);
} }
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 // LazyRecord could generate other values, but we shouldn't be receiving it anyway
// //
// It's better to handle this as a bug // It's better to handle this as a bug
Value::LazyRecord { .. } => unimplemented!("add_source for LazyRecord"), Value::LazyRecord { .. } => unimplemented!("add_source for LazyRecord"),
_ => Ok(()),
} }
});
} }
/// 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,6 +270,7 @@ impl PluginCustomValue {
value: &mut Value, value: &mut Value,
source: &PluginSource, source: &PluginSource,
) -> Result<(), ShellError> { ) -> Result<(), ShellError> {
value.recurse_mut(&mut |value| {
let span = value.span(); let span = value.span();
match value { match value {
// Set source on custom value // Set source on custom value
@ -317,7 +288,10 @@ impl PluginCustomValue {
name: custom_value.name().to_owned(), name: custom_value.name().to_owned(),
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: custom_value
.source
.as_ref()
.map(|s| s.name().to_owned()),
}) })
} }
} else { } else {
@ -330,49 +304,22 @@ impl PluginCustomValue {
}) })
} }
} }
// Any values that can contain other values need to be handled recursively // LazyRecord would be a problem for us, since it could return something else the
Value::Range { val, .. } => { // next time, and we have to collect it anyway to serialize it. Collect it in place,
Self::verify_source(&mut val.from, source)?; // and then verify the source of the result
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::LazyRecord { val, .. } => {
*value = val.collect()?; *value = val.collect()?;
Self::verify_source(value, source) Ok(())
} }
_ => Ok(()),
} }
})
} }
/// 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> {
value.recurse_mut(&mut |value| {
let span = value.span(); let span = value.span();
match value { match value {
Value::CustomValue { ref val, .. } => { Value::CustomValue { ref val, .. } => {
@ -385,48 +332,20 @@ impl PluginCustomValue {
Ok(()) Ok(())
} }
} }
// Any values that can contain other values need to be handled recursively // Collect LazyRecord before proceeding
Value::Range { ref mut val, .. } => { Value::LazyRecord { ref 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()?; *value = val.collect()?;
Self::serialize_custom_values_in(value) Ok(())
} }
_ => Ok(()),
} }
})
} }
/// 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> {
value.recurse_mut(&mut |value| {
let span = value.span(); let span = value.span();
match value { match value {
Value::CustomValue { ref val, .. } => { Value::CustomValue { ref val, .. } => {
@ -439,43 +358,14 @@ impl PluginCustomValue {
Ok(()) Ok(())
} }
} }
// Any values that can contain other values need to be handled recursively // Collect LazyRecord before proceeding
Value::Range { ref mut val, .. } => { Value::LazyRecord { ref 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()?; *value = val.collect()?;
Self::deserialize_custom_values_in(value) Ok(())
} }
_ => Ok(()),
} }
})
} }
} }

View file

@ -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 {