Support for all custom value operations on plugin custom values (#12088)

# Description

Adds support for the following operations on plugin custom values, in
addition to `to_base_value` which was already present:

- `follow_path_int()`
- `follow_path_string()`
- `partial_cmp()`
- `operation()`
- `Drop` (notification, if opted into with
`CustomValue::notify_plugin_on_drop`)

There are additionally customizable methods within the `Plugin` and
`StreamingPlugin` traits for implementing these functions in a way that
requires access to the plugin state, as a registered handle model such
as might be used in a dataframes plugin would.

`Value::append` was also changed to handle custom values correctly.

# User-Facing Changes

- Signature of `CustomValue::follow_path_string` and
`CustomValue::follow_path_int` changed to give access to the span of the
custom value itself, useful for some errors.
- Plugins using custom values have to be recompiled because the engine
will try to do custom value operations that aren't supported
- Plugins can do more things 🎉 

# Tests + Formatting
Tests were added for all of the new custom values functionality.

- 🟢 `toolkit fmt`
- 🟢 `toolkit clippy`
- 🟢 `toolkit test`
- 🟢 `toolkit test stdlib`

# After Submitting
- [ ] Document protocol reference `CustomValueOp` variants:
  - [ ] `FollowPathInt`
  - [ ] `FollowPathString`
  - [ ] `PartialCmp`
  - [ ] `Operation`
  - [ ] `Dropped`
- [ ] Document `notify_on_drop` optional field in `PluginCustomValue`
This commit is contained in:
Devyn Cairns 2024-03-12 02:37:08 -07:00 committed by GitHub
parent 8a250d2e08
commit 73f3c0b60b
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
19 changed files with 1065 additions and 156 deletions

View file

@ -34,13 +34,23 @@ impl CustomValue for NuDataFrame {
self self
} }
fn follow_path_int(&self, count: usize, span: Span) -> Result<Value, ShellError> { fn follow_path_int(
self.get_value(count, span) &self,
_self_span: Span,
count: usize,
path_span: Span,
) -> Result<Value, ShellError> {
self.get_value(count, path_span)
} }
fn follow_path_string(&self, column_name: String, span: Span) -> Result<Value, ShellError> { fn follow_path_string(
let column = self.column(&column_name, span)?; &self,
Ok(column.into_value(span)) _self_span: Span,
column_name: String,
path_span: Span,
) -> Result<Value, ShellError> {
let column = self.column(&column_name, path_span)?;
Ok(column.into_value(path_span))
} }
fn partial_cmp(&self, other: &Value) -> Option<std::cmp::Ordering> { fn partial_cmp(&self, other: &Value) -> Option<std::cmp::Ordering> {

View file

@ -372,19 +372,29 @@ impl CustomValue for SQLiteDatabase {
self self
} }
fn follow_path_int(&self, _count: usize, span: Span) -> Result<Value, ShellError> { fn follow_path_int(
&self,
_self_span: Span,
_index: usize,
path_span: Span,
) -> Result<Value, ShellError> {
// In theory we could support this, but tables don't have an especially well-defined order // In theory we could support this, but tables don't have an especially well-defined order
Err(ShellError::IncompatiblePathAccess { type_name: "SQLite databases do not support integer-indexed access. Try specifying a table name instead".into(), span }) Err(ShellError::IncompatiblePathAccess { type_name: "SQLite databases do not support integer-indexed access. Try specifying a table name instead".into(), span: path_span })
} }
fn follow_path_string(&self, _column_name: String, span: Span) -> Result<Value, ShellError> { fn follow_path_string(
let db = open_sqlite_db(&self.path, span)?; &self,
_self_span: Span,
_column_name: String,
path_span: Span,
) -> Result<Value, ShellError> {
let db = open_sqlite_db(&self.path, path_span)?;
read_single_table(db, _column_name, span, self.ctrlc.clone()).map_err(|e| { read_single_table(db, _column_name, path_span, self.ctrlc.clone()).map_err(|e| {
ShellError::GenericError { ShellError::GenericError {
error: "Failed to read from SQLite database".into(), error: "Failed to read from SQLite database".into(),
msg: e.to_string(), msg: e.to_string(),
span: Some(span), span: Some(path_span),
help: None, help: None,
inner: vec![], inner: vec![],
} }

View file

@ -16,7 +16,7 @@ nu-protocol = { path = "../nu-protocol", version = "0.91.1" }
bincode = "1.3" bincode = "1.3"
rmp-serde = "1.1" rmp-serde = "1.1"
serde = { version = "1.0" } serde = "1.0"
serde_json = { workspace = true } serde_json = { workspace = true }
log = "0.4" log = "0.4"
miette = { workspace = true } miette = { workspace = true }

View file

@ -12,8 +12,8 @@ use nu_protocol::{
use crate::{ use crate::{
protocol::{ protocol::{
CallInfo, CustomValueOp, EngineCall, EngineCallId, EngineCallResponse, PluginCall, CallInfo, CustomValueOp, EngineCall, EngineCallId, EngineCallResponse, Ordering,
PluginCallId, PluginCallResponse, PluginCustomValue, PluginInput, PluginOption, PluginCall, PluginCallId, PluginCallResponse, PluginCustomValue, PluginInput, PluginOption,
ProtocolInfo, ProtocolInfo,
}, },
LabeledError, PluginOutput, LabeledError, PluginOutput,
@ -683,6 +683,16 @@ impl EngineInterface {
self.write(PluginOutput::Option(PluginOption::GcDisabled(disabled)))?; self.write(PluginOutput::Option(PluginOption::GcDisabled(disabled)))?;
self.flush() self.flush()
} }
/// Write a call response of [`Ordering`], for `partial_cmp`.
pub(crate) fn write_ordering(
&self,
ordering: Option<impl Into<Ordering>>,
) -> Result<(), ShellError> {
let response = PluginCallResponse::Ordering(ordering.map(|o| o.into()));
self.write(PluginOutput::CallResponse(self.context()?, response))?;
self.flush()
}
} }
impl Interface for EngineInterface { impl Interface for EngineInterface {

View file

@ -496,7 +496,7 @@ fn manager_consume_call_custom_value_op_forwards_to_receiver_with_context() -> R
op, op,
} => { } => {
assert_eq!(Some(32), engine.context); assert_eq!(Some(32), engine.context);
assert_eq!("TestCustomValue", custom_value.item.name); assert_eq!("TestCustomValue", custom_value.item.name());
assert!( assert!(
matches!(op, CustomValueOp::ToBaseValue), matches!(op, CustomValueOp::ToBaseValue),
"incorrect op: {op:?}" "incorrect op: {op:?}"
@ -600,11 +600,12 @@ fn manager_prepare_pipeline_data_embeds_deserialization_errors_in_streams() -> R
{ {
let manager = TestCase::new().engine(); let manager = TestCase::new().engine();
let invalid_custom_value = PluginCustomValue { let invalid_custom_value = PluginCustomValue::new(
name: "Invalid".into(), "Invalid".into(),
data: vec![0; 8], // should fail to decode to anything vec![0; 8], // should fail to decode to anything
source: None, false,
}; None,
);
let span = Span::new(20, 30); let span = Span::new(20, 30);
let data = manager.prepare_pipeline_data( let data = manager.prepare_pipeline_data(
@ -965,9 +966,9 @@ fn interface_prepare_pipeline_data_serializes_custom_values() -> Result<(), Shel
.expect("custom value is not a PluginCustomValue, probably not serialized"); .expect("custom value is not a PluginCustomValue, probably not serialized");
let expected = test_plugin_custom_value(); let expected = test_plugin_custom_value();
assert_eq!(expected.name, custom_value.name); assert_eq!(expected.name(), custom_value.name());
assert_eq!(expected.data, custom_value.data); assert_eq!(expected.data(), custom_value.data());
assert!(custom_value.source.is_none()); assert!(custom_value.source().is_none());
Ok(()) Ok(())
} }
@ -994,9 +995,9 @@ fn interface_prepare_pipeline_data_serializes_custom_values_in_streams() -> Resu
.expect("custom value is not a PluginCustomValue, probably not serialized"); .expect("custom value is not a PluginCustomValue, probably not serialized");
let expected = test_plugin_custom_value(); let expected = test_plugin_custom_value();
assert_eq!(expected.name, custom_value.name); assert_eq!(expected.name(), custom_value.name());
assert_eq!(expected.data, custom_value.data); assert_eq!(expected.data(), custom_value.data());
assert!(custom_value.source.is_none()); assert!(custom_value.source().is_none());
Ok(()) Ok(())
} }

View file

@ -6,15 +6,15 @@ use std::{
}; };
use nu_protocol::{ use nu_protocol::{
IntoInterruptiblePipelineData, ListStream, PipelineData, PluginSignature, ShellError, Spanned, ast::Operator, IntoInterruptiblePipelineData, IntoSpanned, ListStream, PipelineData,
Value, PluginSignature, ShellError, Span, Spanned, Value,
}; };
use crate::{ use crate::{
plugin::{context::PluginExecutionContext, gc::PluginGc, PluginSource}, plugin::{context::PluginExecutionContext, gc::PluginGc, PluginSource},
protocol::{ protocol::{
CallInfo, CustomValueOp, EngineCall, EngineCallId, EngineCallResponse, PluginCall, CallInfo, CustomValueOp, EngineCall, EngineCallId, EngineCallResponse, Ordering,
PluginCallId, PluginCallResponse, PluginCustomValue, PluginInput, PluginOption, PluginCall, PluginCallId, PluginCallResponse, PluginCustomValue, PluginInput, PluginOption,
PluginOutput, ProtocolInfo, StreamId, StreamMessage, PluginOutput, ProtocolInfo, StreamId, StreamMessage,
}, },
sequence::Sequence, sequence::Sequence,
@ -454,6 +454,9 @@ impl InterfaceManager for PluginInterfaceManager {
let response = match response { let response = match response {
PluginCallResponse::Error(err) => PluginCallResponse::Error(err), PluginCallResponse::Error(err) => PluginCallResponse::Error(err),
PluginCallResponse::Signature(sigs) => PluginCallResponse::Signature(sigs), PluginCallResponse::Signature(sigs) => PluginCallResponse::Signature(sigs),
PluginCallResponse::Ordering(ordering) => {
PluginCallResponse::Ordering(ordering)
}
PluginCallResponse::PipelineData(data) => { PluginCallResponse::PipelineData(data) => {
// If there's an error with initializing this stream, change it to a plugin // If there's an error with initializing this stream, change it to a plugin
// error response, but send it anyway // error response, but send it anyway
@ -804,22 +807,93 @@ impl PluginInterface {
} }
} }
/// Do a custom value op that expects a value response (i.e. most of them)
fn custom_value_op_expecting_value(
&self,
value: Spanned<PluginCustomValue>,
op: CustomValueOp,
) -> Result<Value, ShellError> {
let op_name = op.name();
let span = value.span;
let call = PluginCall::CustomValueOp(value, op);
match self.plugin_call(call, &None)? {
PluginCallResponse::PipelineData(out_data) => Ok(out_data.into_value(span)),
PluginCallResponse::Error(err) => Err(err.into()),
_ => Err(ShellError::PluginFailedToDecode {
msg: format!("Received unexpected response to custom value {op_name}() call"),
}),
}
}
/// Collapse a custom value to its base value. /// Collapse a custom value to its base value.
pub(crate) fn custom_value_to_base_value( pub(crate) fn custom_value_to_base_value(
&self, &self,
value: Spanned<PluginCustomValue>, value: Spanned<PluginCustomValue>,
) -> Result<Value, ShellError> { ) -> Result<Value, ShellError> {
let span = value.span; self.custom_value_op_expecting_value(value, CustomValueOp::ToBaseValue)
let call = PluginCall::CustomValueOp(value, CustomValueOp::ToBaseValue); }
/// Follow a numbered cell path on a custom value - e.g. `value.0`.
pub(crate) fn custom_value_follow_path_int(
&self,
value: Spanned<PluginCustomValue>,
index: Spanned<usize>,
) -> Result<Value, ShellError> {
self.custom_value_op_expecting_value(value, CustomValueOp::FollowPathInt(index))
}
/// Follow a named cell path on a custom value - e.g. `value.column`.
pub(crate) fn custom_value_follow_path_string(
&self,
value: Spanned<PluginCustomValue>,
column_name: Spanned<String>,
) -> Result<Value, ShellError> {
self.custom_value_op_expecting_value(value, CustomValueOp::FollowPathString(column_name))
}
/// Invoke comparison logic for custom values.
pub(crate) fn custom_value_partial_cmp(
&self,
value: PluginCustomValue,
mut other_value: Value,
) -> Result<Option<Ordering>, ShellError> {
PluginCustomValue::verify_source(&mut other_value, &self.state.source)?;
// Note: the protocol is always designed to have a span with the custom value, but this
// operation doesn't support one.
let call = PluginCall::CustomValueOp(
value.into_spanned(Span::unknown()),
CustomValueOp::PartialCmp(other_value),
);
match self.plugin_call(call, &None)? { match self.plugin_call(call, &None)? {
PluginCallResponse::PipelineData(out_data) => Ok(out_data.into_value(span)), PluginCallResponse::Ordering(ordering) => Ok(ordering),
PluginCallResponse::Error(err) => Err(err.into()), PluginCallResponse::Error(err) => Err(err.into()),
_ => Err(ShellError::PluginFailedToDecode { _ => Err(ShellError::PluginFailedToDecode {
msg: "Received unexpected response to plugin CustomValueOp::ToBaseValue call" msg: "Received unexpected response to custom value partial_cmp() call".into(),
.into(),
}), }),
} }
} }
/// Invoke functionality for an operator on a custom value.
pub(crate) fn custom_value_operation(
&self,
left: Spanned<PluginCustomValue>,
operator: Spanned<Operator>,
mut right: Value,
) -> Result<Value, ShellError> {
PluginCustomValue::verify_source(&mut right, &self.state.source)?;
self.custom_value_op_expecting_value(left, CustomValueOp::Operation(operator, right))
}
/// Notify the plugin about a dropped custom value.
pub(crate) fn custom_value_dropped(&self, value: PluginCustomValue) -> Result<(), ShellError> {
// Note: the protocol is always designed to have a span with the custom value, but this
// operation doesn't support one.
self.custom_value_op_expecting_value(
value.into_spanned(Span::unknown()),
CustomValueOp::Dropped,
)
.map(|_| ())
}
} }
/// Check that custom values in call arguments come from the right source /// Check that custom values in call arguments come from the right source

View file

@ -649,7 +649,7 @@ fn manager_prepare_pipeline_data_adds_source_to_values() -> Result<(), ShellErro
.downcast_ref() .downcast_ref()
.expect("custom value is not a PluginCustomValue"); .expect("custom value is not a PluginCustomValue");
if let Some(source) = &custom_value.source { if let Some(source) = custom_value.source() {
assert_eq!("test", source.name()); assert_eq!("test", source.name());
} else { } else {
panic!("source was not set"); panic!("source was not set");
@ -679,7 +679,7 @@ fn manager_prepare_pipeline_data_adds_source_to_list_streams() -> Result<(), She
.downcast_ref() .downcast_ref()
.expect("custom value is not a PluginCustomValue"); .expect("custom value is not a PluginCustomValue");
if let Some(source) = &custom_value.source { if let Some(source) = custom_value.source() {
assert_eq!("test", source.name()); assert_eq!("test", source.name());
} else { } else {
panic!("source was not set"); panic!("source was not set");
@ -1092,12 +1092,13 @@ fn interface_custom_value_to_base_value() -> Result<(), ShellError> {
fn normal_values(interface: &PluginInterface) -> Vec<Value> { fn normal_values(interface: &PluginInterface) -> Vec<Value> {
vec![ vec![
Value::test_int(5), Value::test_int(5),
Value::test_custom_value(Box::new(PluginCustomValue { Value::test_custom_value(Box::new(PluginCustomValue::new(
name: "SomeTest".into(), "SomeTest".into(),
data: vec![1, 2, 3], vec![1, 2, 3],
false,
// Has the same source, so it should be accepted // Has the same source, so it should be accepted
source: Some(interface.state.source.clone()), Some(interface.state.source.clone()),
})), ))),
] ]
} }
@ -1145,17 +1146,19 @@ fn bad_custom_values() -> Vec<Value> {
// Native custom value (not PluginCustomValue) should be rejected // Native custom value (not PluginCustomValue) should be rejected
Value::test_custom_value(Box::new(expected_test_custom_value())), Value::test_custom_value(Box::new(expected_test_custom_value())),
// Has no source, so it should be rejected // Has no source, so it should be rejected
Value::test_custom_value(Box::new(PluginCustomValue { Value::test_custom_value(Box::new(PluginCustomValue::new(
name: "SomeTest".into(), "SomeTest".into(),
data: vec![1, 2, 3], vec![1, 2, 3],
source: None, false,
})), None,
))),
// Has a different source, so it should be rejected // Has a different source, so it should be rejected
Value::test_custom_value(Box::new(PluginCustomValue { Value::test_custom_value(Box::new(PluginCustomValue::new(
name: "SomeTest".into(), "SomeTest".into(),
data: vec![1, 2, 3], vec![1, 2, 3],
source: Some(PluginSource::new_fake("pluto").into()), false,
})), Some(PluginSource::new_fake("pluto").into()),
))),
] ]
} }

View file

@ -1,6 +1,8 @@
use nu_engine::documentation::get_flags_section; use nu_engine::documentation::get_flags_section;
use nu_protocol::ast::Operator;
use std::cmp::Ordering;
use std::ffi::OsStr; use std::ffi::OsStr;
use std::sync::mpsc::TrySendError; use std::sync::mpsc::TrySendError;
use std::sync::{mpsc, Arc, Mutex}; use std::sync::{mpsc, Arc, Mutex};
@ -21,7 +23,9 @@ use std::os::unix::process::CommandExt;
#[cfg(windows)] #[cfg(windows)]
use std::os::windows::process::CommandExt; use std::os::windows::process::CommandExt;
use nu_protocol::{PipelineData, PluginSignature, ShellError, Spanned, Value}; use nu_protocol::{
CustomValue, IntoSpanned, PipelineData, PluginSignature, ShellError, Spanned, Value,
};
use self::gc::PluginGc; use self::gc::PluginGc;
@ -279,6 +283,112 @@ pub trait Plugin: Sync {
call: &EvaluatedCall, call: &EvaluatedCall,
input: &Value, input: &Value,
) -> Result<Value, LabeledError>; ) -> Result<Value, LabeledError>;
/// Collapse a custom value to plain old data.
///
/// The default implementation of this method just calls [`CustomValue::to_base_value`], but
/// the method can be implemented differently if accessing plugin state is desirable.
fn custom_value_to_base_value(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
) -> Result<Value, LabeledError> {
let _ = engine;
custom_value
.item
.to_base_value(custom_value.span)
.map_err(LabeledError::from)
}
/// Follow a numbered cell path on a custom value - e.g. `value.0`.
///
/// The default implementation of this method just calls [`CustomValue::follow_path_int`], but
/// the method can be implemented differently if accessing plugin state is desirable.
fn custom_value_follow_path_int(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
index: Spanned<usize>,
) -> Result<Value, LabeledError> {
let _ = engine;
custom_value
.item
.follow_path_int(custom_value.span, index.item, index.span)
.map_err(LabeledError::from)
}
/// Follow a named cell path on a custom value - e.g. `value.column`.
///
/// The default implementation of this method just calls [`CustomValue::follow_path_string`],
/// but the method can be implemented differently if accessing plugin state is desirable.
fn custom_value_follow_path_string(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
column_name: Spanned<String>,
) -> Result<Value, LabeledError> {
let _ = engine;
custom_value
.item
.follow_path_string(custom_value.span, column_name.item, column_name.span)
.map_err(LabeledError::from)
}
/// Implement comparison logic for custom values.
///
/// The default implementation of this method just calls [`CustomValue::partial_cmp`], but
/// the method can be implemented differently if accessing plugin state is desirable.
///
/// Note that returning an error here is unlikely to produce desired behavior, as `partial_cmp`
/// lacks a way to produce an error. At the moment the engine just logs the error, and the
/// comparison returns `None`.
fn custom_value_partial_cmp(
&self,
engine: &EngineInterface,
custom_value: Box<dyn CustomValue>,
other_value: Value,
) -> Result<Option<Ordering>, LabeledError> {
let _ = engine;
Ok(custom_value.partial_cmp(&other_value))
}
/// Implement functionality for an operator on a custom value.
///
/// The default implementation of this method just calls [`CustomValue::operation`], but
/// the method can be implemented differently if accessing plugin state is desirable.
fn custom_value_operation(
&self,
engine: &EngineInterface,
left: Spanned<Box<dyn CustomValue>>,
operator: Spanned<Operator>,
right: Value,
) -> Result<Value, LabeledError> {
let _ = engine;
left.item
.operation(left.span, operator.item, operator.span, &right)
.map_err(LabeledError::from)
}
/// Handle a notification that all copies of a custom value within the engine have been dropped.
///
/// This notification is only sent if [`CustomValue::notify_plugin_on_drop`] was true. Unlike
/// the other custom value handlers, a span is not provided.
///
/// Note that a new custom value is created each time it is sent to the engine - if you intend
/// to accept a custom value and send it back, you may need to implement some kind of unique
/// reference counting in your plugin, as you will receive multiple drop notifications even if
/// the data within is identical.
///
/// The default implementation does nothing. Any error generated here is unlikely to be visible
/// to the user, and will only show up in the engine's log output.
fn custom_value_dropped(
&self,
engine: &EngineInterface,
custom_value: Box<dyn CustomValue>,
) -> Result<(), LabeledError> {
let _ = (engine, custom_value);
Ok(())
}
} }
/// The streaming API for a Nushell plugin /// The streaming API for a Nushell plugin
@ -357,6 +467,112 @@ pub trait StreamingPlugin: Sync {
call: &EvaluatedCall, call: &EvaluatedCall,
input: PipelineData, input: PipelineData,
) -> Result<PipelineData, LabeledError>; ) -> Result<PipelineData, LabeledError>;
/// Collapse a custom value to plain old data.
///
/// The default implementation of this method just calls [`CustomValue::to_base_value`], but
/// the method can be implemented differently if accessing plugin state is desirable.
fn custom_value_to_base_value(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
) -> Result<Value, LabeledError> {
let _ = engine;
custom_value
.item
.to_base_value(custom_value.span)
.map_err(LabeledError::from)
}
/// Follow a numbered cell path on a custom value - e.g. `value.0`.
///
/// The default implementation of this method just calls [`CustomValue::follow_path_int`], but
/// the method can be implemented differently if accessing plugin state is desirable.
fn custom_value_follow_path_int(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
index: Spanned<usize>,
) -> Result<Value, LabeledError> {
let _ = engine;
custom_value
.item
.follow_path_int(custom_value.span, index.item, index.span)
.map_err(LabeledError::from)
}
/// Follow a named cell path on a custom value - e.g. `value.column`.
///
/// The default implementation of this method just calls [`CustomValue::follow_path_string`],
/// but the method can be implemented differently if accessing plugin state is desirable.
fn custom_value_follow_path_string(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
column_name: Spanned<String>,
) -> Result<Value, LabeledError> {
let _ = engine;
custom_value
.item
.follow_path_string(custom_value.span, column_name.item, column_name.span)
.map_err(LabeledError::from)
}
/// Implement comparison logic for custom values.
///
/// The default implementation of this method just calls [`CustomValue::partial_cmp`], but
/// the method can be implemented differently if accessing plugin state is desirable.
///
/// Note that returning an error here is unlikely to produce desired behavior, as `partial_cmp`
/// lacks a way to produce an error. At the moment the engine just logs the error, and the
/// comparison returns `None`.
fn custom_value_partial_cmp(
&self,
engine: &EngineInterface,
custom_value: Box<dyn CustomValue>,
other_value: Value,
) -> Result<Option<Ordering>, LabeledError> {
let _ = engine;
Ok(custom_value.partial_cmp(&other_value))
}
/// Implement functionality for an operator on a custom value.
///
/// The default implementation of this method just calls [`CustomValue::operation`], but
/// the method can be implemented differently if accessing plugin state is desirable.
fn custom_value_operation(
&self,
engine: &EngineInterface,
left: Spanned<Box<dyn CustomValue>>,
operator: Spanned<Operator>,
right: Value,
) -> Result<Value, LabeledError> {
let _ = engine;
left.item
.operation(left.span, operator.item, operator.span, &right)
.map_err(LabeledError::from)
}
/// Handle a notification that all copies of a custom value within the engine have been dropped.
///
/// This notification is only sent if [`CustomValue::notify_plugin_on_drop`] was true. Unlike
/// the other custom value handlers, a span is not provided.
///
/// Note that a new custom value is created each time it is sent to the engine - if you intend
/// to accept a custom value and send it back, you may need to implement some kind of unique
/// reference counting in your plugin, as you will receive multiple drop notifications even if
/// the data within is identical.
///
/// The default implementation does nothing. Any error generated here is unlikely to be visible
/// to the user, and will only show up in the engine's log output.
fn custom_value_dropped(
&self,
engine: &EngineInterface,
custom_value: Box<dyn CustomValue>,
) -> Result<(), LabeledError> {
let _ = (engine, custom_value);
Ok(())
}
} }
/// All [Plugin]s can be used as [StreamingPlugin]s, but input streams will be fully consumed /// All [Plugin]s can be used as [StreamingPlugin]s, but input streams will be fully consumed
@ -381,6 +597,59 @@ impl<T: Plugin> StreamingPlugin for T {
<Self as Plugin>::run(self, name, engine, call, &input_value) <Self as Plugin>::run(self, name, engine, call, &input_value)
.map(|value| PipelineData::Value(value, None)) .map(|value| PipelineData::Value(value, None))
} }
fn custom_value_to_base_value(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
) -> Result<Value, LabeledError> {
<Self as Plugin>::custom_value_to_base_value(self, engine, custom_value)
}
fn custom_value_follow_path_int(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
index: Spanned<usize>,
) -> Result<Value, LabeledError> {
<Self as Plugin>::custom_value_follow_path_int(self, engine, custom_value, index)
}
fn custom_value_follow_path_string(
&self,
engine: &EngineInterface,
custom_value: Spanned<Box<dyn CustomValue>>,
column_name: Spanned<String>,
) -> Result<Value, LabeledError> {
<Self as Plugin>::custom_value_follow_path_string(self, engine, custom_value, column_name)
}
fn custom_value_partial_cmp(
&self,
engine: &EngineInterface,
custom_value: Box<dyn CustomValue>,
other_value: Value,
) -> Result<Option<Ordering>, LabeledError> {
<Self as Plugin>::custom_value_partial_cmp(self, engine, custom_value, other_value)
}
fn custom_value_operation(
&self,
engine: &EngineInterface,
left: Spanned<Box<dyn CustomValue>>,
operator: Spanned<Operator>,
right: Value,
) -> Result<Value, LabeledError> {
<Self as Plugin>::custom_value_operation(self, engine, left, operator, right)
}
fn custom_value_dropped(
&self,
engine: &EngineInterface,
custom_value: Box<dyn CustomValue>,
) -> Result<(), LabeledError> {
<Self as Plugin>::custom_value_dropped(self, engine, custom_value)
}
} }
/// Function used to implement the communication protocol between /// Function used to implement the communication protocol between
@ -580,7 +849,7 @@ pub fn serve_plugin(plugin: &impl StreamingPlugin, encoder: impl PluginEncoder +
custom_value, custom_value,
op, op,
} => { } => {
try_or_report!(engine, custom_value_op(&engine, custom_value, op)); try_or_report!(engine, custom_value_op(plugin, &engine, custom_value, op));
} }
} }
} }
@ -591,22 +860,65 @@ pub fn serve_plugin(plugin: &impl StreamingPlugin, encoder: impl PluginEncoder +
} }
fn custom_value_op( fn custom_value_op(
plugin: &impl StreamingPlugin,
engine: &EngineInterface, engine: &EngineInterface,
custom_value: Spanned<PluginCustomValue>, custom_value: Spanned<PluginCustomValue>,
op: CustomValueOp, op: CustomValueOp,
) -> Result<(), ShellError> { ) -> Result<(), ShellError> {
let local_value = custom_value let local_value = custom_value
.item .item
.deserialize_to_custom_value(custom_value.span)?; .deserialize_to_custom_value(custom_value.span)?
.into_spanned(custom_value.span);
match op { match op {
CustomValueOp::ToBaseValue => { CustomValueOp::ToBaseValue => {
let result = local_value let result = plugin
.to_base_value(custom_value.span) .custom_value_to_base_value(engine, local_value)
.map(|value| PipelineData::Value(value, None)); .map(|value| PipelineData::Value(value, None));
engine engine
.write_response(result) .write_response(result)
.and_then(|writer| writer.write_background())?; .and_then(|writer| writer.write())
Ok(()) }
CustomValueOp::FollowPathInt(index) => {
let result = plugin
.custom_value_follow_path_int(engine, local_value, index)
.map(|value| PipelineData::Value(value, None));
engine
.write_response(result)
.and_then(|writer| writer.write())
}
CustomValueOp::FollowPathString(column_name) => {
let result = plugin
.custom_value_follow_path_string(engine, local_value, column_name)
.map(|value| PipelineData::Value(value, None));
engine
.write_response(result)
.and_then(|writer| writer.write())
}
CustomValueOp::PartialCmp(mut other_value) => {
PluginCustomValue::deserialize_custom_values_in(&mut other_value)?;
match plugin.custom_value_partial_cmp(engine, local_value.item, other_value) {
Ok(ordering) => engine.write_ordering(ordering),
Err(err) => engine
.write_response(Err(err))
.and_then(|writer| writer.write()),
}
}
CustomValueOp::Operation(operator, mut right) => {
PluginCustomValue::deserialize_custom_values_in(&mut right)?;
let result = plugin
.custom_value_operation(engine, local_value, operator, right)
.map(|value| PipelineData::Value(value, None));
engine
.write_response(result)
.and_then(|writer| writer.write())
}
CustomValueOp::Dropped => {
let result = plugin
.custom_value_dropped(engine, local_value.item)
.map(|_| PipelineData::Empty);
engine
.write_response(result)
.and_then(|writer| writer.write())
} }
} }
} }

View file

@ -10,8 +10,8 @@ pub(crate) mod test_util;
pub use evaluated_call::EvaluatedCall; pub use evaluated_call::EvaluatedCall;
use nu_protocol::{ use nu_protocol::{
engine::Closure, Config, PipelineData, PluginSignature, RawStream, ShellError, Span, Spanned, ast::Operator, engine::Closure, Config, PipelineData, PluginSignature, RawStream, ShellError,
Value, Span, Spanned, Value,
}; };
pub use plugin_custom_value::PluginCustomValue; pub use plugin_custom_value::PluginCustomValue;
pub use protocol_info::ProtocolInfo; pub use protocol_info::ProtocolInfo;
@ -131,6 +131,31 @@ pub enum PluginCall<D> {
pub enum CustomValueOp { pub enum CustomValueOp {
/// [`to_base_value()`](nu_protocol::CustomValue::to_base_value) /// [`to_base_value()`](nu_protocol::CustomValue::to_base_value)
ToBaseValue, ToBaseValue,
/// [`follow_path_int()`](nu_protocol::CustomValue::follow_path_int)
FollowPathInt(Spanned<usize>),
/// [`follow_path_string()`](nu_protocol::CustomValue::follow_path_string)
FollowPathString(Spanned<String>),
/// [`partial_cmp()`](nu_protocol::CustomValue::partial_cmp)
PartialCmp(Value),
/// [`operation()`](nu_protocol::CustomValue::operation)
Operation(Spanned<Operator>, Value),
/// Notify that the custom value has been dropped, if
/// [`notify_plugin_on_drop()`](nu_protocol::CustomValue::notify_plugin_on_drop) is true
Dropped,
}
impl CustomValueOp {
/// Get the name of the op, for error messages.
pub(crate) fn name(&self) -> &'static str {
match self {
CustomValueOp::ToBaseValue => "to_base_value",
CustomValueOp::FollowPathInt(_) => "follow_path_int",
CustomValueOp::FollowPathString(_) => "follow_path_string",
CustomValueOp::PartialCmp(_) => "partial_cmp",
CustomValueOp::Operation(_, _) => "operation",
CustomValueOp::Dropped => "dropped",
}
}
} }
/// Any data sent to the plugin /// Any data sent to the plugin
@ -306,6 +331,7 @@ impl From<ShellError> for LabeledError {
pub enum PluginCallResponse<D> { pub enum PluginCallResponse<D> {
Error(LabeledError), Error(LabeledError),
Signature(Vec<PluginSignature>), Signature(Vec<PluginSignature>),
Ordering(Option<Ordering>),
PipelineData(D), PipelineData(D),
} }
@ -330,6 +356,34 @@ pub enum PluginOption {
GcDisabled(bool), GcDisabled(bool),
} }
/// This is just a serializable version of [std::cmp::Ordering], and can be converted 1:1
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
pub enum Ordering {
Less,
Equal,
Greater,
}
impl From<std::cmp::Ordering> for Ordering {
fn from(value: std::cmp::Ordering) -> Self {
match value {
std::cmp::Ordering::Less => Ordering::Less,
std::cmp::Ordering::Equal => Ordering::Equal,
std::cmp::Ordering::Greater => Ordering::Greater,
}
}
}
impl From<Ordering> for std::cmp::Ordering {
fn from(value: Ordering) -> Self {
match value {
Ordering::Less => std::cmp::Ordering::Less,
Ordering::Equal => std::cmp::Ordering::Equal,
Ordering::Greater => std::cmp::Ordering::Greater,
}
}
}
/// Information received from the plugin /// Information received from the plugin
/// ///
/// Note: exported for internal use, not public. /// Note: exported for internal use, not public.

