2019-08-31 02:59:21 +02:00
|
|
|
use crate::commands::UnevaluatedCallInfo;
|
2020-04-27 04:04:54 +02:00
|
|
|
use crate::commands::WholeStreamCommand;
|
Extract core stuff into own crates
This commit extracts five new crates:
- nu-source, which contains the core source-code handling logic in Nu,
including Text, Span, and also the pretty.rs-based debug logic
- nu-parser, which is the parser and expander logic
- nu-protocol, which is the bulk of the types and basic conveniences
used by plugins
- nu-errors, which contains ShellError, ParseError and error handling
conveniences
- nu-textview, which is the textview plugin extracted into a crate
One of the major consequences of this refactor is that it's no longer
possible to `impl X for Spanned<Y>` outside of the `nu-source` crate, so
a lot of types became more concrete (Value became a concrete type
instead of Spanned<Value>, for example).
This also turned a number of inherent methods in the main nu crate into
plain functions (impl Value {} became a bunch of functions in the
`value` namespace in `crate::data::value`).
2019-11-26 03:30:48 +01:00
|
|
|
use crate::context::CommandRegistry;
|
2019-08-07 19:49:11 +02:00
|
|
|
use crate::prelude::*;
|
Extract core stuff into own crates
This commit extracts five new crates:
- nu-source, which contains the core source-code handling logic in Nu,
including Text, Span, and also the pretty.rs-based debug logic
- nu-parser, which is the parser and expander logic
- nu-protocol, which is the bulk of the types and basic conveniences
used by plugins
- nu-errors, which contains ShellError, ParseError and error handling
conveniences
- nu-textview, which is the textview plugin extracted into a crate
One of the major consequences of this refactor is that it's no longer
possible to `impl X for Spanned<Y>` outside of the `nu-source` crate, so
a lot of types became more concrete (Value became a concrete type
instead of Spanned<Value>, for example).
This also turned a number of inherent methods in the main nu crate into
plain functions (impl Value {} became a bunch of functions in the
`value` namespace in `crate::data::value`).
2019-11-26 03:30:48 +01:00
|
|
|
use nu_errors::ShellError;
|
|
|
|
use nu_protocol::{
|
2020-04-27 04:04:54 +02:00
|
|
|
CommandAction, Primitive, ReturnSuccess, Signature, SyntaxShape, UntaggedValue, Value,
|
Extract core stuff into own crates
This commit extracts five new crates:
- nu-source, which contains the core source-code handling logic in Nu,
including Text, Span, and also the pretty.rs-based debug logic
- nu-parser, which is the parser and expander logic
- nu-protocol, which is the bulk of the types and basic conveniences
used by plugins
- nu-errors, which contains ShellError, ParseError and error handling
conveniences
- nu-textview, which is the textview plugin extracted into a crate
One of the major consequences of this refactor is that it's no longer
possible to `impl X for Spanned<Y>` outside of the `nu-source` crate, so
a lot of types became more concrete (Value became a concrete type
instead of Spanned<Value>, for example).
This also turned a number of inherent methods in the main nu crate into
plain functions (impl Value {} became a bunch of functions in the
`value` namespace in `crate::data::value`).
2019-11-26 03:30:48 +01:00
|
|
|
};
|
2020-04-27 04:04:54 +02:00
|
|
|
use nu_source::Tagged;
|
|
|
|
use std::path::PathBuf;
|
2019-08-07 19:49:11 +02:00
|
|
|
|
2019-08-14 19:02:39 +02:00
|
|
|
pub struct Enter;
|
2019-08-09 06:51:21 +02:00
|
|
|
|
2020-04-27 04:04:54 +02:00
|
|
|
#[derive(Deserialize)]
|
|
|
|
pub struct EnterArgs {
|
|
|
|
location: Tagged<PathBuf>,
|
2020-06-12 02:37:43 +02:00
|
|
|
encoding: Option<Tagged<String>>,
|
2020-04-27 04:04:54 +02:00
|
|
|
}
|
|
|
|
|
2020-05-29 10:22:52 +02:00
|
|
|
#[async_trait]
|
2020-04-27 04:04:54 +02:00
|
|
|
impl WholeStreamCommand for Enter {
|
2019-08-14 19:02:39 +02:00
|
|
|
fn name(&self) -> &str {
|
|
|
|
"enter"
|
2019-08-07 19:49:11 +02:00
|
|
|
}
|
|
|
|
|
Extract core stuff into own crates
This commit extracts five new crates:
- nu-source, which contains the core source-code handling logic in Nu,
including Text, Span, and also the pretty.rs-based debug logic
- nu-parser, which is the parser and expander logic
- nu-protocol, which is the bulk of the types and basic conveniences
used by plugins
- nu-errors, which contains ShellError, ParseError and error handling
conveniences
- nu-textview, which is the textview plugin extracted into a crate
One of the major consequences of this refactor is that it's no longer
possible to `impl X for Spanned<Y>` outside of the `nu-source` crate, so
a lot of types became more concrete (Value became a concrete type
instead of Spanned<Value>, for example).
This also turned a number of inherent methods in the main nu crate into
plain functions (impl Value {} became a bunch of functions in the
`value` namespace in `crate::data::value`).
2019-11-26 03:30:48 +01:00
|
|
|
fn signature(&self) -> Signature {
|
2020-06-12 02:37:43 +02:00
|
|
|
Signature::build("enter")
|
|
|
|
.required(
|
|
|
|
"location",
|
|
|
|
SyntaxShape::Path,
|
|
|
|
"the location to create a new shell from",
|
|
|
|
)
|
|
|
|
.named(
|
|
|
|
"encoding",
|
|
|
|
SyntaxShape::String,
|
|
|
|
"encoding to use to open file",
|
|
|
|
Some('e'),
|
|
|
|
)
|
2019-08-14 19:02:39 +02:00
|
|
|
}
|
|
|
|
|
2019-08-30 00:52:32 +02:00
|
|
|
fn usage(&self) -> &str {
|
2020-06-12 02:37:43 +02:00
|
|
|
r#"Create a new shell and begin at this path.
|
|
|
|
|
|
|
|
Multiple encodings are supported for reading text files by using
|
|
|
|
the '--encoding <encoding>' parameter. Here is an example of a few:
|
|
|
|
big5, euc-jp, euc-kr, gbk, iso-8859-1, utf-16, cp1252, latin5
|
|
|
|
|
|
|
|
For a more complete list of encodings please refer to the encoding_rs
|
|
|
|
documentation link at https://docs.rs/encoding_rs/0.8.23/encoding_rs/#statics"#
|
2019-08-30 00:52:32 +02:00
|
|
|
}
|
|
|
|
|
2020-05-29 10:22:52 +02:00
|
|
|
async fn run(
|
2019-08-14 19:02:39 +02:00
|
|
|
&self,
|
2020-04-27 04:04:54 +02:00
|
|
|
args: CommandArgs,
|
Extract core stuff into own crates
This commit extracts five new crates:
- nu-source, which contains the core source-code handling logic in Nu,
including Text, Span, and also the pretty.rs-based debug logic
- nu-parser, which is the parser and expander logic
- nu-protocol, which is the bulk of the types and basic conveniences
used by plugins
- nu-errors, which contains ShellError, ParseError and error handling
conveniences
- nu-textview, which is the textview plugin extracted into a crate
One of the major consequences of this refactor is that it's no longer
possible to `impl X for Spanned<Y>` outside of the `nu-source` crate, so
a lot of types became more concrete (Value became a concrete type
instead of Spanned<Value>, for example).
This also turned a number of inherent methods in the main nu crate into
plain functions (impl Value {} became a bunch of functions in the
`value` namespace in `crate::data::value`).
2019-11-26 03:30:48 +01:00
|
|
|
registry: &CommandRegistry,
|
2019-08-24 21:36:19 +02:00
|
|
|
) -> Result<OutputStream, ShellError> {
|
2020-05-16 05:18:24 +02:00
|
|
|
enter(args, registry)
|
2020-04-27 04:04:54 +02:00
|
|
|
}
|
2020-05-12 07:17:17 +02:00
|
|
|
|
2020-05-18 14:56:01 +02:00
|
|
|
fn examples(&self) -> Vec<Example> {
|
|
|
|
vec![
|
2020-05-12 07:17:17 +02:00
|
|
|
Example {
|
|
|
|
description: "Enter a path as a new shell",
|
|
|
|
example: "enter ../projectB",
|
2020-05-18 14:56:01 +02:00
|
|
|
result: None,
|
2020-05-12 07:17:17 +02:00
|
|
|
},
|
|
|
|
Example {
|
|
|
|
description: "Enter a file as a new shell",
|
|
|
|
example: "enter package.json",
|
2020-05-18 14:56:01 +02:00
|
|
|
result: None,
|
2020-05-12 07:17:17 +02:00
|
|
|
},
|
2020-06-12 02:37:43 +02:00
|
|
|
Example {
|
|
|
|
description: "Enters file with iso-8859-1 encoding",
|
|
|
|
example: "enter file.csv --encoding iso-8859-1",
|
|
|
|
result: None,
|
|
|
|
},
|
2020-05-12 07:17:17 +02:00
|
|
|
]
|
|
|
|
}
|
2020-04-27 04:04:54 +02:00
|
|
|
}
|
2019-08-30 00:52:32 +02:00
|
|
|
|
2020-05-16 05:18:24 +02:00
|
|
|
fn enter(raw_args: CommandArgs, registry: &CommandRegistry) -> Result<OutputStream, ShellError> {
|
|
|
|
let registry = registry.clone();
|
|
|
|
let stream = async_stream! {
|
|
|
|
let scope = raw_args.call_info.scope.clone();
|
|
|
|
let shell_manager = raw_args.shell_manager.clone();
|
|
|
|
let head = raw_args.call_info.args.head.clone();
|
|
|
|
let ctrl_c = raw_args.ctrl_c.clone();
|
2020-05-30 20:31:50 +02:00
|
|
|
let current_errors = raw_args.current_errors.clone();
|
2020-05-16 05:18:24 +02:00
|
|
|
let host = raw_args.host.clone();
|
|
|
|
let tag = raw_args.call_info.name_tag.clone();
|
2020-06-12 02:37:43 +02:00
|
|
|
let (EnterArgs { location, encoding }, _) = raw_args.process(®istry).await?;
|
2020-05-16 05:18:24 +02:00
|
|
|
let location_string = location.display().to_string();
|
|
|
|
let location_clone = location_string.clone();
|
|
|
|
|
|
|
|
if location_string.starts_with("help") {
|
|
|
|
let spec = location_string.split(':').collect::<Vec<&str>>();
|
|
|
|
|
|
|
|
if spec.len() == 2 {
|
|
|
|
let (_, command) = (spec[0], spec[1]);
|
|
|
|
|
|
|
|
if registry.has(command) {
|
|
|
|
yield Ok(ReturnSuccess::Action(CommandAction::EnterHelpShell(
|
|
|
|
UntaggedValue::string(command).into_value(Tag::unknown()),
|
|
|
|
)));
|
|
|
|
return;
|
|
|
|
}
|
2020-04-27 04:04:54 +02:00
|
|
|
}
|
2020-05-16 05:18:24 +02:00
|
|
|
yield Ok(ReturnSuccess::Action(CommandAction::EnterHelpShell(
|
|
|
|
UntaggedValue::nothing().into_value(Tag::unknown()),
|
|
|
|
)));
|
|
|
|
} else if location.is_dir() {
|
|
|
|
yield Ok(ReturnSuccess::Action(CommandAction::EnterShell(
|
|
|
|
location_clone,
|
|
|
|
)));
|
|
|
|
} else {
|
2020-04-27 04:04:54 +02:00
|
|
|
// If it's a file, attempt to open the file as a value and enter it
|
2020-05-16 05:18:24 +02:00
|
|
|
let cwd = shell_manager.path();
|
2019-08-31 02:59:21 +02:00
|
|
|
|
2020-04-27 04:04:54 +02:00
|
|
|
let full_path = std::path::PathBuf::from(cwd);
|
2019-08-31 02:59:21 +02:00
|
|
|
|
2020-04-27 04:04:54 +02:00
|
|
|
let (file_extension, contents, contents_tag) =
|
|
|
|
crate::commands::open::fetch(
|
|
|
|
&full_path,
|
|
|
|
&PathBuf::from(location_clone),
|
|
|
|
tag.span,
|
2020-06-12 02:37:43 +02:00
|
|
|
match encoding {
|
|
|
|
Some(e) => e.to_string(),
|
|
|
|
_ => "".to_string()
|
|
|
|
}
|
2020-04-27 04:04:54 +02:00
|
|
|
).await?;
|
2019-08-31 02:59:21 +02:00
|
|
|
|
2020-04-27 04:04:54 +02:00
|
|
|
match contents {
|
|
|
|
UntaggedValue::Primitive(Primitive::String(_)) => {
|
|
|
|
let tagged_contents = contents.into_value(&contents_tag);
|
|
|
|
|
|
|
|
if let Some(extension) = file_extension {
|
2020-05-04 10:44:33 +02:00
|
|
|
let command_name = format!("from {}", extension);
|
2020-04-27 04:04:54 +02:00
|
|
|
if let Some(converter) =
|
|
|
|
registry.get_command(&command_name)
|
|
|
|
{
|
|
|
|
let new_args = RawCommandArgs {
|
2020-05-16 05:18:24 +02:00
|
|
|
host,
|
|
|
|
ctrl_c,
|
2020-05-30 20:31:50 +02:00
|
|
|
current_errors,
|
2020-05-16 05:18:24 +02:00
|
|
|
shell_manager,
|
2020-04-27 04:04:54 +02:00
|
|
|
call_info: UnevaluatedCallInfo {
|
|
|
|
args: nu_protocol::hir::Call {
|
2020-05-16 05:18:24 +02:00
|
|
|
head,
|
2020-04-27 04:04:54 +02:00
|
|
|
positional: None,
|
|
|
|
named: None,
|
|
|
|
span: Span::unknown(),
|
|
|
|
is_last: false,
|
|
|
|
},
|
2020-05-16 05:18:24 +02:00
|
|
|
name_tag: tag.clone(),
|
|
|
|
scope: scope.clone()
|
2020-04-27 04:04:54 +02:00
|
|
|
},
|
|
|
|
};
|
|
|
|
let mut result = converter.run(
|
|
|
|
new_args.with_input(vec![tagged_contents]),
|
|
|
|
®istry,
|
2020-05-29 10:22:52 +02:00
|
|
|
).await;
|
2020-04-27 04:04:54 +02:00
|
|
|
let result_vec: Vec<Result<ReturnSuccess, ShellError>> =
|
|
|
|
result.drain_vec().await;
|
|
|
|
for res in result_vec {
|
|
|
|
match res {
|
|
|
|
Ok(ReturnSuccess::Value(Value {
|
|
|
|
value,
|
|
|
|
..
|
|
|
|
})) => {
|
|
|
|
yield Ok(ReturnSuccess::Action(CommandAction::EnterValueShell(
|
|
|
|
Value {
|
|
|
|
value,
|
|
|
|
tag: contents_tag.clone(),
|
|
|
|
})));
|
2019-08-31 02:59:21 +02:00
|
|
|
}
|
2020-04-27 04:04:54 +02:00
|
|
|
x => yield x,
|
2019-08-31 02:59:21 +02:00
|
|
|
}
|
|
|
|
}
|
2020-04-27 04:04:54 +02:00
|
|
|
} else {
|
|
|
|
yield Ok(ReturnSuccess::Action(CommandAction::EnterValueShell(tagged_contents)));
|
2019-08-31 02:59:21 +02:00
|
|
|
}
|
2020-04-27 04:04:54 +02:00
|
|
|
} else {
|
|
|
|
yield Ok(ReturnSuccess::Action(CommandAction::EnterValueShell(tagged_contents)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
let tagged_contents = contents.into_value(contents_tag);
|
|
|
|
|
|
|
|
yield Ok(ReturnSuccess::Action(CommandAction::EnterValueShell(tagged_contents)));
|
2019-08-31 02:59:21 +02:00
|
|
|
}
|
|
|
|
}
|
2020-05-16 05:18:24 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(stream.to_output_stream())
|
2019-08-07 19:49:11 +02:00
|
|
|
}
|
2020-05-18 14:56:01 +02:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::Enter;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn examples_work_as_expected() {
|
|
|
|
use crate::examples::test as test_examples;
|
|
|
|
|
|
|
|
test_examples(Enter {})
|
|
|
|
}
|
|
|
|
}
|