mirror of
https://github.com/nushell/nushell
synced 2025-01-14 14:14:13 +00:00
Fix issue 6223 (#6241)
* Fix6223 * clippy fix Co-authored-by: Frank <v-frankz@microsoft.com>
This commit is contained in:
parent
34ab4d8360
commit
a217bc0715
3 changed files with 88 additions and 26 deletions
|
@ -1,9 +1,10 @@
|
||||||
use super::get_shells;
|
use super::{get_current_shell, get_shells};
|
||||||
use nu_engine::{current_dir, CallExt};
|
use nu_engine::{current_dir, CallExt};
|
||||||
use nu_protocol::ast::Call;
|
use nu_protocol::ast::Call;
|
||||||
use nu_protocol::engine::{Command, EngineState, Stack};
|
use nu_protocol::engine::{Command, EngineState, Stack};
|
||||||
use nu_protocol::{
|
use nu_protocol::{
|
||||||
Category, Example, PipelineData, ShellError, Signature, Spanned, SyntaxShape, Value,
|
Category, Example, IntoInterruptiblePipelineData, PipelineData, ShellError, Signature, Spanned,
|
||||||
|
SyntaxShape, Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Source a file for environment variables.
|
/// Source a file for environment variables.
|
||||||
|
@ -17,7 +18,7 @@ impl Command for GotoShell {
|
||||||
|
|
||||||
fn signature(&self) -> Signature {
|
fn signature(&self) -> Signature {
|
||||||
Signature::build("g")
|
Signature::build("g")
|
||||||
.required(
|
.optional(
|
||||||
"shell_number",
|
"shell_number",
|
||||||
SyntaxShape::Int,
|
SyntaxShape::Int,
|
||||||
"shell number to change to",
|
"shell number to change to",
|
||||||
|
@ -26,7 +27,7 @@ impl Command for GotoShell {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn usage(&self) -> &str {
|
fn usage(&self) -> &str {
|
||||||
"Switch to a given shell."
|
"Switch to a given shell, or list all shells if no given shell number."
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run(
|
fn run(
|
||||||
|
@ -36,7 +37,8 @@ impl Command for GotoShell {
|
||||||
call: &Call,
|
call: &Call,
|
||||||
_input: PipelineData,
|
_input: PipelineData,
|
||||||
) -> Result<PipelineData, ShellError> {
|
) -> Result<PipelineData, ShellError> {
|
||||||
let new_shell: Spanned<i64> = call.req(engine_state, stack, 0)?;
|
let span = call.head;
|
||||||
|
let new_shell: Option<Spanned<i64>> = call.opt(engine_state, stack, 0)?;
|
||||||
|
|
||||||
let cwd = current_dir(engine_state, stack)?;
|
let cwd = current_dir(engine_state, stack)?;
|
||||||
let cwd = Value::String {
|
let cwd = Value::String {
|
||||||
|
@ -46,34 +48,62 @@ impl Command for GotoShell {
|
||||||
|
|
||||||
let shells = get_shells(engine_state, stack, cwd);
|
let shells = get_shells(engine_state, stack, cwd);
|
||||||
|
|
||||||
let new_path = if let Some(v) = shells.get(new_shell.item as usize) {
|
match new_shell {
|
||||||
v.clone()
|
Some(shell_span) => {
|
||||||
} else {
|
let new_path = if let Some(v) = shells.get(shell_span.item as usize) {
|
||||||
return Err(ShellError::NotFound(new_shell.span));
|
v.clone()
|
||||||
};
|
} else {
|
||||||
|
return Err(ShellError::NotFound(shell_span.span));
|
||||||
|
};
|
||||||
|
|
||||||
stack.add_env_var(
|
stack.add_env_var(
|
||||||
"NUSHELL_SHELLS".into(),
|
"NUSHELL_SHELLS".into(),
|
||||||
Value::List {
|
Value::List {
|
||||||
vals: shells,
|
vals: shells,
|
||||||
span: call.head,
|
span: call.head,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
stack.add_env_var(
|
stack.add_env_var(
|
||||||
"NUSHELL_CURRENT_SHELL".into(),
|
"NUSHELL_CURRENT_SHELL".into(),
|
||||||
Value::Int {
|
Value::Int {
|
||||||
val: new_shell.item,
|
val: shell_span.item,
|
||||||
span: call.head,
|
span: call.head,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
stack.add_env_var("PWD".into(), new_path);
|
stack.add_env_var("PWD".into(), new_path);
|
||||||
|
|
||||||
Ok(PipelineData::new(call.head))
|
Ok(PipelineData::new(call.head))
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
let current_shell = get_current_shell(engine_state, stack);
|
||||||
|
|
||||||
|
Ok(shells
|
||||||
|
.into_iter()
|
||||||
|
.enumerate()
|
||||||
|
.map(move |(idx, val)| Value::Record {
|
||||||
|
cols: vec!["active".to_string(), "path".to_string()],
|
||||||
|
vals: vec![
|
||||||
|
Value::Bool {
|
||||||
|
val: idx == current_shell,
|
||||||
|
span,
|
||||||
|
},
|
||||||
|
val,
|
||||||
|
],
|
||||||
|
span,
|
||||||
|
})
|
||||||
|
.into_pipeline_data(None))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn examples(&self) -> Vec<Example> {
|
fn examples(&self) -> Vec<Example> {
|
||||||
vec![
|
vec![
|
||||||
|
Example {
|
||||||
|
description: "Lists all open shells",
|
||||||
|
example: r#"g"#,
|
||||||
|
result: None,
|
||||||
|
},
|
||||||
Example {
|
Example {
|
||||||
description: "Make two directories and enter new shells for them, use `g` to jump to the specific shell",
|
description: "Make two directories and enter new shells for them, use `g` to jump to the specific shell",
|
||||||
example: r#"mkdir foo bar; enter foo; enter ../bar; g 1"#,
|
example: r#"mkdir foo bar; enter foo; enter ../bar; g 1"#,
|
||||||
|
|
31
crates/nu-command/tests/commands/g.rs
Normal file
31
crates/nu-command/tests/commands/g.rs
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
use nu_test_support::{nu, pipeline};
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn list_shells() {
|
||||||
|
let actual = nu!(
|
||||||
|
cwd: ".", pipeline(
|
||||||
|
r#"g | get path | length "#
|
||||||
|
));
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "1");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn enter_shell() {
|
||||||
|
let actual = nu!(
|
||||||
|
cwd: ".", pipeline(
|
||||||
|
r#"g 0"#
|
||||||
|
));
|
||||||
|
|
||||||
|
assert!(actual.err.is_empty());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn enter_not_exist_shell() {
|
||||||
|
let actual = nu!(
|
||||||
|
cwd: ".", pipeline(
|
||||||
|
r#"g 1"#
|
||||||
|
));
|
||||||
|
|
||||||
|
assert!(actual.err.contains("Not found"));
|
||||||
|
}
|
|
@ -22,6 +22,7 @@ mod find;
|
||||||
mod first;
|
mod first;
|
||||||
mod flatten;
|
mod flatten;
|
||||||
mod format;
|
mod format;
|
||||||
|
mod g;
|
||||||
mod get;
|
mod get;
|
||||||
mod group_by;
|
mod group_by;
|
||||||
mod hash_;
|
mod hash_;
|
||||||
|
|
Loading…
Reference in a new issue