View file

@ -1,9 +1,9 @@
use std::sync::Arc; use std::{cmp::Ordering, sync::Arc};
use nu_protocol::{CustomValue, ShellError, Span, Spanned, Value}; use nu_protocol::{ast::Operator, CustomValue, IntoSpanned, ShellError, Span, Value};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::plugin::PluginSource; use crate::plugin::{PluginInterface, PluginSource};
#[cfg(test)] #[cfg(test)]
mod tests; mod tests;
@ -22,41 +22,171 @@ mod tests;
/// values sent matches the plugin it is being sent to. /// values sent matches the plugin it is being sent to.
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize)]
pub struct PluginCustomValue { pub struct PluginCustomValue {
/// The name of the custom value as defined by the plugin (`value_string()`) #[serde(flatten)]
pub name: String, shared: SerdeArc<SharedContent>,
/// The bincoded representation of the custom value on the plugin side
pub data: Vec<u8>,
/// Which plugin the custom value came from. This is not defined on the plugin side. The engine /// Which plugin the custom value came from. This is not defined on the plugin side. The engine
/// side is responsible for maintaining it, and it is not sent over the serialization boundary. /// side is responsible for maintaining it, and it is not sent over the serialization boundary.
#[serde(skip, default)] #[serde(skip, default)]
pub(crate) source: Option<Arc<PluginSource>>, source: Option<Arc<PluginSource>>,
}
/// Content shared across copies of a plugin custom value.
#[derive(Clone, Debug, Serialize, Deserialize)]
struct SharedContent {
/// The name of the custom value as defined by the plugin (`value_string()`)
name: String,
/// The bincoded representation of the custom value on the plugin side
data: Vec<u8>,
/// True if the custom value should notify the source if all copies of it are dropped.
///
/// This is not serialized if `false`, since most custom values don't need it.
#[serde(default, skip_serializing_if = "is_false")]
notify_on_drop: bool,
}
fn is_false(b: &bool) -> bool {
!b
} }
#[typetag::serde] #[typetag::serde]
impl CustomValue for PluginCustomValue { impl CustomValue for PluginCustomValue {
fn clone_value(&self, span: nu_protocol::Span) -> nu_protocol::Value { fn clone_value(&self, span: Span) -> Value {
Value::custom_value(Box::new(self.clone()), span) Value::custom_value(Box::new(self.clone()), span)
} }
fn value_string(&self) -> String { fn value_string(&self) -> String {
self.name.clone() self.name().to_owned()
} }
fn to_base_value( fn to_base_value(&self, span: Span) -> Result<Value, ShellError> {
self.get_plugin(Some(span), "get base value")?
.custom_value_to_base_value(self.clone().into_spanned(span))
}
fn follow_path_int(
&self, &self,
span: nu_protocol::Span, self_span: Span,
) -> Result<nu_protocol::Value, nu_protocol::ShellError> { index: usize,
path_span: Span,
) -> Result<Value, ShellError> {
self.get_plugin(Some(self_span), "follow cell path")?
.custom_value_follow_path_int(
self.clone().into_spanned(self_span),
index.into_spanned(path_span),
)
}
fn follow_path_string(
&self,
self_span: Span,
column_name: String,
path_span: Span,
) -> Result<Value, ShellError> {
self.get_plugin(Some(self_span), "follow cell path")?
.custom_value_follow_path_string(
self.clone().into_spanned(self_span),
column_name.into_spanned(path_span),
)
}
fn partial_cmp(&self, other: &Value) -> Option<Ordering> {
self.get_plugin(Some(other.span()), "perform comparison")
.and_then(|plugin| {
// We're passing Span::unknown() here because we don't have one, and it probably
// shouldn't matter here and is just a consequence of the API
plugin.custom_value_partial_cmp(self.clone(), other.clone())
})
.unwrap_or_else(|err| {
// We can't do anything with the error other than log it.
log::warn!(
"Error in partial_cmp on plugin custom value (source={source:?}): {err}",
source = self.source
);
None
})
.map(|ordering| ordering.into())
}
fn operation(
&self,
lhs_span: Span,
operator: Operator,
op_span: Span,
right: &Value,
) -> Result<Value, ShellError> {
self.get_plugin(Some(lhs_span), "invoke operator")?
.custom_value_operation(
self.clone().into_spanned(lhs_span),
operator.into_spanned(op_span),
right.clone(),
)
}
fn as_any(&self) -> &dyn std::any::Any {
self
}
}
impl PluginCustomValue {
/// Create a new [`PluginCustomValue`].
pub(crate) fn new(
name: String,
data: Vec<u8>,
notify_on_drop: bool,
source: Option<Arc<PluginSource>>,
) -> PluginCustomValue {
PluginCustomValue {
shared: SerdeArc(Arc::new(SharedContent {
name,
data,
notify_on_drop,
})),
source,
}
}
/// The name of the custom value as defined by the plugin (`value_string()`)
pub fn name(&self) -> &str {
&self.shared.name
}
/// The bincoded representation of the custom value on the plugin side
pub fn data(&self) -> &[u8] {
&self.shared.data
}
/// True if the custom value should notify the source if all copies of it are dropped.
pub fn notify_on_drop(&self) -> bool {
self.shared.notify_on_drop
}
/// Which plugin the custom value came from. This is not defined on the plugin side. The engine
/// side is responsible for maintaining it, and it is not sent over the serialization boundary.
#[cfg(test)]
pub(crate) fn source(&self) -> &Option<Arc<PluginSource>> {
&self.source
}
/// Create the [`PluginCustomValue`] with the given source.
#[cfg(test)]
pub(crate) fn with_source(mut self, source: Option<Arc<PluginSource>>) -> PluginCustomValue {
self.source = source;
self
}
/// Helper to get the plugin to implement an op
fn get_plugin(&self, span: Option<Span>, for_op: &str) -> Result<PluginInterface, ShellError> {
let wrap_err = |err: ShellError| ShellError::GenericError { let wrap_err = |err: ShellError| ShellError::GenericError {
error: format!( error: format!(
"Unable to spawn plugin `{}` to get base value", "Unable to spawn plugin `{}` to {for_op}",
self.source self.source
.as_ref() .as_ref()
.map(|s| s.name()) .map(|s| s.name())
.unwrap_or("<unknown>") .unwrap_or("<unknown>")
), ),
msg: err.to_string(), msg: err.to_string(),
span: Some(span), span,
help: None, help: None,
inner: vec![err], inner: vec![err],
}; };
@ -69,25 +199,13 @@ impl CustomValue for PluginCustomValue {
// Envs probably should be passed here, but it's likely that the plugin is already running // Envs probably should be passed here, but it's likely that the plugin is already running
let empty_envs = std::iter::empty::<(&str, &str)>(); let empty_envs = std::iter::empty::<(&str, &str)>();
let plugin = source
.persistent(Some(span))
.and_then(|p| p.get(|| Ok(empty_envs)))
.map_err(wrap_err)?;
plugin source
.custom_value_to_base_value(Spanned { .persistent(span)
item: self.clone(), .and_then(|p| p.get(|| Ok(empty_envs)))
span,
})
.map_err(wrap_err) .map_err(wrap_err)
} }
fn as_any(&self) -> &dyn std::any::Any {
self
}
}
impl PluginCustomValue {
/// Serialize a custom value into a [`PluginCustomValue`]. This should only be done on the /// Serialize a custom value into a [`PluginCustomValue`]. This should only be done on the
/// plugin side. /// plugin side.
pub(crate) fn serialize_from_custom_value( pub(crate) fn serialize_from_custom_value(
@ -95,12 +213,9 @@ impl PluginCustomValue {
span: Span, span: Span,
) -> Result<PluginCustomValue, ShellError> { ) -> Result<PluginCustomValue, ShellError> {
let name = custom_value.value_string(); let name = custom_value.value_string();
let notify_on_drop = custom_value.notify_plugin_on_drop();
bincode::serialize(custom_value) bincode::serialize(custom_value)
.map(|data| PluginCustomValue { .map(|data| PluginCustomValue::new(name, data, notify_on_drop, None))
name,
data,
source: None,
})
.map_err(|err| ShellError::CustomValueFailedToEncode { .map_err(|err| ShellError::CustomValueFailedToEncode {
msg: err.to_string(), msg: err.to_string(),
span, span,
@ -113,7 +228,7 @@ impl PluginCustomValue {
&self, &self,
span: Span, span: Span,
) -> Result<Box<dyn CustomValue>, ShellError> { ) -> Result<Box<dyn CustomValue>, ShellError> {
bincode::deserialize::<Box<dyn CustomValue>>(&self.data).map_err(|err| { bincode::deserialize::<Box<dyn CustomValue>>(self.data()).map_err(|err| {
ShellError::CustomValueFailedToDecode { ShellError::CustomValueFailedToDecode {
msg: err.to_string(), msg: err.to_string(),
span, span,
@ -199,7 +314,7 @@ impl PluginCustomValue {
Ok(()) Ok(())
} else { } else {
Err(ShellError::CustomValueIncorrectForPlugin { Err(ShellError::CustomValueIncorrectForPlugin {
name: custom_value.name.clone(), 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()),
@ -363,3 +478,62 @@ impl PluginCustomValue {
} }
} }
} }
impl Drop for PluginCustomValue {
fn drop(&mut self) {
// If the custom value specifies notify_on_drop and this is the last copy, we need to let
// the plugin know about it if we can.
if self.source.is_some() && self.notify_on_drop() && Arc::strong_count(&self.shared) == 1 {
self.get_plugin(None, "drop")
// While notifying drop, we don't need a copy of the source
.and_then(|plugin| {
plugin.custom_value_dropped(PluginCustomValue {
shared: self.shared.clone(),
source: None,
})
})
.unwrap_or_else(|err| {
// We shouldn't do anything with the error except log it
let name = self.name();
log::warn!("Failed to notify drop of custom value ({name}): {err}")
});
}
}
}
/// A serializable `Arc`, to avoid having to have the serde `rc` feature enabled.
#[derive(Clone, Debug)]
#[repr(transparent)]
struct SerdeArc<T>(Arc<T>);
impl<T> Serialize for SerdeArc<T>
where
T: Serialize,
{
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
self.0.serialize(serializer)
}
}
impl<'de, T> Deserialize<'de> for SerdeArc<T>
where
T: Deserialize<'de>,
{
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
T::deserialize(deserializer).map(Arc::new).map(SerdeArc)
}
}
impl<T> std::ops::Deref for SerdeArc<T> {
type Target = Arc<T>;
fn deref(&self) -> &Self::Target {
&self.0
}
}

View file

@ -19,7 +19,7 @@ fn serialize_deserialize() -> Result<(), ShellError> {
let original_value = TestCustomValue(32); let original_value = TestCustomValue(32);
let span = Span::test_data(); let span = Span::test_data();
let serialized = PluginCustomValue::serialize_from_custom_value(&original_value, span)?; let serialized = PluginCustomValue::serialize_from_custom_value(&original_value, span)?;
assert_eq!(original_value.value_string(), serialized.name); assert_eq!(original_value.value_string(), serialized.name());
assert!(serialized.source.is_none()); assert!(serialized.source.is_none());
let deserialized = serialized.deserialize_to_custom_value(span)?; let deserialized = serialized.deserialize_to_custom_value(span)?;
let downcasted = deserialized let downcasted = deserialized
@ -36,8 +36,8 @@ fn expected_serialize_output() -> Result<(), ShellError> {
let span = Span::test_data(); let span = Span::test_data();
let serialized = PluginCustomValue::serialize_from_custom_value(&original_value, span)?; let serialized = PluginCustomValue::serialize_from_custom_value(&original_value, span)?;
assert_eq!( assert_eq!(
test_plugin_custom_value().data, test_plugin_custom_value().data(),
serialized.data, serialized.data(),
"The bincode configuration is probably different from what we expected. \ "The bincode configuration is probably different from what we expected. \
Fix test_plugin_custom_value() to match it" Fix test_plugin_custom_value() to match it"
); );
@ -417,8 +417,11 @@ fn serialize_in_root() -> Result<(), ShellError> {
let custom_value = val.as_custom_value()?; let custom_value = val.as_custom_value()?;
if let Some(plugin_custom_value) = custom_value.as_any().downcast_ref::<PluginCustomValue>() { if let Some(plugin_custom_value) = custom_value.as_any().downcast_ref::<PluginCustomValue>() {
assert_eq!("TestCustomValue", plugin_custom_value.name); assert_eq!("TestCustomValue", plugin_custom_value.name());
assert_eq!(test_plugin_custom_value().data, plugin_custom_value.data); assert_eq!(
test_plugin_custom_value().data(),
plugin_custom_value.data()
);
assert!(plugin_custom_value.source.is_none()); assert!(plugin_custom_value.source.is_none());
} else { } else {
panic!("Failed to downcast to PluginCustomValue"); panic!("Failed to downcast to PluginCustomValue");
@ -443,7 +446,8 @@ fn serialize_in_range() -> Result<(), ShellError> {
.downcast_ref() .downcast_ref()
.unwrap_or_else(|| panic!("{name} not PluginCustomValue")); .unwrap_or_else(|| panic!("{name} not PluginCustomValue"));
assert_eq!( assert_eq!(
"TestCustomValue", plugin_custom_value.name, "TestCustomValue",
plugin_custom_value.name(),
"{name} name not set correctly" "{name} name not set correctly"
); );
Ok(()) Ok(())
@ -465,7 +469,8 @@ fn serialize_in_record() -> Result<(), ShellError> {
.downcast_ref() .downcast_ref()
.unwrap_or_else(|| panic!("'{key}' not PluginCustomValue")); .unwrap_or_else(|| panic!("'{key}' not PluginCustomValue"));
assert_eq!( assert_eq!(
"TestCustomValue", plugin_custom_value.name, "TestCustomValue",
plugin_custom_value.name(),
"'{key}' name not set correctly" "'{key}' name not set correctly"
); );
Ok(()) Ok(())
@ -484,7 +489,8 @@ fn serialize_in_list() -> Result<(), ShellError> {
.downcast_ref() .downcast_ref()
.unwrap_or_else(|| panic!("[{index}] not PluginCustomValue")); .unwrap_or_else(|| panic!("[{index}] not PluginCustomValue"));
assert_eq!( assert_eq!(
"TestCustomValue", plugin_custom_value.name, "TestCustomValue",
plugin_custom_value.name(),
"[{index}] name not set correctly" "[{index}] name not set correctly"
); );
Ok(()) Ok(())
@ -506,7 +512,8 @@ fn serialize_in_closure() -> Result<(), ShellError> {
.downcast_ref() .downcast_ref()
.unwrap_or_else(|| panic!("[{index}] not PluginCustomValue")); .unwrap_or_else(|| panic!("[{index}] not PluginCustomValue"));
assert_eq!( assert_eq!(
"TestCustomValue", plugin_custom_value.name, "TestCustomValue",
plugin_custom_value.name(),
"[{index}] name not set correctly" "[{index}] name not set correctly"
); );
Ok(()) Ok(())

View file

@ -31,11 +31,7 @@ pub(crate) fn test_plugin_custom_value() -> PluginCustomValue {
let data = bincode::serialize(&expected_test_custom_value() as &dyn CustomValue) let data = bincode::serialize(&expected_test_custom_value() as &dyn CustomValue)
.expect("bincode serialization of the expected_test_custom_value() failed"); .expect("bincode serialization of the expected_test_custom_value() failed");
PluginCustomValue { PluginCustomValue::new("TestCustomValue".into(), data, false, None)
name: "TestCustomValue".into(),
data,
source: None,
}
} }
pub(crate) fn expected_test_custom_value() -> TestCustomValue { pub(crate) fn expected_test_custom_value() -> TestCustomValue {
@ -43,8 +39,5 @@ pub(crate) fn expected_test_custom_value() -> TestCustomValue {
} }
pub(crate) fn test_plugin_custom_value_with_source() -> PluginCustomValue { pub(crate) fn test_plugin_custom_value_with_source() -> PluginCustomValue {
PluginCustomValue { test_plugin_custom_value().with_source(Some(PluginSource::new_fake("test").into()))
source: Some(PluginSource::new_fake("test").into()),
..test_plugin_custom_value()
}
} }

View file

@ -176,11 +176,7 @@ macro_rules! generate_tests {
let custom_value_op = PluginCall::CustomValueOp( let custom_value_op = PluginCall::CustomValueOp(
Spanned { Spanned {
item: PluginCustomValue { item: PluginCustomValue::new("Foo".into(), data.clone(), false, None),
name: "Foo".into(),
data: data.clone(),
source: None,
},
span, span,
}, },
CustomValueOp::ToBaseValue, CustomValueOp::ToBaseValue,
@ -200,8 +196,8 @@ macro_rules! generate_tests {
match returned { match returned {
PluginInput::Call(2, PluginCall::CustomValueOp(val, op)) => { PluginInput::Call(2, PluginCall::CustomValueOp(val, op)) => {
assert_eq!("Foo", val.item.name); assert_eq!("Foo", val.item.name());
assert_eq!(data, val.item.data); assert_eq!(data, val.item.data());
assert_eq!(span, val.span); assert_eq!(span, val.span);
#[allow(unreachable_patterns)] #[allow(unreachable_patterns)]
match op { match op {
@ -320,11 +316,12 @@ macro_rules! generate_tests {
let span = Span::new(2, 30); let span = Span::new(2, 30);
let value = Value::custom_value( let value = Value::custom_value(
Box::new(PluginCustomValue { Box::new(PluginCustomValue::new(
name: name.into(), name.into(),
data: data.clone(), data.clone(),
source: None, true,
}), None,
)),
span, span,
); );
@ -354,8 +351,9 @@ macro_rules! generate_tests {
.as_any() .as_any()
.downcast_ref::<PluginCustomValue>() .downcast_ref::<PluginCustomValue>()
{ {
assert_eq!(name, plugin_val.name); assert_eq!(name, plugin_val.name());
assert_eq!(data, plugin_val.data); assert_eq!(data, plugin_val.data());
assert!(plugin_val.notify_on_drop());
} else { } else {
panic!("returned CustomValue is not a PluginCustomValue"); panic!("returned CustomValue is not a PluginCustomValue");
} }

View file

@ -26,18 +26,30 @@ pub trait CustomValue: fmt::Debug + Send + Sync {
fn as_any(&self) -> &dyn std::any::Any; fn as_any(&self) -> &dyn std::any::Any;
/// Follow cell path by numeric index (e.g. rows) /// Follow cell path by numeric index (e.g. rows)
fn follow_path_int(&self, _count: usize, span: Span) -> Result<Value, ShellError> { fn follow_path_int(
&self,
self_span: Span,
index: usize,
path_span: Span,
) -> Result<Value, ShellError> {
let _ = (self_span, index);
Err(ShellError::IncompatiblePathAccess { Err(ShellError::IncompatiblePathAccess {
type_name: self.value_string(), type_name: self.value_string(),
span, span: path_span,
}) })
} }
/// Follow cell path by string key (e.g. columns) /// Follow cell path by string key (e.g. columns)
fn follow_path_string(&self, _column_name: String, span: Span) -> Result<Value, ShellError> { fn follow_path_string(
&self,
self_span: Span,
column_name: String,
path_span: Span,
) -> Result<Value, ShellError> {
let _ = (self_span, column_name);
Err(ShellError::IncompatiblePathAccess { Err(ShellError::IncompatiblePathAccess {
type_name: self.value_string(), type_name: self.value_string(),
span, span: path_span,
}) })
} }
@ -54,11 +66,23 @@ pub trait CustomValue: fmt::Debug + Send + Sync {
/// Default impl raises [`ShellError::UnsupportedOperator`]. /// Default impl raises [`ShellError::UnsupportedOperator`].
fn operation( fn operation(
&self, &self,
_lhs_span: Span, lhs_span: Span,
operator: Operator, operator: Operator,
op: Span, op: Span,
_right: &Value, right: &Value,
) -> Result<Value, ShellError> { ) -> Result<Value, ShellError> {
let _ = (lhs_span, right);
Err(ShellError::UnsupportedOperator { operator, span: op }) Err(ShellError::UnsupportedOperator { operator, span: op })
} }
/// For custom values in plugins: return `true` here if you would like to be notified when all
/// copies of this custom value are dropped in the engine.
///
/// The notification will take place via
/// [`.custom_value_dropped()`](crate::StreamingPlugin::custom_value_dropped) on the plugin.
///
/// The default is `false`.
fn notify_plugin_on_drop(&self) -> bool {
false
}
} }

View file

@ -1106,8 +1106,9 @@ impl Value {
}); });
} }
} }
Value::CustomValue { val, .. } => { Value::CustomValue { ref val, .. } => {
current = match val.follow_path_int(*count, *origin_span) { current =
match val.follow_path_int(current.span(), *count, *origin_span) {
Ok(val) => val, Ok(val) => val,
Err(err) => { Err(err) => {
if *optional { if *optional {
@ -1249,8 +1250,22 @@ impl Value {
current = Value::list(list, span); current = Value::list(list, span);
} }
Value::CustomValue { val, .. } => { Value::CustomValue { ref val, .. } => {
current = val.follow_path_string(column_name.clone(), *origin_span)?; current = match val.follow_path_string(
current.span(),
column_name.clone(),
*origin_span,
) {
Ok(val) => val,
Err(err) => {
if *optional {
return Ok(Value::nothing(*origin_span));
// short-circuit
} else {
return Err(err);
}
}
}
} }
Value::Nothing { .. } if *optional => { Value::Nothing { .. } if *optional => {
return Ok(Value::nothing(*origin_span)); // short-circuit return Ok(Value::nothing(*origin_span)); // short-circuit
@ -2652,6 +2667,9 @@ impl Value {
val.extend(rhs); val.extend(rhs);
Ok(Value::binary(val, span)) Ok(Value::binary(val, span))
} }
(Value::CustomValue { val: lhs, .. }, rhs) => {
lhs.operation(self.span(), Operator::Math(Math::Append), op, rhs)
}
_ => Err(ShellError::OperatorMismatch { _ => Err(ShellError::OperatorMismatch {
op_span: op, op_span: op,
lhs_ty: self.get_type().to_string(), lhs_ty: self.get_type().to_string(),

View file

@ -1,7 +1,9 @@
use nu_protocol::{CustomValue, ShellError, Span, Value}; use std::cmp::Ordering;
use nu_protocol::{ast, CustomValue, ShellError, Span, Value};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
pub struct CoolCustomValue { pub struct CoolCustomValue {
pub(crate) cool: String, pub(crate) cool: String,
} }
@ -44,7 +46,7 @@ impl CoolCustomValue {
#[typetag::serde] #[typetag::serde]
impl CustomValue for CoolCustomValue { impl CustomValue for CoolCustomValue {
fn clone_value(&self, span: nu_protocol::Span) -> Value { fn clone_value(&self, span: Span) -> Value {
Value::custom_value(Box::new(self.clone()), span) Value::custom_value(Box::new(self.clone()), span)
} }
@ -52,13 +54,94 @@ impl CustomValue for CoolCustomValue {
self.typetag_name().to_string() self.typetag_name().to_string()
} }
fn to_base_value(&self, span: nu_protocol::Span) -> Result<Value, ShellError> { fn to_base_value(&self, span: Span) -> Result<Value, ShellError> {
Ok(Value::string( Ok(Value::string(
format!("I used to be a custom value! My data was ({})", self.cool), format!("I used to be a custom value! My data was ({})", self.cool),
span, span,
)) ))
} }
fn follow_path_int(
&self,
_self_span: Span,
index: usize,
path_span: Span,
) -> Result<Value, ShellError> {
if index == 0 {
Ok(Value::string(&self.cool, path_span))
} else {
Err(ShellError::AccessBeyondEnd {
max_idx: 0,
span: path_span,
})
}
}
fn follow_path_string(
&self,
self_span: Span,
column_name: String,
path_span: Span,
) -> Result<Value, ShellError> {
if column_name == "cool" {
Ok(Value::string(&self.cool, path_span))
} else {
Err(ShellError::CantFindColumn {
col_name: column_name,
span: path_span,
src_span: self_span,
})
}
}
fn partial_cmp(&self, other: &Value) -> Option<Ordering> {
if let Value::CustomValue { val, .. } = other {
val.as_any()
.downcast_ref()
.and_then(|other: &CoolCustomValue| PartialOrd::partial_cmp(self, other))
} else {
None
}
}
fn operation(
&self,
lhs_span: Span,
operator: ast::Operator,
op_span: Span,
right: &Value,
) -> Result<Value, ShellError> {
match operator {
// Append the string inside `cool`
ast::Operator::Math(ast::Math::Append) => {
if let Some(right) = right
.as_custom_value()
.ok()
.and_then(|c| c.as_any().downcast_ref::<CoolCustomValue>())
{
Ok(Value::custom_value(
Box::new(CoolCustomValue {
cool: format!("{}{}", self.cool, right.cool),
}),
op_span,
))
} else {
Err(ShellError::OperatorMismatch {
op_span,
lhs_ty: self.typetag_name().into(),
lhs_span,
rhs_ty: right.get_type().to_string(),
rhs_span: right.span(),
})
}
}
_ => Err(ShellError::UnsupportedOperator {
operator,
span: op_span,
}),
}
}
fn as_any(&self) -> &dyn std::any::Any { fn as_any(&self) -> &dyn std::any::Any {
self self
} }

View file

@ -0,0 +1,50 @@
use nu_protocol::{record, CustomValue, ShellError, Span, Value};
use serde::{Deserialize, Serialize};
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct DropCheck {
pub(crate) msg: String,
}
impl DropCheck {
pub(crate) fn new(msg: String) -> DropCheck {
DropCheck { msg }
}
pub(crate) fn into_value(self, span: Span) -> Value {
Value::custom_value(Box::new(self), span)
}
pub(crate) fn notify(&self) {
eprintln!("DropCheck was dropped: {}", self.msg);
}
}
#[typetag::serde]
impl CustomValue for DropCheck {
fn clone_value(&self, span: Span) -> Value {
self.clone().into_value(span)
}
fn value_string(&self) -> String {
"DropCheck".into()
}
fn to_base_value(&self, span: Span) -> Result<Value, ShellError> {
Ok(Value::record(
record! {
"msg" => Value::string(&self.msg, span)
},
span,
))
}
fn as_any(&self) -> &dyn std::any::Any {
self
}
fn notify_plugin_on_drop(&self) -> bool {
// This is what causes Nushell to let us know when the value is dropped
true
}
}

View file

@ -1,11 +1,14 @@
mod cool_custom_value; mod cool_custom_value;
mod drop_check;
mod second_custom_value; mod second_custom_value;
use cool_custom_value::CoolCustomValue; use cool_custom_value::CoolCustomValue;
use drop_check::DropCheck;
use second_custom_value::SecondCustomValue;
use nu_plugin::{serve_plugin, EngineInterface, MsgPackSerializer, Plugin}; use nu_plugin::{serve_plugin, EngineInterface, MsgPackSerializer, Plugin};
use nu_plugin::{EvaluatedCall, LabeledError}; use nu_plugin::{EvaluatedCall, LabeledError};
use nu_protocol::{Category, PluginSignature, ShellError, SyntaxShape, Value}; use nu_protocol::{Category, CustomValue, PluginSignature, ShellError, SyntaxShape, Value};
use second_custom_value::SecondCustomValue;
struct CustomValuePlugin; struct CustomValuePlugin;
@ -34,6 +37,10 @@ impl Plugin for CustomValuePlugin {
"the custom value to update", "the custom value to update",
) )
.category(Category::Experimental), .category(Category::Experimental),
PluginSignature::build("custom-value drop-check")
.usage("Generates a custom value that prints a message when dropped")
.required("msg", SyntaxShape::String, "the message to print on drop")
.category(Category::Experimental),
] ]
} }
@ -49,6 +56,7 @@ impl Plugin for CustomValuePlugin {
"custom-value generate2" => self.generate2(engine, call), "custom-value generate2" => self.generate2(engine, call),
"custom-value update" => self.update(call, input), "custom-value update" => self.update(call, input),
"custom-value update-arg" => self.update(call, &call.req(0)?), "custom-value update-arg" => self.update(call, &call.req(0)?),
"custom-value drop-check" => self.drop_check(call),
_ => Err(LabeledError { _ => Err(LabeledError {
label: "Plugin call with wrong name signature".into(), label: "Plugin call with wrong name signature".into(),
msg: "the signature used to call the plugin does not match any name in the plugin signature vector".into(), msg: "the signature used to call the plugin does not match any name in the plugin signature vector".into(),
@ -56,6 +64,18 @@ impl Plugin for CustomValuePlugin {
}), }),
} }
} }
fn custom_value_dropped(
&self,
_engine: &EngineInterface,
custom_value: Box<dyn CustomValue>,
) -> Result<(), LabeledError> {
// This is how we implement our drop behavior for DropCheck.
if let Some(drop_check) = custom_value.as_any().downcast_ref::<DropCheck>() {
drop_check.notify();
}
Ok(())
}
} }
impl CustomValuePlugin { impl CustomValuePlugin {
@ -101,6 +121,10 @@ impl CustomValuePlugin {
} }
.into()) .into())
} }
fn drop_check(&self, call: &EvaluatedCall) -> Result<Value, LabeledError> {
Ok(DropCheck::new(call.req(0)?).into_value(call.head))
}
} }
fn main() { fn main() {

View file

@ -79,6 +79,57 @@ fn can_get_describe_plugin_custom_values() {
assert_eq!(actual.out, "CoolCustomValue"); assert_eq!(actual.out, "CoolCustomValue");
} }
#[test]
fn can_get_plugin_custom_value_int_cell_path() {
let actual = nu_with_plugins!(
cwd: "tests",
plugin: ("nu_plugin_custom_values"),
"(custom-value generate).0"
);
assert_eq!(actual.out, "abc");
}
#[test]
fn can_get_plugin_custom_value_string_cell_path() {
let actual = nu_with_plugins!(
cwd: "tests",
plugin: ("nu_plugin_custom_values"),
"(custom-value generate).cool"
);
assert_eq!(actual.out, "abc");
}
#[test]
fn can_sort_plugin_custom_values() {
let actual = nu_with_plugins!(
cwd: "tests",
plugin: ("nu_plugin_custom_values"),
"[(custom-value generate | custom-value update) (custom-value generate)] | sort | each { print } | ignore"
);
assert_eq!(
actual.out,
"I used to be a custom value! My data was (abc)\
I used to be a custom value! My data was (abcxyz)"
);
}
#[test]
fn can_append_plugin_custom_values() {
let actual = nu_with_plugins!(
cwd: "tests",
plugin: ("nu_plugin_custom_values"),
"(custom-value generate) ++ (custom-value generate)"
);
assert_eq!(
actual.out,
"I used to be a custom value! My data was (abcabc)"
);
}
// There are currently no custom values defined by the engine that aren't hidden behind an extra // There are currently no custom values defined by the engine that aren't hidden behind an extra
// feature // feature
#[cfg(feature = "sqlite")] #[cfg(feature = "sqlite")]
@ -116,3 +167,16 @@ fn fails_if_passing_custom_values_across_plugins() {
.err .err
.contains("the `inc` plugin does not support this kind of value")); .contains("the `inc` plugin does not support this kind of value"));
} }
#[test]
fn drop_check_custom_value_prints_message_on_drop() {
let actual = nu_with_plugins!(
cwd: "tests",
plugin: ("nu_plugin_custom_values"),
// We build an array with the value copied twice to verify that it only gets dropped once
"do { |v| [$v $v] } (custom-value drop-check 'Hello') | ignore"
);
assert_eq!(actual.err, "DropCheck was dropped: Hello\n");
assert!(actual.status.success());
}