mirror of
https://github.com/nushell/nushell
synced 2025-01-13 21:55:07 +00:00
Allow main command to define top-level module command (#7764)
This commit is contained in:
parent
8d5165c449
commit
3552d03f6c
13 changed files with 633 additions and 135 deletions
|
@ -151,15 +151,11 @@ pub fn help_modules(
|
||||||
long_desc.push_str(&format!("{G}Module{RESET}: {C}{name}{RESET}"));
|
long_desc.push_str(&format!("{G}Module{RESET}: {C}{name}{RESET}"));
|
||||||
long_desc.push_str("\n\n");
|
long_desc.push_str("\n\n");
|
||||||
|
|
||||||
if !module.decls.is_empty() {
|
if !module.decls.is_empty() || module.main.is_some() {
|
||||||
let commands: Vec<(Vec<u8>, DeclId)> = engine_state.get_decls_sorted(false).collect();
|
let commands: Vec<(Vec<u8>, DeclId)> = engine_state.get_decls_sorted(false).collect();
|
||||||
|
|
||||||
let mut module_commands: Vec<(&[u8], DeclId)> = module
|
let mut module_commands = module.decls();
|
||||||
.decls
|
module_commands.sort_by(|a, b| a.0.cmp(&b.0));
|
||||||
.iter()
|
|
||||||
.map(|(name, id)| (name.as_ref(), *id))
|
|
||||||
.collect();
|
|
||||||
module_commands.sort_by(|a, b| a.0.cmp(b.0));
|
|
||||||
|
|
||||||
let commands_str = module_commands
|
let commands_str = module_commands
|
||||||
.iter()
|
.iter()
|
||||||
|
|
|
@ -302,12 +302,14 @@ fn parse_module(
|
||||||
is_debug: bool,
|
is_debug: bool,
|
||||||
span: Span,
|
span: Span,
|
||||||
) -> Result<PipelineData, ShellError> {
|
) -> Result<PipelineData, ShellError> {
|
||||||
|
let filename = filename.unwrap_or_else(|| "empty".to_string());
|
||||||
|
|
||||||
let start = working_set.next_span_start();
|
let start = working_set.next_span_start();
|
||||||
working_set.add_file(filename.unwrap_or_else(|| "empty".to_string()), contents);
|
working_set.add_file(filename.clone(), contents);
|
||||||
let end = working_set.next_span_start();
|
let end = working_set.next_span_start();
|
||||||
|
|
||||||
let new_span = Span::new(start, end);
|
let new_span = Span::new(start, end);
|
||||||
let (_, _, _, err) = parse_module_block(working_set, new_span, &[]);
|
let (_, _, _, err) = parse_module_block(working_set, new_span, filename.as_bytes(), &[]);
|
||||||
|
|
||||||
if err.is_some() {
|
if err.is_some() {
|
||||||
if is_debug {
|
if is_debug {
|
||||||
|
|
|
@ -314,3 +314,31 @@ fn help_usage_extra_usage() {
|
||||||
assert!(!actual.out.contains("alias_line2"));
|
assert!(!actual.out.contains("alias_line2"));
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn help_modules_main_1() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam {
|
||||||
|
export def main [] { 'foo' };
|
||||||
|
}"#,
|
||||||
|
"help spam",
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.out.contains(" spam"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn help_modules_main_2() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam {
|
||||||
|
export def main [] { 'foo' };
|
||||||
|
}"#,
|
||||||
|
"help modules | where name == spam | get 0.commands.0",
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
|
@ -211,3 +211,96 @@ fn use_module_creates_accurate_did_you_mean_2() {
|
||||||
"command 'foo' was not found but it exists in module 'spam'; try importing it with `use`"
|
"command 'foo' was not found but it exists in module 'spam'; try importing it with `use`"
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn use_main_1() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { export def main [] { "spam" } }"#,
|
||||||
|
r#"use spam"#,
|
||||||
|
r#"spam"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn use_main_2() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { export def main [] { "spam" } }"#,
|
||||||
|
r#"use spam main"#,
|
||||||
|
r#"spam"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn use_main_3() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { export def main [] { "spam" } }"#,
|
||||||
|
r#"use spam [ main ]"#,
|
||||||
|
r#"spam"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn use_main_4() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { export def main [] { "spam" } }"#,
|
||||||
|
r#"use spam *"#,
|
||||||
|
r#"spam"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn use_main_def_env() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { export def-env main [] { let-env SPAM = "spam" } }"#,
|
||||||
|
r#"use spam"#,
|
||||||
|
r#"spam"#,
|
||||||
|
r#"$env.SPAM"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn use_main_def_known_external() {
|
||||||
|
// note: requires installed cargo
|
||||||
|
let inp = &[
|
||||||
|
r#"module cargo { export extern main [] }"#,
|
||||||
|
r#"use cargo"#,
|
||||||
|
r#"cargo --version"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.out.contains("cargo"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn use_main_not_exported() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { def main [] { "spam" } }"#,
|
||||||
|
r#"use spam"#,
|
||||||
|
r#"spam"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.err.contains("external_command"));
|
||||||
|
}
|
||||||
|
|
|
@ -505,9 +505,9 @@ impl<'e, 's> ScopeData<'e, 's> {
|
||||||
let module = self.engine_state.get_module(**module_id);
|
let module = self.engine_state.get_module(**module_id);
|
||||||
|
|
||||||
let export_commands: Vec<Value> = module
|
let export_commands: Vec<Value> = module
|
||||||
.decls
|
.decls()
|
||||||
.keys()
|
.iter()
|
||||||
.map(|bytes| Value::string(String::from_utf8_lossy(bytes), span))
|
.map(|(bytes, _)| Value::string(String::from_utf8_lossy(bytes), span))
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
let export_aliases: Vec<Value> = module
|
let export_aliases: Vec<Value> = module
|
||||||
|
|
|
@ -213,6 +213,26 @@ pub enum ParseError {
|
||||||
#[diagnostic(code(nu::parser::cyclical_module_import), url(docsrs), help("{0}"))]
|
#[diagnostic(code(nu::parser::cyclical_module_import), url(docsrs), help("{0}"))]
|
||||||
CyclicalModuleImport(String, #[label = "detected cyclical module import"] Span),
|
CyclicalModuleImport(String, #[label = "detected cyclical module import"] Span),
|
||||||
|
|
||||||
|
#[error("Can't export {0} named same as the module.")]
|
||||||
|
#[diagnostic(
|
||||||
|
code(nu::parser::named_as_module),
|
||||||
|
url(docsrs),
|
||||||
|
help("Module {1} can't export {0} named the same as the module. Either change the module name, or export `main` custom command.")
|
||||||
|
)]
|
||||||
|
NamedAsModule(
|
||||||
|
String,
|
||||||
|
String,
|
||||||
|
#[label = "can't export from module {1}"] Span,
|
||||||
|
),
|
||||||
|
|
||||||
|
#[error("Can't export alias defined as 'main'.")]
|
||||||
|
#[diagnostic(
|
||||||
|
code(nu::parser::export_main_alias_not_allowed),
|
||||||
|
url(docsrs),
|
||||||
|
help("Exporting aliases as 'main' is not allowed. Either rename the alias or convert it to a custom command.")
|
||||||
|
)]
|
||||||
|
ExportMainAliasNotAllowed(#[label = "can't export from module"] Span),
|
||||||
|
|
||||||
#[error("Active overlay not found.")]
|
#[error("Active overlay not found.")]
|
||||||
#[diagnostic(code(nu::parser::active_overlay_not_found), url(docsrs))]
|
#[diagnostic(code(nu::parser::active_overlay_not_found), url(docsrs))]
|
||||||
ActiveOverlayNotFound(#[label = "not an active overlay"] Span),
|
ActiveOverlayNotFound(#[label = "not an active overlay"] Span),
|
||||||
|
@ -450,6 +470,8 @@ impl ParseError {
|
||||||
ParseError::AliasNotValid(s) => *s,
|
ParseError::AliasNotValid(s) => *s,
|
||||||
ParseError::CommandDefNotValid(s) => *s,
|
ParseError::CommandDefNotValid(s) => *s,
|
||||||
ParseError::ModuleNotFound(s) => *s,
|
ParseError::ModuleNotFound(s) => *s,
|
||||||
|
ParseError::NamedAsModule(_, _, s) => *s,
|
||||||
|
ParseError::ExportMainAliasNotAllowed(s) => *s,
|
||||||
ParseError::CyclicalModuleImport(_, s) => *s,
|
ParseError::CyclicalModuleImport(_, s) => *s,
|
||||||
ParseError::ModuleOrOverlayNotFound(s) => *s,
|
ParseError::ModuleOrOverlayNotFound(s) => *s,
|
||||||
ParseError::ActiveOverlayNotFound(s) => *s,
|
ParseError::ActiveOverlayNotFound(s) => *s,
|
||||||
|
|
|
@ -232,6 +232,7 @@ pub fn parse_for(
|
||||||
pub fn parse_def(
|
pub fn parse_def(
|
||||||
working_set: &mut StateWorkingSet,
|
working_set: &mut StateWorkingSet,
|
||||||
lite_command: &LiteCommand,
|
lite_command: &LiteCommand,
|
||||||
|
module_name: Option<&[u8]>,
|
||||||
expand_aliases_denylist: &[usize],
|
expand_aliases_denylist: &[usize],
|
||||||
) -> (Pipeline, Option<ParseError>) {
|
) -> (Pipeline, Option<ParseError>) {
|
||||||
let spans = &lite_command.parts[..];
|
let spans = &lite_command.parts[..];
|
||||||
|
@ -337,9 +338,39 @@ pub fn parse_def(
|
||||||
|
|
||||||
let mut error = None;
|
let mut error = None;
|
||||||
|
|
||||||
if let (Some(name), Some(mut signature), Some(block_id)) =
|
let name = if let Some(name) = name_expr.as_string() {
|
||||||
(&name_expr.as_string(), sig.as_signature(), block.as_block())
|
if let Some(mod_name) = module_name {
|
||||||
{
|
if name.as_bytes() == mod_name {
|
||||||
|
let name_expr_span = name_expr.span;
|
||||||
|
|
||||||
|
return (
|
||||||
|
Pipeline::from_vec(vec![Expression {
|
||||||
|
expr: Expr::Call(call),
|
||||||
|
span: call_span,
|
||||||
|
ty: Type::Any,
|
||||||
|
custom_completion: None,
|
||||||
|
}]),
|
||||||
|
Some(ParseError::NamedAsModule(
|
||||||
|
"command".to_string(),
|
||||||
|
name,
|
||||||
|
name_expr_span,
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
name
|
||||||
|
} else {
|
||||||
|
return (
|
||||||
|
garbage_pipeline(spans),
|
||||||
|
Some(ParseError::UnknownState(
|
||||||
|
"Could not get string from string expression".into(),
|
||||||
|
name_expr.span,
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
if let (Some(mut signature), Some(block_id)) = (sig.as_signature(), block.as_block()) {
|
||||||
if let Some(decl_id) = working_set.find_predecl(name.as_bytes()) {
|
if let Some(decl_id) = working_set.find_predecl(name.as_bytes()) {
|
||||||
let declaration = working_set.get_decl_mut(decl_id);
|
let declaration = working_set.get_decl_mut(decl_id);
|
||||||
|
|
||||||
|
@ -398,17 +429,8 @@ pub fn parse_def(
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(name) = name_expr.as_string() {
|
|
||||||
// It's OK if it returns None: The decl was already merged in previous parse pass.
|
// It's OK if it returns None: The decl was already merged in previous parse pass.
|
||||||
working_set.merge_predecl(name.as_bytes());
|
working_set.merge_predecl(name.as_bytes());
|
||||||
} else {
|
|
||||||
error = error.or_else(|| {
|
|
||||||
Some(ParseError::UnknownState(
|
|
||||||
"Could not get string from string expression".into(),
|
|
||||||
name_expr.span,
|
|
||||||
))
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
(
|
(
|
||||||
Pipeline::from_vec(vec![Expression {
|
Pipeline::from_vec(vec![Expression {
|
||||||
|
@ -424,6 +446,7 @@ pub fn parse_def(
|
||||||
pub fn parse_extern(
|
pub fn parse_extern(
|
||||||
working_set: &mut StateWorkingSet,
|
working_set: &mut StateWorkingSet,
|
||||||
lite_command: &LiteCommand,
|
lite_command: &LiteCommand,
|
||||||
|
module_name: Option<&[u8]>,
|
||||||
expand_aliases_denylist: &[usize],
|
expand_aliases_denylist: &[usize],
|
||||||
) -> (Pipeline, Option<ParseError>) {
|
) -> (Pipeline, Option<ParseError>) {
|
||||||
let spans = &lite_command.parts;
|
let spans = &lite_command.parts;
|
||||||
|
@ -495,16 +518,45 @@ pub fn parse_extern(
|
||||||
|
|
||||||
if let (Some(name_expr), Some(sig)) = (name_expr, sig) {
|
if let (Some(name_expr), Some(sig)) = (name_expr, sig) {
|
||||||
if let (Some(name), Some(mut signature)) = (&name_expr.as_string(), sig.as_signature()) {
|
if let (Some(name), Some(mut signature)) = (&name_expr.as_string(), sig.as_signature()) {
|
||||||
|
if let Some(mod_name) = module_name {
|
||||||
|
if name.as_bytes() == mod_name {
|
||||||
|
let name_expr_span = name_expr.span;
|
||||||
|
return (
|
||||||
|
Pipeline::from_vec(vec![Expression {
|
||||||
|
expr: Expr::Call(call),
|
||||||
|
span: call_span,
|
||||||
|
ty: Type::Any,
|
||||||
|
custom_completion: None,
|
||||||
|
}]),
|
||||||
|
Some(ParseError::NamedAsModule(
|
||||||
|
"known external".to_string(),
|
||||||
|
name.clone(),
|
||||||
|
name_expr_span,
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(decl_id) = working_set.find_predecl(name.as_bytes()) {
|
if let Some(decl_id) = working_set.find_predecl(name.as_bytes()) {
|
||||||
let declaration = working_set.get_decl_mut(decl_id);
|
let declaration = working_set.get_decl_mut(decl_id);
|
||||||
|
|
||||||
signature.name = name.clone();
|
let external_name = if let Some(mod_name) = module_name {
|
||||||
|
if name.as_bytes() == b"main" {
|
||||||
|
String::from_utf8_lossy(mod_name).to_string()
|
||||||
|
} else {
|
||||||
|
name.clone()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
name.clone()
|
||||||
|
};
|
||||||
|
|
||||||
|
signature.name = external_name.clone();
|
||||||
signature.usage = usage.clone();
|
signature.usage = usage.clone();
|
||||||
signature.extra_usage = extra_usage.clone();
|
signature.extra_usage = extra_usage.clone();
|
||||||
signature.allows_unknown_args = true;
|
signature.allows_unknown_args = true;
|
||||||
|
|
||||||
let decl = KnownExternal {
|
let decl = KnownExternal {
|
||||||
name: name.to_string(),
|
name: external_name,
|
||||||
usage: [usage, extra_usage].join("\n"),
|
usage: [usage, extra_usage].join("\n"),
|
||||||
signature,
|
signature,
|
||||||
};
|
};
|
||||||
|
@ -546,6 +598,7 @@ pub fn parse_extern(
|
||||||
pub fn parse_alias(
|
pub fn parse_alias(
|
||||||
working_set: &mut StateWorkingSet,
|
working_set: &mut StateWorkingSet,
|
||||||
lite_command: &LiteCommand,
|
lite_command: &LiteCommand,
|
||||||
|
module_name: Option<&[u8]>,
|
||||||
expand_aliases_denylist: &[usize],
|
expand_aliases_denylist: &[usize],
|
||||||
) -> (Pipeline, Option<ParseError>) {
|
) -> (Pipeline, Option<ParseError>) {
|
||||||
let spans = &lite_command.parts;
|
let spans = &lite_command.parts;
|
||||||
|
@ -644,6 +697,37 @@ pub fn parse_alias(
|
||||||
} else {
|
} else {
|
||||||
alias_name.to_vec()
|
alias_name.to_vec()
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if let Some(mod_name) = module_name {
|
||||||
|
if alias_name == mod_name {
|
||||||
|
return (
|
||||||
|
Pipeline::from_vec(vec![Expression {
|
||||||
|
expr: Expr::Call(call),
|
||||||
|
span: span(spans),
|
||||||
|
ty: output,
|
||||||
|
custom_completion: None,
|
||||||
|
}]),
|
||||||
|
Some(ParseError::NamedAsModule(
|
||||||
|
"alias".to_string(),
|
||||||
|
String::from_utf8_lossy(&alias_name).to_string(),
|
||||||
|
spans[split_id],
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if &alias_name == b"main" {
|
||||||
|
return (
|
||||||
|
Pipeline::from_vec(vec![Expression {
|
||||||
|
expr: Expr::Call(call),
|
||||||
|
span: span(spans),
|
||||||
|
ty: output,
|
||||||
|
custom_completion: None,
|
||||||
|
}]),
|
||||||
|
Some(ParseError::ExportMainAliasNotAllowed(spans[split_id])),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let _equals = working_set.get_span_contents(spans[split_id + 1]);
|
let _equals = working_set.get_span_contents(spans[split_id + 1]);
|
||||||
|
|
||||||
let replacement = spans[(split_id + 2)..].to_vec();
|
let replacement = spans[(split_id + 2)..].to_vec();
|
||||||
|
@ -773,16 +857,16 @@ pub fn parse_export_in_block(
|
||||||
}
|
}
|
||||||
|
|
||||||
match full_name.as_slice() {
|
match full_name.as_slice() {
|
||||||
b"export alias" => parse_alias(working_set, lite_command, expand_aliases_denylist),
|
b"export alias" => parse_alias(working_set, lite_command, None, expand_aliases_denylist),
|
||||||
b"export def" | b"export def-env" => {
|
b"export def" | b"export def-env" => {
|
||||||
parse_def(working_set, lite_command, expand_aliases_denylist)
|
parse_def(working_set, lite_command, None, expand_aliases_denylist)
|
||||||
}
|
}
|
||||||
b"export use" => {
|
b"export use" => {
|
||||||
let (pipeline, _, err) =
|
let (pipeline, _, err) =
|
||||||
parse_use(working_set, &lite_command.parts, expand_aliases_denylist);
|
parse_use(working_set, &lite_command.parts, expand_aliases_denylist);
|
||||||
(pipeline, err)
|
(pipeline, err)
|
||||||
}
|
}
|
||||||
b"export extern" => parse_extern(working_set, lite_command, expand_aliases_denylist),
|
b"export extern" => parse_extern(working_set, lite_command, None, expand_aliases_denylist),
|
||||||
_ => (
|
_ => (
|
||||||
garbage_pipeline(&lite_command.parts),
|
garbage_pipeline(&lite_command.parts),
|
||||||
Some(ParseError::UnexpectedKeyword(
|
Some(ParseError::UnexpectedKeyword(
|
||||||
|
@ -797,6 +881,7 @@ pub fn parse_export_in_block(
|
||||||
pub fn parse_export_in_module(
|
pub fn parse_export_in_module(
|
||||||
working_set: &mut StateWorkingSet,
|
working_set: &mut StateWorkingSet,
|
||||||
lite_command: &LiteCommand,
|
lite_command: &LiteCommand,
|
||||||
|
module_name: &[u8],
|
||||||
expand_aliases_denylist: &[usize],
|
expand_aliases_denylist: &[usize],
|
||||||
) -> (Pipeline, Vec<Exportable>, Option<ParseError>) {
|
) -> (Pipeline, Vec<Exportable>, Option<ParseError>) {
|
||||||
let spans = &lite_command.parts[..];
|
let spans = &lite_command.parts[..];
|
||||||
|
@ -856,8 +941,12 @@ 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 (pipeline, err) =
|
let (pipeline, err) = parse_def(
|
||||||
parse_def(working_set, &lite_command, expand_aliases_denylist);
|
working_set,
|
||||||
|
&lite_command,
|
||||||
|
Some(module_name),
|
||||||
|
expand_aliases_denylist,
|
||||||
|
);
|
||||||
error = error.or(err);
|
error = error.or(err);
|
||||||
|
|
||||||
let export_def_decl_id =
|
let export_def_decl_id =
|
||||||
|
@ -924,8 +1013,12 @@ 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 (pipeline, err) =
|
let (pipeline, err) = parse_def(
|
||||||
parse_def(working_set, &lite_command, expand_aliases_denylist);
|
working_set,
|
||||||
|
&lite_command,
|
||||||
|
Some(module_name),
|
||||||
|
expand_aliases_denylist,
|
||||||
|
);
|
||||||
error = error.or(err);
|
error = error.or(err);
|
||||||
|
|
||||||
let export_def_decl_id =
|
let export_def_decl_id =
|
||||||
|
@ -993,8 +1086,12 @@ 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 (pipeline, err) =
|
let (pipeline, err) = parse_extern(
|
||||||
parse_extern(working_set, &lite_command, expand_aliases_denylist);
|
working_set,
|
||||||
|
&lite_command,
|
||||||
|
Some(module_name),
|
||||||
|
expand_aliases_denylist,
|
||||||
|
);
|
||||||
error = error.or(err);
|
error = error.or(err);
|
||||||
|
|
||||||
let export_def_decl_id =
|
let export_def_decl_id =
|
||||||
|
@ -1062,8 +1159,12 @@ 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 (pipeline, err) =
|
let (pipeline, err) = parse_alias(
|
||||||
parse_alias(working_set, &lite_command, expand_aliases_denylist);
|
working_set,
|
||||||
|
&lite_command,
|
||||||
|
Some(module_name),
|
||||||
|
expand_aliases_denylist,
|
||||||
|
);
|
||||||
error = error.or(err);
|
error = error.or(err);
|
||||||
|
|
||||||
let export_alias_decl_id =
|
let export_alias_decl_id =
|
||||||
|
@ -1349,6 +1450,7 @@ fn collect_first_comments(tokens: &[Token]) -> Vec<Span> {
|
||||||
pub fn parse_module_block(
|
pub fn parse_module_block(
|
||||||
working_set: &mut StateWorkingSet,
|
working_set: &mut StateWorkingSet,
|
||||||
span: Span,
|
span: Span,
|
||||||
|
module_name: &[u8],
|
||||||
expand_aliases_denylist: &[usize],
|
expand_aliases_denylist: &[usize],
|
||||||
) -> (Block, Module, Vec<Span>, Option<ParseError>) {
|
) -> (Block, Module, Vec<Span>, Option<ParseError>) {
|
||||||
let mut error = None;
|
let mut error = None;
|
||||||
|
@ -1373,7 +1475,7 @@ pub fn parse_module_block(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut module = Module::from_span(span);
|
let mut module = Module::from_span(module_name.to_vec(), span);
|
||||||
|
|
||||||
let block: Block = output
|
let block: Block = output
|
||||||
.block
|
.block
|
||||||
|
@ -1386,20 +1488,32 @@ pub fn parse_module_block(
|
||||||
|
|
||||||
let (pipeline, err) = match name {
|
let (pipeline, err) = match name {
|
||||||
b"def" | b"def-env" => {
|
b"def" | b"def-env" => {
|
||||||
let (pipeline, err) =
|
let (pipeline, err) = parse_def(
|
||||||
parse_def(working_set, command, expand_aliases_denylist);
|
working_set,
|
||||||
|
command,
|
||||||
|
None, // using commands named as the module locally is OK
|
||||||
|
expand_aliases_denylist,
|
||||||
|
);
|
||||||
|
|
||||||
(pipeline, err)
|
(pipeline, err)
|
||||||
}
|
}
|
||||||
b"extern" => {
|
b"extern" => {
|
||||||
let (pipeline, err) =
|
let (pipeline, err) = parse_extern(
|
||||||
parse_extern(working_set, command, expand_aliases_denylist);
|
working_set,
|
||||||
|
command,
|
||||||
|
None,
|
||||||
|
expand_aliases_denylist,
|
||||||
|
);
|
||||||
|
|
||||||
(pipeline, err)
|
(pipeline, err)
|
||||||
}
|
}
|
||||||
b"alias" => {
|
b"alias" => {
|
||||||
let (pipeline, err) =
|
let (pipeline, err) = parse_alias(
|
||||||
parse_alias(working_set, command, expand_aliases_denylist);
|
working_set,
|
||||||
|
command,
|
||||||
|
None, // using aliases named as the module locally is OK
|
||||||
|
expand_aliases_denylist,
|
||||||
|
);
|
||||||
|
|
||||||
(pipeline, err)
|
(pipeline, err)
|
||||||
}
|
}
|
||||||
|
@ -1413,6 +1527,7 @@ pub fn parse_module_block(
|
||||||
let (pipe, exportables, err) = parse_export_in_module(
|
let (pipe, exportables, err) = parse_export_in_module(
|
||||||
working_set,
|
working_set,
|
||||||
command,
|
command,
|
||||||
|
module_name,
|
||||||
expand_aliases_denylist,
|
expand_aliases_denylist,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -1420,8 +1535,12 @@ pub fn parse_module_block(
|
||||||
for exportable in exportables {
|
for exportable in exportables {
|
||||||
match exportable {
|
match exportable {
|
||||||
Exportable::Decl { name, id } => {
|
Exportable::Decl { name, id } => {
|
||||||
|
if &name == b"main" {
|
||||||
|
module.main = Some(id);
|
||||||
|
} else {
|
||||||
module.add_decl(name, id);
|
module.add_decl(name, id);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
Exportable::Alias { name, id } => {
|
Exportable::Alias { name, id } => {
|
||||||
module.add_alias(name, id);
|
module.add_alias(name, id);
|
||||||
}
|
}
|
||||||
|
@ -1519,8 +1638,12 @@ pub fn parse_module(
|
||||||
|
|
||||||
let block_span = Span::new(start, end);
|
let block_span = Span::new(start, end);
|
||||||
|
|
||||||
let (block, module, inner_comments, err) =
|
let (block, module, inner_comments, err) = parse_module_block(
|
||||||
parse_module_block(working_set, block_span, expand_aliases_denylist);
|
working_set,
|
||||||
|
block_span,
|
||||||
|
module_name.as_bytes(),
|
||||||
|
expand_aliases_denylist,
|
||||||
|
);
|
||||||
error = error.or(err);
|
error = error.or(err);
|
||||||
|
|
||||||
let block_id = working_set.add_block(block);
|
let block_id = working_set.add_block(block);
|
||||||
|
@ -1762,6 +1885,7 @@ pub fn parse_use(
|
||||||
let (block, module, module_comments, err) = parse_module_block(
|
let (block, module, module_comments, err) = parse_module_block(
|
||||||
working_set,
|
working_set,
|
||||||
Span::new(span_start, span_end),
|
Span::new(span_start, span_end),
|
||||||
|
module_name.as_bytes(),
|
||||||
expand_aliases_denylist,
|
expand_aliases_denylist,
|
||||||
);
|
);
|
||||||
error = error.or(err);
|
error = error.or(err);
|
||||||
|
@ -1801,14 +1925,16 @@ pub fn parse_use(
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
error = error.or(Some(ParseError::ModuleNotFound(import_pattern.head.span)));
|
return (
|
||||||
|
Pipeline::from_vec(vec![Expression {
|
||||||
let old_span = import_pattern.head.span;
|
expr: Expr::Call(call),
|
||||||
|
span: span(spans),
|
||||||
let mut import_pattern = ImportPattern::new();
|
ty: Type::Any,
|
||||||
import_pattern.head.span = old_span;
|
custom_completion: None,
|
||||||
|
}]),
|
||||||
(import_pattern, Module::new())
|
vec![],
|
||||||
|
Some(ParseError::ModuleNotFound(import_pattern.head.span)),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return (
|
return (
|
||||||
|
@ -1831,12 +1957,18 @@ pub fn parse_use(
|
||||||
let mut decl_output = vec![];
|
let mut decl_output = vec![];
|
||||||
let mut alias_output = vec![];
|
let mut alias_output = vec![];
|
||||||
|
|
||||||
if let Some(id) = module.get_decl_id(name) {
|
if name == b"main" {
|
||||||
|
if let Some(id) = &module.main {
|
||||||
|
decl_output.push((import_pattern.head.name.clone(), *id));
|
||||||
|
} else {
|
||||||
|
error = error.or(Some(ParseError::ExportNotFound(*span)));
|
||||||
|
}
|
||||||
|
} else if let Some(id) = module.get_decl_id(name) {
|
||||||
decl_output.push((name.clone(), id));
|
decl_output.push((name.clone(), id));
|
||||||
} else if let Some(id) = module.get_alias_id(name) {
|
} else if let Some(id) = module.get_alias_id(name) {
|
||||||
alias_output.push((name.clone(), id));
|
alias_output.push((name.clone(), id));
|
||||||
} else {
|
} else {
|
||||||
error = error.or(Some(ParseError::ExportNotFound(*span)))
|
error = error.or(Some(ParseError::ExportNotFound(*span)));
|
||||||
}
|
}
|
||||||
|
|
||||||
(decl_output, alias_output)
|
(decl_output, alias_output)
|
||||||
|
@ -1846,7 +1978,13 @@ pub fn parse_use(
|
||||||
let mut alias_output = vec![];
|
let mut alias_output = vec![];
|
||||||
|
|
||||||
for (name, span) in names {
|
for (name, span) in names {
|
||||||
if let Some(id) = module.get_decl_id(name) {
|
if name == b"main" {
|
||||||
|
if let Some(id) = &module.main {
|
||||||
|
decl_output.push((import_pattern.head.name.clone(), *id));
|
||||||
|
} else {
|
||||||
|
error = error.or(Some(ParseError::ExportNotFound(*span)));
|
||||||
|
}
|
||||||
|
} else if let Some(id) = module.get_decl_id(name) {
|
||||||
decl_output.push((name.clone(), id));
|
decl_output.push((name.clone(), id));
|
||||||
} else if let Some(id) = module.get_alias_id(name) {
|
} else if let Some(id) = module.get_alias_id(name) {
|
||||||
alias_output.push((name.clone(), id));
|
alias_output.push((name.clone(), id));
|
||||||
|
@ -1992,6 +2130,7 @@ pub fn parse_hide(
|
||||||
error = error.or(err);
|
error = error.or(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// module used only internally, not saved anywhere
|
||||||
let (is_module, module) = if let Some(module_id) =
|
let (is_module, module) = if let Some(module_id) =
|
||||||
working_set.find_module(&import_pattern.head.name)
|
working_set.find_module(&import_pattern.head.name)
|
||||||
{
|
{
|
||||||
|
@ -2000,19 +2139,19 @@ pub fn parse_hide(
|
||||||
// The pattern head can be:
|
// The pattern head can be:
|
||||||
if let Some(id) = working_set.find_alias(&import_pattern.head.name) {
|
if let Some(id) = working_set.find_alias(&import_pattern.head.name) {
|
||||||
// an alias,
|
// an alias,
|
||||||
let mut module = Module::new();
|
let mut module = Module::new(b"tmp".to_vec());
|
||||||
module.add_alias(import_pattern.head.name.clone(), id);
|
module.add_alias(import_pattern.head.name.clone(), id);
|
||||||
|
|
||||||
(false, module)
|
(false, module)
|
||||||
} else if let Some(id) = working_set.find_decl(&import_pattern.head.name, &Type::Any) {
|
} else if let Some(id) = working_set.find_decl(&import_pattern.head.name, &Type::Any) {
|
||||||
// a custom command,
|
// a custom command,
|
||||||
let mut module = Module::new();
|
let mut module = Module::new(b"tmp".to_vec());
|
||||||
module.add_decl(import_pattern.head.name.clone(), id);
|
module.add_decl(import_pattern.head.name.clone(), id);
|
||||||
|
|
||||||
(false, module)
|
(false, module)
|
||||||
} else {
|
} else {
|
||||||
// , or it could be an env var (handled by the engine)
|
// , or it could be an env var (handled by the engine)
|
||||||
(false, Module::new())
|
(false, Module::new(b"tmp".to_vec()))
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return (
|
return (
|
||||||
|
@ -2038,7 +2177,14 @@ pub fn parse_hide(
|
||||||
let mut aliases = vec![];
|
let mut aliases = vec![];
|
||||||
let mut decls = vec![];
|
let mut decls = vec![];
|
||||||
|
|
||||||
if let Some(item) = module.alias_name_with_head(name, &import_pattern.head.name)
|
if name == b"main" {
|
||||||
|
if module.main.is_some() {
|
||||||
|
decls.push(import_pattern.head.name.clone());
|
||||||
|
} else {
|
||||||
|
error = error.or(Some(ParseError::ExportNotFound(*span)));
|
||||||
|
}
|
||||||
|
} else if let Some(item) =
|
||||||
|
module.alias_name_with_head(name, &import_pattern.head.name)
|
||||||
{
|
{
|
||||||
aliases.push(item);
|
aliases.push(item);
|
||||||
} else if let Some(item) =
|
} else if let Some(item) =
|
||||||
|
@ -2056,7 +2202,14 @@ pub fn parse_hide(
|
||||||
let mut decls = vec![];
|
let mut decls = vec![];
|
||||||
|
|
||||||
for (name, span) in names {
|
for (name, span) in names {
|
||||||
if let Some(item) =
|
if name == b"main" {
|
||||||
|
if module.main.is_some() {
|
||||||
|
decls.push(import_pattern.head.name.clone());
|
||||||
|
} else {
|
||||||
|
error = error.or(Some(ParseError::ExportNotFound(*span)));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else if let Some(item) =
|
||||||
module.alias_name_with_head(name, &import_pattern.head.name)
|
module.alias_name_with_head(name, &import_pattern.head.name)
|
||||||
{
|
{
|
||||||
aliases.push(item);
|
aliases.push(item);
|
||||||
|
@ -2346,7 +2499,11 @@ pub fn parse_overlay_new(
|
||||||
custom_completion: None,
|
custom_completion: None,
|
||||||
}]);
|
}]);
|
||||||
|
|
||||||
let module_id = working_set.add_module(&overlay_name, Module::new(), vec![]);
|
let module_id = working_set.add_module(
|
||||||
|
&overlay_name,
|
||||||
|
Module::new(overlay_name.as_bytes().to_vec()),
|
||||||
|
vec![],
|
||||||
|
);
|
||||||
|
|
||||||
working_set.add_overlay(
|
working_set.add_overlay(
|
||||||
overlay_name.as_bytes().to_vec(),
|
overlay_name.as_bytes().to_vec(),
|
||||||
|
@ -2525,7 +2682,12 @@ pub fn parse_overlay_use(
|
||||||
|
|
||||||
if let Some(new_module_id) = working_set.find_module(overlay_name.as_bytes()) {
|
if let Some(new_module_id) = working_set.find_module(overlay_name.as_bytes()) {
|
||||||
if !do_reload && (module_id == new_module_id) {
|
if !do_reload && (module_id == new_module_id) {
|
||||||
(overlay_name, Module::new(), module_id, false)
|
(
|
||||||
|
overlay_name,
|
||||||
|
Module::new(working_set.get_module(module_id).name.clone()),
|
||||||
|
module_id,
|
||||||
|
false,
|
||||||
|
)
|
||||||
} else {
|
} else {
|
||||||
// The origin module of an overlay changed => update it
|
// The origin module of an overlay changed => update it
|
||||||
(
|
(
|
||||||
|
@ -2536,7 +2698,8 @@ pub fn parse_overlay_use(
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
(overlay_name, Module::new(), module_id, true)
|
let module_name = overlay_name.as_bytes().to_vec();
|
||||||
|
(overlay_name, Module::new(module_name), module_id, true)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Create a new overlay from a module
|
// Create a new overlay from a module
|
||||||
|
@ -2586,6 +2749,7 @@ pub fn parse_overlay_use(
|
||||||
let (block, module, module_comments, err) = parse_module_block(
|
let (block, module, module_comments, err) = parse_module_block(
|
||||||
working_set,
|
working_set,
|
||||||
Span::new(span_start, span_end),
|
Span::new(span_start, span_end),
|
||||||
|
overlay_name.as_bytes(),
|
||||||
expand_aliases_denylist,
|
expand_aliases_denylist,
|
||||||
);
|
);
|
||||||
error = error.or(err);
|
error = error.or(err);
|
||||||
|
|
|
@ -5291,8 +5291,8 @@ pub fn parse_builtin_commands(
|
||||||
let name = working_set.get_span_contents(lite_command.parts[0]);
|
let name = working_set.get_span_contents(lite_command.parts[0]);
|
||||||
|
|
||||||
match name {
|
match name {
|
||||||
b"def" | b"def-env" => parse_def(working_set, lite_command, expand_aliases_denylist),
|
b"def" | b"def-env" => parse_def(working_set, lite_command, None, expand_aliases_denylist),
|
||||||
b"extern" => parse_extern(working_set, lite_command, expand_aliases_denylist),
|
b"extern" => parse_extern(working_set, lite_command, None, expand_aliases_denylist),
|
||||||
b"let" | b"const" => {
|
b"let" | b"const" => {
|
||||||
parse_let_or_const(working_set, &lite_command.parts, expand_aliases_denylist)
|
parse_let_or_const(working_set, &lite_command.parts, expand_aliases_denylist)
|
||||||
}
|
}
|
||||||
|
@ -5301,7 +5301,7 @@ pub fn parse_builtin_commands(
|
||||||
let (expr, err) = parse_for(working_set, &lite_command.parts, expand_aliases_denylist);
|
let (expr, err) = parse_for(working_set, &lite_command.parts, expand_aliases_denylist);
|
||||||
(Pipeline::from_vec(vec![expr]), err)
|
(Pipeline::from_vec(vec![expr]), err)
|
||||||
}
|
}
|
||||||
b"alias" => parse_alias(working_set, lite_command, expand_aliases_denylist),
|
b"alias" => parse_alias(working_set, lite_command, None, expand_aliases_denylist),
|
||||||
b"module" => parse_module(working_set, lite_command, expand_aliases_denylist),
|
b"module" => parse_module(working_set, lite_command, expand_aliases_denylist),
|
||||||
b"use" => {
|
b"use" => {
|
||||||
let (pipeline, _, err) =
|
let (pipeline, _, err) =
|
||||||
|
|
|
@ -170,7 +170,7 @@ impl EngineState {
|
||||||
decls: vec![],
|
decls: vec![],
|
||||||
aliases: vec![],
|
aliases: vec![],
|
||||||
blocks: vec![],
|
blocks: vec![],
|
||||||
modules: vec![Module::new()],
|
modules: vec![Module::new(DEFAULT_OVERLAY_NAME.as_bytes().to_vec())],
|
||||||
usage: Usage::new(),
|
usage: Usage::new(),
|
||||||
// make sure we have some default overlay:
|
// make sure we have some default overlay:
|
||||||
scope: ScopeFrame::with_empty_overlay(
|
scope: ScopeFrame::with_empty_overlay(
|
||||||
|
@ -651,24 +651,6 @@ impl EngineState {
|
||||||
}
|
}
|
||||||
|
|
||||||
None
|
None
|
||||||
|
|
||||||
// for (module_id, m) in self.modules.iter().enumerate() {
|
|
||||||
// if m.has_decl(name) {
|
|
||||||
// for overlay_frame in self.active_overlays(&[]).iter() {
|
|
||||||
// let module_name = overlay_frame.modules.iter().find_map(|(key, &val)| {
|
|
||||||
// if val == module_id {
|
|
||||||
// Some(key)
|
|
||||||
// } else {
|
|
||||||
// None
|
|
||||||
// }
|
|
||||||
// });
|
|
||||||
// if let Some(final_name) = module_name {
|
|
||||||
// return Some(&final_name[..]);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// None
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn find_overlay(&self, name: &[u8]) -> Option<OverlayId> {
|
pub fn find_overlay(&self, name: &[u8]) -> Option<OverlayId> {
|
||||||
|
|
|
@ -2,33 +2,36 @@ use crate::{AliasId, BlockId, DeclId, Span};
|
||||||
|
|
||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
|
|
||||||
// TODO: Move the import pattern matching logic here from use/hide commands and
|
|
||||||
// parse_use/parse_hide
|
|
||||||
|
|
||||||
/// Collection of definitions that can be exported from a module
|
/// Collection of definitions that can be exported from a module
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct Module {
|
pub struct Module {
|
||||||
|
pub name: Vec<u8>,
|
||||||
pub decls: IndexMap<Vec<u8>, DeclId>,
|
pub decls: IndexMap<Vec<u8>, DeclId>,
|
||||||
pub aliases: IndexMap<Vec<u8>, AliasId>,
|
pub aliases: IndexMap<Vec<u8>, AliasId>,
|
||||||
pub env_block: Option<BlockId>,
|
pub env_block: Option<BlockId>, // `export-env { ... }` block
|
||||||
|
pub main: Option<DeclId>, // `export def main`
|
||||||
pub span: Option<Span>,
|
pub span: Option<Span>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Module {
|
impl Module {
|
||||||
pub fn new() -> Self {
|
pub fn new(name: Vec<u8>) -> Self {
|
||||||
Module {
|
Module {
|
||||||
|
name,
|
||||||
decls: IndexMap::new(),
|
decls: IndexMap::new(),
|
||||||
aliases: IndexMap::new(),
|
aliases: IndexMap::new(),
|
||||||
env_block: None,
|
env_block: None,
|
||||||
|
main: None,
|
||||||
span: None,
|
span: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn from_span(span: Span) -> Self {
|
pub fn from_span(name: Vec<u8>, span: Span) -> Self {
|
||||||
Module {
|
Module {
|
||||||
|
name,
|
||||||
decls: IndexMap::new(),
|
decls: IndexMap::new(),
|
||||||
aliases: IndexMap::new(),
|
aliases: IndexMap::new(),
|
||||||
env_block: None,
|
env_block: None,
|
||||||
|
main: None,
|
||||||
span: Some(span),
|
span: Some(span),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -63,6 +66,10 @@ impl Module {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn has_decl(&self, name: &[u8]) -> bool {
|
pub fn has_decl(&self, name: &[u8]) -> bool {
|
||||||
|
if name == self.name && self.main.is_some() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
self.decls.contains_key(name)
|
self.decls.contains_key(name)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -93,7 +100,8 @@ impl Module {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn decls_with_head(&self, head: &[u8]) -> Vec<(Vec<u8>, DeclId)> {
|
pub fn decls_with_head(&self, head: &[u8]) -> Vec<(Vec<u8>, DeclId)> {
|
||||||
self.decls
|
let mut result: Vec<(Vec<u8>, DeclId)> = self
|
||||||
|
.decls
|
||||||
.iter()
|
.iter()
|
||||||
.map(|(name, id)| {
|
.map(|(name, id)| {
|
||||||
let mut new_name = head.to_vec();
|
let mut new_name = head.to_vec();
|
||||||
|
@ -101,11 +109,18 @@ impl Module {
|
||||||
new_name.extend(name);
|
new_name.extend(name);
|
||||||
(new_name, *id)
|
(new_name, *id)
|
||||||
})
|
})
|
||||||
.collect()
|
.collect();
|
||||||
|
|
||||||
|
if let Some(decl_id) = self.main {
|
||||||
|
result.push((self.name.clone(), decl_id));
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn decl_names_with_head(&self, head: &[u8]) -> Vec<Vec<u8>> {
|
pub fn decl_names_with_head(&self, head: &[u8]) -> Vec<Vec<u8>> {
|
||||||
self.decls
|
let mut result: Vec<Vec<u8>> = self
|
||||||
|
.decls
|
||||||
.keys()
|
.keys()
|
||||||
.map(|name| {
|
.map(|name| {
|
||||||
let mut new_name = head.to_vec();
|
let mut new_name = head.to_vec();
|
||||||
|
@ -113,7 +128,13 @@ impl Module {
|
||||||
new_name.extend(name);
|
new_name.extend(name);
|
||||||
new_name
|
new_name
|
||||||
})
|
})
|
||||||
.collect()
|
.collect();
|
||||||
|
|
||||||
|
if self.main.is_some() {
|
||||||
|
result.push(self.name.clone());
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn aliases_with_head(&self, head: &[u8]) -> Vec<(Vec<u8>, AliasId)> {
|
pub fn aliases_with_head(&self, head: &[u8]) -> Vec<(Vec<u8>, AliasId)> {
|
||||||
|
@ -141,14 +162,27 @@ impl Module {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn decls(&self) -> Vec<(Vec<u8>, DeclId)> {
|
pub fn decls(&self) -> Vec<(Vec<u8>, DeclId)> {
|
||||||
self.decls
|
let mut result: Vec<(Vec<u8>, DeclId)> = self
|
||||||
|
.decls
|
||||||
.iter()
|
.iter()
|
||||||
.map(|(name, id)| (name.clone(), *id))
|
.map(|(name, id)| (name.clone(), *id))
|
||||||
.collect()
|
.collect();
|
||||||
|
|
||||||
|
if let Some(decl_id) = self.main {
|
||||||
|
result.push((self.name.clone(), decl_id));
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn decl_names(&self) -> Vec<Vec<u8>> {
|
pub fn decl_names(&self) -> Vec<Vec<u8>> {
|
||||||
self.decls.keys().cloned().collect()
|
let mut result: Vec<Vec<u8>> = self.decls.keys().cloned().collect();
|
||||||
|
|
||||||
|
if self.main.is_some() {
|
||||||
|
result.push(self.name.clone());
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn alias_names(&self) -> Vec<Vec<u8>> {
|
pub fn alias_names(&self) -> Vec<Vec<u8>> {
|
||||||
|
@ -162,9 +196,3 @@ impl Module {
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Module {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self::new()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -3,17 +3,14 @@ use crate::tests::{fail_test, run_test, TestResult};
|
||||||
// TODO: Test the use/hide tests also as separate lines in REPL (i.e., with merging the delta in between)
|
// TODO: Test the use/hide tests also as separate lines in REPL (i.e., with merging the delta in between)
|
||||||
#[test]
|
#[test]
|
||||||
fn hides_def() -> TestResult {
|
fn hides_def() -> TestResult {
|
||||||
fail_test(
|
fail_test(r#"def foo [] { "foo" }; hide foo; foo"#, "external_command")
|
||||||
r#"def foo [] { "foo" }; hide foo; foo"#,
|
|
||||||
"", // we just care if it errors
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn hides_alias() -> TestResult {
|
fn hides_alias() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"alias foo = echo "foo"; hide foo; foo"#,
|
r#"alias foo = echo "foo"; hide foo; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,7 +49,7 @@ fn hides_env_then_redefines() -> TestResult {
|
||||||
fn hides_def_in_scope_1() -> TestResult {
|
fn hides_def_in_scope_1() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"def foo [] { "foo" }; do { hide foo; foo }"#,
|
r#"def foo [] { "foo" }; do { hide foo; foo }"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -68,7 +65,7 @@ fn hides_def_in_scope_2() -> TestResult {
|
||||||
fn hides_def_in_scope_3() -> TestResult {
|
fn hides_def_in_scope_3() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"def foo [] { "foo" }; do { hide foo; def foo [] { "bar" }; hide foo; foo }"#,
|
r#"def foo [] { "foo" }; do { hide foo; def foo [] { "bar" }; hide foo; foo }"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -76,7 +73,7 @@ fn hides_def_in_scope_3() -> TestResult {
|
||||||
fn hides_def_in_scope_4() -> TestResult {
|
fn hides_def_in_scope_4() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"def foo [] { "foo" }; do { def foo [] { "bar" }; hide foo; hide foo; foo }"#,
|
r#"def foo [] { "foo" }; do { def foo [] { "bar" }; hide foo; hide foo; foo }"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -84,7 +81,7 @@ fn hides_def_in_scope_4() -> TestResult {
|
||||||
fn hides_alias_in_scope_1() -> TestResult {
|
fn hides_alias_in_scope_1() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"alias foo = echo "foo"; do { hide foo; foo }"#,
|
r#"alias foo = echo "foo"; do { hide foo; foo }"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,7 +97,7 @@ fn hides_alias_in_scope_2() -> TestResult {
|
||||||
fn hides_alias_in_scope_3() -> TestResult {
|
fn hides_alias_in_scope_3() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"alias foo = echo "foo"; do { hide foo; alias foo = echo "bar"; hide foo; foo }"#,
|
r#"alias foo = echo "foo"; do { hide foo; alias foo = echo "bar"; hide foo; foo }"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -108,7 +105,7 @@ fn hides_alias_in_scope_3() -> TestResult {
|
||||||
fn hides_alias_in_scope_4() -> TestResult {
|
fn hides_alias_in_scope_4() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"alias foo = echo "foo"; do { alias foo = echo "bar"; hide foo; hide foo; foo }"#,
|
r#"alias foo = echo "foo"; do { alias foo = echo "bar"; hide foo; hide foo; foo }"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -213,7 +210,7 @@ fn hides_alias_runs_def_2() -> TestResult {
|
||||||
fn hides_alias_and_def() -> TestResult {
|
fn hides_alias_and_def() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"alias foo = echo "foo"; def foo [] { "bar" }; hide foo; hide foo; foo"#,
|
r#"alias foo = echo "foo"; def foo [] { "bar" }; hide foo; hide foo; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -221,7 +218,7 @@ fn hides_alias_and_def() -> TestResult {
|
||||||
fn hides_def_import_1() -> TestResult {
|
fn hides_def_import_1() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export def foo [] { "foo" } }; use spam; hide spam foo; spam foo"#,
|
r#"module spam { export def foo [] { "foo" } }; use spam; hide spam foo; spam foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -229,7 +226,7 @@ fn hides_def_import_1() -> TestResult {
|
||||||
fn hides_def_import_2() -> TestResult {
|
fn hides_def_import_2() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export def foo [] { "foo" } }; use spam; hide spam; spam foo"#,
|
r#"module spam { export def foo [] { "foo" } }; use spam; hide spam; spam foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -237,7 +234,7 @@ fn hides_def_import_2() -> TestResult {
|
||||||
fn hides_def_import_3() -> TestResult {
|
fn hides_def_import_3() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export def foo [] { "foo" } }; use spam; hide spam [foo]; spam foo"#,
|
r#"module spam { export def foo [] { "foo" } }; use spam; hide spam [foo]; spam foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -245,7 +242,7 @@ fn hides_def_import_3() -> TestResult {
|
||||||
fn hides_def_import_4() -> TestResult {
|
fn hides_def_import_4() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export def foo [] { "foo" } }; use spam foo; hide foo; foo"#,
|
r#"module spam { export def foo [] { "foo" } }; use spam foo; hide foo; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -253,7 +250,7 @@ fn hides_def_import_4() -> TestResult {
|
||||||
fn hides_def_import_5() -> TestResult {
|
fn hides_def_import_5() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export def foo [] { "foo" } }; use spam *; hide foo; foo"#,
|
r#"module spam { export def foo [] { "foo" } }; use spam *; hide foo; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -261,7 +258,7 @@ fn hides_def_import_5() -> TestResult {
|
||||||
fn hides_def_import_6() -> TestResult {
|
fn hides_def_import_6() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export def foo [] { "foo" } }; use spam *; hide spam *; foo"#,
|
r#"module spam { export def foo [] { "foo" } }; use spam *; hide spam *; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -277,7 +274,7 @@ fn hides_def_import_then_reimports() -> TestResult {
|
||||||
fn hides_alias_import_1() -> TestResult {
|
fn hides_alias_import_1() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export alias foo = "foo" }; use spam; hide spam foo; spam foo"#,
|
r#"module spam { export alias foo = "foo" }; use spam; hide spam foo; spam foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -285,7 +282,7 @@ fn hides_alias_import_1() -> TestResult {
|
||||||
fn hides_alias_import_2() -> TestResult {
|
fn hides_alias_import_2() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export alias foo = "foo" }; use spam; hide spam; spam foo"#,
|
r#"module spam { export alias foo = "foo" }; use spam; hide spam; spam foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -293,7 +290,7 @@ fn hides_alias_import_2() -> TestResult {
|
||||||
fn hides_alias_import_3() -> TestResult {
|
fn hides_alias_import_3() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export alias foo = "foo" }; use spam; hide spam [foo]; spam foo"#,
|
r#"module spam { export alias foo = "foo" }; use spam; hide spam [foo]; spam foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -301,7 +298,7 @@ fn hides_alias_import_3() -> TestResult {
|
||||||
fn hides_alias_import_4() -> TestResult {
|
fn hides_alias_import_4() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export alias foo = "foo" }; use spam foo; hide foo; foo"#,
|
r#"module spam { export alias foo = "foo" }; use spam foo; hide foo; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -309,7 +306,7 @@ fn hides_alias_import_4() -> TestResult {
|
||||||
fn hides_alias_import_5() -> TestResult {
|
fn hides_alias_import_5() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export alias foo = "foo" }; use spam *; hide foo; foo"#,
|
r#"module spam { export alias foo = "foo" }; use spam *; hide foo; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -317,7 +314,7 @@ fn hides_alias_import_5() -> TestResult {
|
||||||
fn hides_alias_import_6() -> TestResult {
|
fn hides_alias_import_6() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export alias foo = "foo" }; use spam *; hide spam *; foo"#,
|
r#"module spam { export alias foo = "foo" }; use spam *; hide spam *; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -338,7 +335,6 @@ fn hides_env_import_1() -> TestResult {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[ignore = "Re-enable after virtualenv update"]
|
|
||||||
fn hides_def_runs_env_import() -> TestResult {
|
fn hides_def_runs_env_import() -> TestResult {
|
||||||
run_test(
|
run_test(
|
||||||
r#"module spam { export-env { let-env foo = "foo" }; export def foo [] { "bar" } }; use spam foo; hide foo; $env.foo"#,
|
r#"module spam { export-env { let-env foo = "foo" }; export def foo [] { "bar" } }; use spam foo; hide foo; $env.foo"#,
|
||||||
|
@ -390,7 +386,7 @@ fn hide_shadowed_env() -> TestResult {
|
||||||
fn hides_all_decls_within_scope() -> TestResult {
|
fn hides_all_decls_within_scope() -> TestResult {
|
||||||
fail_test(
|
fail_test(
|
||||||
r#"module spam { export def foo [] { "bar" } }; def foo [] { "foo" }; use spam foo; hide foo; foo"#,
|
r#"module spam { export def foo [] { "bar" } }; def foo [] { "foo" }; use spam foo; hide foo; foo"#,
|
||||||
"", // we just care if it errors
|
"external_command",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -401,3 +397,35 @@ fn hides_all_envs_within_scope() -> TestResult {
|
||||||
"",
|
"",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn hides_main_import_1() -> TestResult {
|
||||||
|
fail_test(
|
||||||
|
r#"module spam { export def main [] { "foo" } }; use spam; hide spam; spam"#,
|
||||||
|
"external_command",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn hides_main_import_2() -> TestResult {
|
||||||
|
fail_test(
|
||||||
|
r#"module spam { export def main [] { "foo" } }; use spam; hide spam main; spam"#,
|
||||||
|
"external_command",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn hides_main_import_3() -> TestResult {
|
||||||
|
fail_test(
|
||||||
|
r#"module spam { export def main [] { "foo" } }; use spam; hide spam [ main ]; spam"#,
|
||||||
|
"external_command",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn hides_main_import_4() -> TestResult {
|
||||||
|
fail_test(
|
||||||
|
r#"module spam { export def main [] { "foo" } }; use spam; hide spam *; spam"#,
|
||||||
|
"external_command",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
|
@ -474,3 +474,91 @@ fn module_import_const_module_name() {
|
||||||
assert_eq!(actual.out, "foo");
|
assert_eq!(actual.out, "foo");
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn module_valid_def_name() {
|
||||||
|
let inp = &[r#"module spam { def spam [] { "spam" } }"#];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn module_invalid_def_name() {
|
||||||
|
let inp = &[r#"module spam { export def spam [] { "spam" } }"#];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.err.contains("named_as_module"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn module_valid_alias_name_1() {
|
||||||
|
let inp = &[r#"module spam { alias spam = "spam" }"#];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn module_valid_alias_name_2() {
|
||||||
|
let inp = &[r#"module spam { alias main = "spam" }"#];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn module_invalid_alias_name() {
|
||||||
|
let inp = &[r#"module spam { export alias spam = "spam" }"#];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.err.contains("named_as_module"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn module_main_alias_not_allowed() {
|
||||||
|
let inp = &[r#"module spam { export alias main = 'spam' }"#];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.err.contains("export_main_alias_not_allowed"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn module_valid_known_external_name() {
|
||||||
|
let inp = &[r#"module spam { extern spam [] }"#];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn module_invalid_known_external_name() {
|
||||||
|
let inp = &[r#"module spam { export extern spam [] }"#];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.err.contains("named_as_module"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn main_inside_module_is_main() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam {
|
||||||
|
export def main [] { 'foo' };
|
||||||
|
export def foo [] { main }
|
||||||
|
}"#,
|
||||||
|
"use spam foo",
|
||||||
|
"foo",
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "foo");
|
||||||
|
}
|
||||||
|
|
|
@ -1193,3 +1193,70 @@ fn overlay_use_and_reolad_keep_custom() {
|
||||||
assert_eq!(actual.out, "newfoonewfoonewfoo");
|
assert_eq!(actual.out, "newfoonewfoonewfoo");
|
||||||
assert_eq!(actual_repl.out, "newfoonewfoonewfoo");
|
assert_eq!(actual_repl.out, "newfoonewfoonewfoo");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn overlay_use_main() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { export def main [] { "spam" } }"#,
|
||||||
|
r#"overlay use spam"#,
|
||||||
|
r#"spam"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn overlay_use_main_prefix() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { export def main [] { "spam" } }"#,
|
||||||
|
r#"overlay use spam --prefix"#,
|
||||||
|
r#"spam"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn overlay_use_main_def_env() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { export def-env main [] { let-env SPAM = "spam" } }"#,
|
||||||
|
r#"overlay use spam"#,
|
||||||
|
r#"spam"#,
|
||||||
|
r#"$env.SPAM"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "spam");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn overlay_use_main_def_known_external() {
|
||||||
|
// note: requires installed cargo
|
||||||
|
let inp = &[
|
||||||
|
r#"module cargo { export extern main [] }"#,
|
||||||
|
r#"overlay use cargo"#,
|
||||||
|
r#"cargo --version"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.out.contains("cargo"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn overlay_use_main_not_exported() {
|
||||||
|
let inp = &[
|
||||||
|
r#"module spam { def main [] { "spam" } }"#,
|
||||||
|
r#"overlay use spam"#,
|
||||||
|
r#"spam"#,
|
||||||
|
];
|
||||||
|
|
||||||
|
let actual = nu!(cwd: ".", pipeline(&inp.join("; ")));
|
||||||
|
|
||||||
|
assert!(actual.err.contains("external_command"));
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue