mirror of
https://github.com/nushell/nushell
synced 2025-01-26 11:55:20 +00:00
Allow exporting extern-wrapped (#10025)
This commit is contained in:
parent
fe2c498a81
commit
cdf09abcc0
6 changed files with 115 additions and 17 deletions
|
@ -0,0 +1,56 @@
|
||||||
|
use nu_protocol::ast::Call;
|
||||||
|
use nu_protocol::engine::{Command, EngineState, Stack};
|
||||||
|
use nu_protocol::{Category, Example, PipelineData, ShellError, Signature, SyntaxShape, Type};
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct ExportExternWrapped;
|
||||||
|
|
||||||
|
impl Command for ExportExternWrapped {
|
||||||
|
fn name(&self) -> &str {
|
||||||
|
"export extern-wrapped"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn usage(&self) -> &str {
|
||||||
|
"Define an extern with a custom code block and export it from a module."
|
||||||
|
}
|
||||||
|
|
||||||
|
fn signature(&self) -> nu_protocol::Signature {
|
||||||
|
Signature::build("export extern-wrapped")
|
||||||
|
.input_output_types(vec![(Type::Nothing, Type::Nothing)])
|
||||||
|
.required("def_name", SyntaxShape::String, "definition name")
|
||||||
|
.required("params", SyntaxShape::Signature, "parameters")
|
||||||
|
.required("body", SyntaxShape::Block, "wrapper code block")
|
||||||
|
.category(Category::Core)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn extra_usage(&self) -> &str {
|
||||||
|
r#"This command is a parser keyword. For details, check:
|
||||||
|
https://www.nushell.sh/book/thinking_in_nu.html"#
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_parser_keyword(&self) -> bool {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run(
|
||||||
|
&self,
|
||||||
|
_engine_state: &EngineState,
|
||||||
|
_stack: &mut Stack,
|
||||||
|
_call: &Call,
|
||||||
|
_input: PipelineData,
|
||||||
|
) -> Result<PipelineData, ShellError> {
|
||||||
|
Ok(PipelineData::empty())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn examples(&self) -> Vec<Example> {
|
||||||
|
vec![Example {
|
||||||
|
description: "Export the signature for an external command",
|
||||||
|
example: r#"export extern-wrapped my-echo [...rest] { echo $rest }"#,
|
||||||
|
result: None,
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn search_terms(&self) -> Vec<&str> {
|
||||||
|
vec!["signature", "module", "declare"]
|
||||||
|
}
|
||||||
|
}
|
|
@ -15,6 +15,7 @@ mod export_const;
|
||||||
mod export_def;
|
mod export_def;
|
||||||
mod export_def_env;
|
mod export_def_env;
|
||||||
mod export_extern;
|
mod export_extern;
|
||||||
|
mod export_extern_wrapped;
|
||||||
mod export_module;
|
mod export_module;
|
||||||
mod export_use;
|
mod export_use;
|
||||||
mod extern_;
|
mod extern_;
|
||||||
|
@ -55,6 +56,7 @@ pub use export_const::ExportConst;
|
||||||
pub use export_def::ExportDef;
|
pub use export_def::ExportDef;
|
||||||
pub use export_def_env::ExportDefEnv;
|
pub use export_def_env::ExportDefEnv;
|
||||||
pub use export_extern::ExportExtern;
|
pub use export_extern::ExportExtern;
|
||||||
|
pub use export_extern_wrapped::ExportExternWrapped;
|
||||||
pub use export_module::ExportModule;
|
pub use export_module::ExportModule;
|
||||||
pub use export_use::ExportUse;
|
pub use export_use::ExportUse;
|
||||||
pub use extern_::Extern;
|
pub use extern_::Extern;
|
||||||
|
|
|
@ -33,6 +33,7 @@ pub fn create_default_context() -> EngineState {
|
||||||
ExportDef,
|
ExportDef,
|
||||||
ExportDefEnv,
|
ExportDefEnv,
|
||||||
ExportExtern,
|
ExportExtern,
|
||||||
|
ExportExternWrapped,
|
||||||
ExportUse,
|
ExportUse,
|
||||||
ExportModule,
|
ExportModule,
|
||||||
Extern,
|
Extern,
|
||||||
|
|
|
@ -49,6 +49,7 @@ pub const UNALIASABLE_PARSER_KEYWORDS: &[&[u8]] = &[
|
||||||
b"extern",
|
b"extern",
|
||||||
b"extern-wrapped",
|
b"extern-wrapped",
|
||||||
b"export extern",
|
b"export extern",
|
||||||
|
b"export extern-wrapped",
|
||||||
b"alias",
|
b"alias",
|
||||||
b"export alias",
|
b"export alias",
|
||||||
b"export-env",
|
b"export-env",
|
||||||
|
@ -211,7 +212,7 @@ pub fn parse_def_predecl(working_set: &mut StateWorkingSet, spans: &[Span]) {
|
||||||
}
|
}
|
||||||
|
|
||||||
signature.name = name.clone();
|
signature.name = name.clone();
|
||||||
//let decl = signature.predeclare();
|
|
||||||
let decl = KnownExternal {
|
let decl = KnownExternal {
|
||||||
name,
|
name,
|
||||||
usage: "run external command".into(),
|
usage: "run external command".into(),
|
||||||
|
@ -536,10 +537,8 @@ pub fn parse_extern(
|
||||||
// Checking that the function is used with the correct name
|
// Checking that the function is used with the correct name
|
||||||
// Maybe this is not necessary but it is a sanity check
|
// Maybe this is not necessary but it is a sanity check
|
||||||
|
|
||||||
let (name_span, split_id) = if spans.len() > 1
|
let (name_span, split_id) =
|
||||||
&& (working_set.get_span_contents(spans[0]) == b"export"
|
if spans.len() > 1 && (working_set.get_span_contents(spans[0]) == b"export") {
|
||||||
|| working_set.get_span_contents(spans[0]) == b"export-wrapped")
|
|
||||||
{
|
|
||||||
(spans[1], 2)
|
(spans[1], 2)
|
||||||
} else {
|
} else {
|
||||||
(spans[0], 1)
|
(spans[0], 1)
|
||||||
|
@ -548,7 +547,7 @@ pub fn parse_extern(
|
||||||
let extern_call = working_set.get_span_contents(name_span).to_vec();
|
let extern_call = working_set.get_span_contents(name_span).to_vec();
|
||||||
if extern_call != b"extern" && extern_call != b"extern-wrapped" {
|
if extern_call != b"extern" && extern_call != b"extern-wrapped" {
|
||||||
working_set.error(ParseError::UnknownState(
|
working_set.error(ParseError::UnknownState(
|
||||||
"internal error: Wrong call name for extern function".into(),
|
"internal error: Wrong call name for extern or extern-wrapped command".into(),
|
||||||
span(spans),
|
span(spans),
|
||||||
));
|
));
|
||||||
return garbage_pipeline(spans);
|
return garbage_pipeline(spans);
|
||||||
|
@ -1011,6 +1010,7 @@ pub fn parse_export_in_block(
|
||||||
}
|
}
|
||||||
b"export module" => parse_module(working_set, lite_command, None).0,
|
b"export module" => parse_module(working_set, lite_command, None).0,
|
||||||
b"export extern" => parse_extern(working_set, lite_command, None),
|
b"export extern" => parse_extern(working_set, lite_command, None),
|
||||||
|
b"export extern-wrapped" => parse_extern(working_set, lite_command, None),
|
||||||
_ => {
|
_ => {
|
||||||
working_set.error(ParseError::UnexpectedKeyword(
|
working_set.error(ParseError::UnexpectedKeyword(
|
||||||
String::from_utf8_lossy(&full_name).to_string(),
|
String::from_utf8_lossy(&full_name).to_string(),
|
||||||
|
@ -1193,13 +1193,15 @@ pub fn parse_export_in_module(
|
||||||
comments: lite_command.comments.clone(),
|
comments: lite_command.comments.clone(),
|
||||||
parts: spans[1..].to_vec(),
|
parts: spans[1..].to_vec(),
|
||||||
};
|
};
|
||||||
|
let extern_name = [b"export ", kw_name].concat();
|
||||||
|
|
||||||
let pipeline = parse_extern(working_set, &lite_command, Some(module_name));
|
let pipeline = parse_extern(working_set, &lite_command, Some(module_name));
|
||||||
|
|
||||||
let export_def_decl_id = if let Some(id) = working_set.find_decl(b"export extern") {
|
let export_def_decl_id = if let Some(id) = working_set.find_decl(&extern_name) {
|
||||||
id
|
id
|
||||||
} else {
|
} else {
|
||||||
working_set.error(ParseError::InternalError(
|
working_set.error(ParseError::InternalError(
|
||||||
"missing 'export extern' command".into(),
|
"missing 'export extern' or 'export extern-wrapped' command".into(),
|
||||||
export_span,
|
export_span,
|
||||||
));
|
));
|
||||||
return (garbage_pipeline(spans), vec![]);
|
return (garbage_pipeline(spans), vec![]);
|
||||||
|
@ -1460,7 +1462,7 @@ pub fn parse_export_in_module(
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
working_set.error(ParseError::Expected(
|
working_set.error(ParseError::Expected(
|
||||||
"def, def-env, alias, use, module, or extern keyword",
|
"def, def-env, alias, use, module, const, extern or extern-wrapped keyword",
|
||||||
spans[1],
|
spans[1],
|
||||||
));
|
));
|
||||||
|
|
||||||
|
@ -1469,9 +1471,9 @@ pub fn parse_export_in_module(
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
working_set.error(ParseError::MissingPositional(
|
working_set.error(ParseError::MissingPositional(
|
||||||
"def, def-env, extern, alias, use, or module keyword".into(),
|
"def, def-env, alias, use, module, const, extern or extern-wrapped keyword".to_string(),
|
||||||
Span::new(export_span.end, export_span.end),
|
Span::new(export_span.end, export_span.end),
|
||||||
"def, def-env, extern, alias, use, or module keyword.".to_string(),
|
"def, def-env, alias, use, module, const, extern or extern-wrapped keyword".to_string(),
|
||||||
));
|
));
|
||||||
|
|
||||||
vec![]
|
vec![]
|
||||||
|
@ -1794,7 +1796,7 @@ pub fn parse_module_block(
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
working_set.error(ParseError::ExpectedKeyword(
|
working_set.error(ParseError::ExpectedKeyword(
|
||||||
"def, const, def-env, extern, alias, use, module, export or export-env keyword".into(),
|
"def, const, def-env, extern, extern-wrapped, alias, use, module, export or export-env keyword".into(),
|
||||||
command.parts[0],
|
command.parts[0],
|
||||||
));
|
));
|
||||||
|
|
||||||
|
|
|
@ -39,6 +39,34 @@ fn known_external_complex_unknown_args() -> TestResult {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn known_external_from_module() -> TestResult {
|
||||||
|
run_test_contains(
|
||||||
|
r#"module spam {
|
||||||
|
export extern echo []
|
||||||
|
}
|
||||||
|
|
||||||
|
use spam echo
|
||||||
|
echo foo -b -as -9 --abc -- -Dxmy=AKOO - bar
|
||||||
|
"#,
|
||||||
|
"foo -b -as -9 --abc -- -Dxmy=AKOO - bar",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn known_external_wrapped_from_module() -> TestResult {
|
||||||
|
run_test_contains(
|
||||||
|
r#"module spam {
|
||||||
|
export extern-wrapped my-echo [...rest] { ^echo $rest }
|
||||||
|
}
|
||||||
|
|
||||||
|
use spam
|
||||||
|
spam my-echo foo -b -as -9 --abc -- -Dxmy=AKOO - bar
|
||||||
|
"#,
|
||||||
|
"foo -b -as -9 --abc -- -Dxmy=AKOO - bar",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn known_external_short_flag_batch_arg_allowed() -> TestResult {
|
fn known_external_short_flag_batch_arg_allowed() -> TestResult {
|
||||||
run_test_contains("extern echo [-a, -b: int]; echo -ab 10", "-b 10")
|
run_test_contains("extern echo [-a, -b: int]; echo -ab 10", "-b 10")
|
||||||
|
|
|
@ -121,7 +121,7 @@ fn command_not_found_error_suggests_typo_fix() {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn command_not_found_error_shows_not_found() {
|
fn command_not_found_error_shows_not_found_1() {
|
||||||
let actual = nu!(r#"
|
let actual = nu!(r#"
|
||||||
export extern "foo" [];
|
export extern "foo" [];
|
||||||
foo
|
foo
|
||||||
|
@ -129,6 +129,15 @@ fn command_not_found_error_shows_not_found() {
|
||||||
assert!(actual.err.contains("'foo' was not found"));
|
assert!(actual.err.contains("'foo' was not found"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn command_not_found_error_shows_not_found_2() {
|
||||||
|
let actual = nu!(r#"
|
||||||
|
export extern-wrapped my-foo [...rest] { foo };
|
||||||
|
my-foo
|
||||||
|
"#);
|
||||||
|
assert!(actual.err.contains("did you mean"));
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn command_substitution_wont_output_extra_newline() {
|
fn command_substitution_wont_output_extra_newline() {
|
||||||
let actual = nu!(r#"
|
let actual = nu!(r#"
|
||||||
|
|
Loading…
Reference in a new issue