use crate::commands::classified::block::run_block; use crate::commands::classified::maybe_text_codec::{MaybeTextCodec, StringOrBinary}; use crate::evaluation_context::EvaluationContext; use crate::path::canonicalize; use crate::prelude::*; #[cfg(feature = "rustyline-support")] use crate::shell::Helper; use crate::EnvironmentSyncer; use futures_codec::FramedRead; use nu_errors::ShellError; use nu_parser::ParserScope; use nu_protocol::hir::{ClassifiedCommand, Expression, InternalCommand, Literal, NamedArguments}; use nu_protocol::{Primitive, ReturnSuccess, UntaggedValue, Value}; use log::{debug, trace}; #[cfg(feature = "rustyline-support")] use rustyline::{ self, config::Configurer, config::{ColorMode, CompletionType, Config}, error::ReadlineError, At, Cmd, Editor, KeyPress, Movement, Word, }; use std::error::Error; use std::iter::Iterator; use std::path::{Path, PathBuf}; use std::sync::atomic::Ordering; pub fn search_paths() -> Vec { use std::env; let mut search_paths = Vec::new(); // Automatically add path `nu` is in as a search path if let Ok(exe_path) = env::current_exe() { if let Some(exe_dir) = exe_path.parent() { search_paths.push(exe_dir.to_path_buf()); } } if let Ok(config) = nu_data::config::config(Tag::unknown()) { if let Some(plugin_dirs) = config.get("plugin_dirs") { if let Value { value: UntaggedValue::Table(pipelines), .. } = plugin_dirs { for pipeline in pipelines { if let Ok(plugin_dir) = pipeline.as_string() { search_paths.push(PathBuf::from(plugin_dir)); } } } } } search_paths } pub fn create_default_context(interactive: bool) -> Result> { let context = EvaluationContext::basic()?; { use crate::commands::*; context.add_commands(vec![ // Fundamentals whole_stream_command(NuPlugin), whole_stream_command(Set), whole_stream_command(SetEnv), whole_stream_command(Def), whole_stream_command(Source), // System/file operations whole_stream_command(Exec), whole_stream_command(Pwd), whole_stream_command(Ls), whole_stream_command(Du), whole_stream_command(Cd), whole_stream_command(Remove), whole_stream_command(Open), whole_stream_command(Config), whole_stream_command(ConfigGet), whole_stream_command(ConfigSet), whole_stream_command(ConfigSetInto), whole_stream_command(ConfigClear), whole_stream_command(ConfigLoad), whole_stream_command(ConfigRemove), whole_stream_command(ConfigPath), whole_stream_command(Help), whole_stream_command(History), whole_stream_command(Save), whole_stream_command(Touch), whole_stream_command(Cpy), whole_stream_command(Date), whole_stream_command(DateListTimeZone), whole_stream_command(DateNow), whole_stream_command(DateToTable), whole_stream_command(DateToTimeZone), whole_stream_command(DateFormat), whole_stream_command(Cal), whole_stream_command(Mkdir), whole_stream_command(Mv), whole_stream_command(Kill), whole_stream_command(Version), whole_stream_command(Clear), whole_stream_command(Describe), whole_stream_command(Which), whole_stream_command(Debug), whole_stream_command(WithEnv), whole_stream_command(Do), whole_stream_command(Sleep), // Statistics whole_stream_command(Size), whole_stream_command(Count), whole_stream_command(Benchmark), // Metadata whole_stream_command(Tags), // Shells whole_stream_command(Next), whole_stream_command(Previous), whole_stream_command(Shells), whole_stream_command(Enter), whole_stream_command(Exit), // Viz whole_stream_command(Chart), // Viewers whole_stream_command(Autoview), whole_stream_command(Table), // Text manipulation whole_stream_command(Hash), whole_stream_command(HashBase64), whole_stream_command(Split), whole_stream_command(SplitColumn), whole_stream_command(SplitRow), whole_stream_command(SplitChars), whole_stream_command(Lines), whole_stream_command(Echo), whole_stream_command(Parse), whole_stream_command(Str), whole_stream_command(StrToDecimal), whole_stream_command(StrToInteger), whole_stream_command(StrDowncase), whole_stream_command(StrUpcase), whole_stream_command(StrCapitalize), whole_stream_command(StrFindReplace), whole_stream_command(StrFrom), whole_stream_command(StrSubstring), whole_stream_command(StrSet), whole_stream_command(StrToDatetime), whole_stream_command(StrContains), whole_stream_command(StrIndexOf), whole_stream_command(StrTrim), whole_stream_command(StrTrimLeft), whole_stream_command(StrTrimRight), whole_stream_command(StrStartsWith), whole_stream_command(StrEndsWith), whole_stream_command(StrCollect), whole_stream_command(StrLength), whole_stream_command(StrLPad), whole_stream_command(StrReverse), whole_stream_command(StrRPad), whole_stream_command(StrCamelCase), whole_stream_command(StrPascalCase), whole_stream_command(StrKebabCase), whole_stream_command(StrSnakeCase), whole_stream_command(StrScreamingSnakeCase), whole_stream_command(BuildString), whole_stream_command(Ansi), whole_stream_command(Char), // Column manipulation whole_stream_command(Move), whole_stream_command(Reject), whole_stream_command(Select), whole_stream_command(Get), whole_stream_command(Update), whole_stream_command(Insert), whole_stream_command(IntoInt), whole_stream_command(SplitBy), // Row manipulation whole_stream_command(Reverse), whole_stream_command(Append), whole_stream_command(Prepend), whole_stream_command(SortBy), whole_stream_command(GroupBy), whole_stream_command(GroupByDate), whole_stream_command(First), whole_stream_command(Last), whole_stream_command(Every), whole_stream_command(Nth), whole_stream_command(Drop), whole_stream_command(Format), whole_stream_command(FileSize), whole_stream_command(Where), whole_stream_command(If), whole_stream_command(Compact), whole_stream_command(Default), whole_stream_command(Skip), whole_stream_command(SkipUntil), whole_stream_command(SkipWhile), whole_stream_command(Keep), whole_stream_command(KeepUntil), whole_stream_command(KeepWhile), whole_stream_command(Range), whole_stream_command(Rename), whole_stream_command(Uniq), whole_stream_command(Each), whole_stream_command(EachGroup), whole_stream_command(EachWindow), whole_stream_command(Empty), // Table manipulation whole_stream_command(Flatten), whole_stream_command(Move), whole_stream_command(Merge), whole_stream_command(Shuffle), whole_stream_command(Wrap), whole_stream_command(Pivot), whole_stream_command(Headers), whole_stream_command(Reduce), // Data processing whole_stream_command(Histogram), whole_stream_command(Autoenv), whole_stream_command(AutoenvTrust), whole_stream_command(AutoenvUnTrust), whole_stream_command(Math), whole_stream_command(MathAbs), whole_stream_command(MathAverage), whole_stream_command(MathEval), whole_stream_command(MathMedian), whole_stream_command(MathMinimum), whole_stream_command(MathMode), whole_stream_command(MathMaximum), whole_stream_command(MathStddev), whole_stream_command(MathSummation), whole_stream_command(MathVariance), whole_stream_command(MathProduct), whole_stream_command(MathRound), whole_stream_command(MathFloor), whole_stream_command(MathCeil), // File format output whole_stream_command(To), whole_stream_command(ToCSV), whole_stream_command(ToHTML), whole_stream_command(ToJSON), whole_stream_command(ToMarkdown), whole_stream_command(ToTOML), whole_stream_command(ToTSV), whole_stream_command(ToURL), whole_stream_command(ToYAML), whole_stream_command(ToXML), // File format input whole_stream_command(From), whole_stream_command(FromCSV), whole_stream_command(FromEML), whole_stream_command(FromTSV), whole_stream_command(FromSSV), whole_stream_command(FromINI), whole_stream_command(FromJSON), whole_stream_command(FromODS), whole_stream_command(FromTOML), whole_stream_command(FromURL), whole_stream_command(FromXLSX), whole_stream_command(FromXML), whole_stream_command(FromYAML), whole_stream_command(FromYML), whole_stream_command(FromIcs), whole_stream_command(FromVcf), // "Private" commands (not intended to be accessed directly) whole_stream_command(RunExternalCommand { interactive }), // Random value generation whole_stream_command(Random), whole_stream_command(RandomBool), whole_stream_command(RandomDice), #[cfg(feature = "uuid_crate")] whole_stream_command(RandomUUID), whole_stream_command(RandomInteger), whole_stream_command(RandomDecimal), whole_stream_command(RandomChars), // Path whole_stream_command(PathBasename), whole_stream_command(PathCommand), whole_stream_command(PathDirname), whole_stream_command(PathExists), whole_stream_command(PathExpand), whole_stream_command(PathExtension), whole_stream_command(PathFilestem), whole_stream_command(PathType), // Url whole_stream_command(UrlCommand), whole_stream_command(UrlScheme), whole_stream_command(UrlPath), whole_stream_command(UrlHost), whole_stream_command(UrlQuery), whole_stream_command(Seq), whole_stream_command(SeqDates), ]); #[cfg(feature = "clipboard-cli")] { context.add_commands(vec![whole_stream_command(crate::commands::clip::Clip)]); } } Ok(context) } pub async fn run_script_file( file_contents: String, redirect_stdin: bool, ) -> Result<(), Box> { let mut syncer = EnvironmentSyncer::new(); let mut context = create_default_context(false)?; let config = syncer.get_config(); context.configure(&config, |_, ctx| { syncer.load_environment(); syncer.sync_env_vars(ctx); syncer.sync_path_vars(ctx); if let Err(reason) = syncer.autoenv(ctx) { print_err(reason, &Text::from("")); } let _ = register_plugins(ctx); let _ = configure_ctrl_c(ctx); }); let _ = run_startup_commands(&mut context, &config).await; run_script_standalone(file_contents, redirect_stdin, &context, true).await?; Ok(()) } #[cfg(feature = "rustyline-support")] fn convert_rustyline_result_to_string(input: Result) -> LineResult { match input { Ok(s) if s == "history -c" || s == "history --clear" => LineResult::ClearHistory, Ok(s) => LineResult::Success(s), Err(ReadlineError::Interrupted) => LineResult::CtrlC, Err(ReadlineError::Eof) => LineResult::CtrlD, Err(err) => { outln!("Error: {:?}", err); LineResult::Break } } } /// The entry point for the CLI. Will register all known internal commands, load experimental commands, load plugins, then prepare the prompt and line reader for input. #[cfg(feature = "rustyline-support")] pub async fn cli(mut context: EvaluationContext) -> Result<(), Box> { let mut syncer = EnvironmentSyncer::new(); let configuration = syncer.get_config(); let mut rl = default_rustyline_editor_configuration(); context.configure(&configuration, |config, ctx| { syncer.load_environment(); syncer.sync_env_vars(ctx); syncer.sync_path_vars(ctx); if let Err(reason) = syncer.autoenv(ctx) { print_err(reason, &Text::from("")); } let _ = configure_ctrl_c(ctx); let _ = configure_rustyline_editor(&mut rl, config); let helper = Some(nu_line_editor_helper(ctx, config)); rl.set_helper(helper); }); let _ = run_startup_commands(&mut context, &configuration).await; // Give ourselves a scope to work in context.scope.enter_scope(); let history_path = crate::commands::history::history_path(&configuration); let _ = rl.load_history(&history_path); let mut session_text = String::new(); let mut line_start: usize = 0; let skip_welcome_message = configuration .var("skip_welcome_message") .map(|x| x.is_true()) .unwrap_or(false); if !skip_welcome_message { println!( "Welcome to Nushell {} (type 'help' for more info)", clap::crate_version!() ); } #[cfg(windows)] { let _ = ansi_term::enable_ansi_support(); } let mut ctrlcbreak = false; loop { if context.ctrl_c.load(Ordering::SeqCst) { context.ctrl_c.store(false, Ordering::SeqCst); continue; } let cwd = context.shell_manager.path(); let colored_prompt = { if let Some(prompt) = configuration.var("prompt") { let prompt_line = prompt.as_string()?; context.scope.enter_scope(); let (prompt_block, err) = nu_parser::parse(&prompt_line, 0, &context.scope); if err.is_some() { use crate::git::current_branch; context.scope.exit_scope(); format!( "\x1b[32m{}{}\x1b[m> ", cwd, match current_branch() { Some(s) => format!("({})", s), None => "".to_string(), } ) } else { // let env = context.get_env(); let run_result = run_block(&prompt_block, &context, InputStream::empty()).await; context.scope.exit_scope(); match run_result { Ok(result) => match result.collect_string(Tag::unknown()).await { Ok(string_result) => { let errors = context.get_errors(); context.maybe_print_errors(Text::from(prompt_line)); context.clear_errors(); if !errors.is_empty() { "> ".to_string() } else { string_result.item } } Err(e) => { crate::cli::print_err(e, &Text::from(prompt_line)); context.clear_errors(); "> ".to_string() } }, Err(e) => { crate::cli::print_err(e, &Text::from(prompt_line)); context.clear_errors(); "> ".to_string() } } } } else { use crate::git::current_branch; format!( "\x1b[32m{}{}\x1b[m> ", cwd, match current_branch() { Some(s) => format!("({})", s), None => "".to_string(), } ) } }; let prompt = { if let Ok(bytes) = strip_ansi_escapes::strip(&colored_prompt) { String::from_utf8_lossy(&bytes).to_string() } else { "> ".to_string() } }; rl.helper_mut().expect("No helper").colored_prompt = colored_prompt; let mut initial_command = Some(String::new()); let mut readline = Err(ReadlineError::Eof); while let Some(ref cmd) = initial_command { readline = rl.readline_with_initial(&prompt, (&cmd, "")); initial_command = None; } if let Ok(line) = &readline { line_start = session_text.len(); session_text.push_str(line); session_text.push('\n'); } let line = match convert_rustyline_result_to_string(readline) { LineResult::Success(_) => { process_script( &session_text[line_start..], &context, false, line_start, true, ) .await } x => x, }; // Check the config to see if we need to update the path // TODO: make sure config is cached so we don't path this load every call // FIXME: we probably want to be a bit more graceful if we can't set the environment context.configure(&configuration, |config, ctx| { if syncer.did_config_change() { syncer.reload(); syncer.sync_env_vars(ctx); syncer.sync_path_vars(ctx); } if let Err(reason) = syncer.autoenv(ctx) { print_err(reason, &Text::from("")); } let _ = configure_rustyline_editor(&mut rl, config); }); match line { LineResult::Success(line) => { rl.add_history_entry(&line); let _ = rl.save_history(&history_path); context.maybe_print_errors(Text::from(session_text.clone())); } LineResult::ClearHistory => { rl.clear_history(); let _ = rl.save_history(&history_path); } LineResult::Error(line, err) => { rl.add_history_entry(&line); let _ = rl.save_history(&history_path); context.with_host(|_host| { print_err(err, &Text::from(session_text.clone())); }); context.maybe_print_errors(Text::from(session_text.clone())); } LineResult::CtrlC => { let config_ctrlc_exit = config::config(Tag::unknown())? .get("ctrlc_exit") .map(|s| s.value.is_true()) .unwrap_or(false); // default behavior is to allow CTRL-C spamming similar to other shells if !config_ctrlc_exit { continue; } if ctrlcbreak { let _ = rl.save_history(&history_path); std::process::exit(0); } else { context.with_host(|host| host.stdout("CTRL-C pressed (again to quit)")); ctrlcbreak = true; continue; } } LineResult::CtrlD => { context.shell_manager.remove_at_current(); if context.shell_manager.is_empty() { break; } } LineResult::Break => { break; } } ctrlcbreak = false; } // we are ok if we can not save history let _ = rl.save_history(&history_path); Ok(()) } pub fn register_plugins(context: &mut EvaluationContext) -> Result<(), ShellError> { if let Ok(plugins) = crate::plugin::scan(search_paths()) { context.add_commands( plugins .into_iter() .filter(|p| !context.is_command_registered(p.name())) .collect(), ); } Ok(()) } fn configure_ctrl_c(_context: &mut EvaluationContext) -> Result<(), Box> { #[cfg(feature = "ctrlc")] { let cc = _context.ctrl_c.clone(); ctrlc::set_handler(move || { cc.store(true, Ordering::SeqCst); })?; if _context.ctrl_c.load(Ordering::SeqCst) { _context.ctrl_c.store(false, Ordering::SeqCst); } } Ok(()) } async fn run_startup_commands( context: &mut EvaluationContext, config: &dyn nu_data::config::Conf, ) -> Result<(), ShellError> { if let Some(commands) = config.var("startup") { match commands { Value { value: UntaggedValue::Table(pipelines), .. } => { for pipeline in pipelines { if let Ok(pipeline_string) = pipeline.as_string() { let _ = run_script_standalone(pipeline_string, false, context, false).await; } } } _ => { return Err(ShellError::untagged_runtime_error( "expected a table of pipeline strings as startup commands", )) } } } Ok(()) } pub async fn run_script_standalone( script_text: String, redirect_stdin: bool, context: &EvaluationContext, exit_on_error: bool, ) -> Result<(), Box> { let line = process_script(&script_text, context, redirect_stdin, 0, false).await; match line { LineResult::Success(line) => { let error_code = { let errors = context.current_errors.clone(); let errors = errors.lock(); if errors.len() > 0 { 1 } else { 0 } }; context.maybe_print_errors(Text::from(line)); if error_code != 0 && exit_on_error { std::process::exit(error_code); } } LineResult::Error(line, err) => { context.with_host(|_host| { print_err(err, &Text::from(line.clone())); }); context.maybe_print_errors(Text::from(line)); if exit_on_error { std::process::exit(1); } } _ => {} } Ok(()) } #[cfg(feature = "rustyline-support")] fn default_rustyline_editor_configuration() -> Editor { #[cfg(windows)] const DEFAULT_COMPLETION_MODE: CompletionType = CompletionType::Circular; #[cfg(not(windows))] const DEFAULT_COMPLETION_MODE: CompletionType = CompletionType::List; let config = Config::builder().color_mode(ColorMode::Forced).build(); let mut rl: Editor<_> = Editor::with_config(config); // add key bindings to move over a whole word with Ctrl+ArrowLeft and Ctrl+ArrowRight rl.bind_sequence( KeyPress::ControlLeft, Cmd::Move(Movement::BackwardWord(1, Word::Vi)), ); rl.bind_sequence( KeyPress::ControlRight, Cmd::Move(Movement::ForwardWord(1, At::AfterEnd, Word::Vi)), ); // workaround for multiline-paste hang in rustyline (see https://github.com/kkawakam/rustyline/issues/202) rl.bind_sequence(KeyPress::BracketedPasteStart, rustyline::Cmd::Noop); // Let's set the defaults up front and then override them later if the user indicates // defaults taken from here https://github.com/kkawakam/rustyline/blob/2fe886c9576c1ea13ca0e5808053ad491a6fe049/src/config.rs#L150-L167 rl.set_max_history_size(100); rl.set_history_ignore_dups(true); rl.set_history_ignore_space(false); rl.set_completion_type(DEFAULT_COMPLETION_MODE); rl.set_completion_prompt_limit(100); rl.set_keyseq_timeout(-1); rl.set_edit_mode(rustyline::config::EditMode::Emacs); rl.set_auto_add_history(false); rl.set_bell_style(rustyline::config::BellStyle::default()); rl.set_color_mode(rustyline::ColorMode::Enabled); rl.set_tab_stop(8); if let Err(e) = crate::keybinding::load_keybindings(&mut rl) { println!("Error loading keybindings: {:?}", e); } rl } #[cfg(feature = "rustyline-support")] fn configure_rustyline_editor( rl: &mut Editor, config: &dyn nu_data::config::Conf, ) -> Result<(), ShellError> { #[cfg(windows)] const DEFAULT_COMPLETION_MODE: CompletionType = CompletionType::Circular; #[cfg(not(windows))] const DEFAULT_COMPLETION_MODE: CompletionType = CompletionType::List; if let Some(line_editor_vars) = config.var("line_editor") { for (idx, value) in line_editor_vars.row_entries() { match idx.as_ref() { "max_history_size" => { if let Ok(max_history_size) = value.as_u64() { rl.set_max_history_size(max_history_size as usize); } } "history_duplicates" => { // history_duplicates = match value.as_string() { // Ok(s) if s.to_lowercase() == "alwaysadd" => { // rustyline::config::HistoryDuplicates::AlwaysAdd // } // Ok(s) if s.to_lowercase() == "ignoreconsecutive" => { // rustyline::config::HistoryDuplicates::IgnoreConsecutive // } // _ => rustyline::config::HistoryDuplicates::AlwaysAdd, // }; if let Ok(history_duplicates) = value.as_bool() { rl.set_history_ignore_dups(history_duplicates); } } "history_ignore_space" => { if let Ok(history_ignore_space) = value.as_bool() { rl.set_history_ignore_space(history_ignore_space); } } "completion_type" => { let completion_type = match value.as_string() { Ok(s) if s.to_lowercase() == "circular" => { rustyline::config::CompletionType::Circular } Ok(s) if s.to_lowercase() == "list" => { rustyline::config::CompletionType::List } #[cfg(all(unix, feature = "with-fuzzy"))] Ok(s) if s.to_lowercase() == "fuzzy" => { rustyline::config::CompletionType::Fuzzy } _ => DEFAULT_COMPLETION_MODE, }; rl.set_completion_type(completion_type); } "completion_prompt_limit" => { if let Ok(completion_prompt_limit) = value.as_u64() { rl.set_completion_prompt_limit(completion_prompt_limit as usize); } } "keyseq_timeout_ms" => { if let Ok(keyseq_timeout_ms) = value.as_u64() { rl.set_keyseq_timeout(keyseq_timeout_ms as i32); } } "edit_mode" => { let edit_mode = match value.as_string() { Ok(s) if s.to_lowercase() == "vi" => rustyline::config::EditMode::Vi, Ok(s) if s.to_lowercase() == "emacs" => rustyline::config::EditMode::Emacs, _ => rustyline::config::EditMode::Emacs, }; rl.set_edit_mode(edit_mode); // Note: When edit_mode is Emacs, the keyseq_timeout_ms is set to -1 // no matter what you may have configured. This is so that key chords // can be applied without having to do them in a given timeout. So, // it essentially turns off the keyseq timeout. } "auto_add_history" => { if let Ok(auto_add_history) = value.as_bool() { rl.set_auto_add_history(auto_add_history); } } "bell_style" => { let bell_style = match value.as_string() { Ok(s) if s.to_lowercase() == "audible" => { rustyline::config::BellStyle::Audible } Ok(s) if s.to_lowercase() == "none" => rustyline::config::BellStyle::None, Ok(s) if s.to_lowercase() == "visible" => { rustyline::config::BellStyle::Visible } _ => rustyline::config::BellStyle::default(), }; rl.set_bell_style(bell_style); } "color_mode" => { let color_mode = match value.as_string() { Ok(s) if s.to_lowercase() == "enabled" => rustyline::ColorMode::Enabled, Ok(s) if s.to_lowercase() == "forced" => rustyline::ColorMode::Forced, Ok(s) if s.to_lowercase() == "disabled" => rustyline::ColorMode::Disabled, _ => rustyline::ColorMode::Enabled, }; rl.set_color_mode(color_mode); } "tab_stop" => { if let Ok(tab_stop) = value.as_u64() { rl.set_tab_stop(tab_stop as usize); } } _ => (), } } } Ok(()) } #[cfg(feature = "rustyline-support")] fn nu_line_editor_helper( context: &mut EvaluationContext, config: &dyn nu_data::config::Conf, ) -> crate::shell::Helper { let hinter = rustyline_hinter(config); crate::shell::Helper::new(context.clone(), hinter) } #[cfg(feature = "rustyline-support")] fn rustyline_hinter(config: &dyn nu_data::config::Conf) -> Option { if let Some(line_editor_vars) = config.var("line_editor") { for (idx, value) in line_editor_vars.row_entries() { if idx == "show_hints" && value.expect_string() == "false" { return None; } } } Some(rustyline::hint::HistoryHinter {}) } fn chomp_newline(s: &str) -> &str { if let Some(s) = s.strip_suffix('\n') { s } else { s } } #[derive(Debug)] pub enum LineResult { Success(String), Error(String, ShellError), Break, CtrlC, CtrlD, ClearHistory, } pub async fn parse_and_eval(line: &str, ctx: &EvaluationContext) -> Result { // FIXME: do we still need this? let line = if let Some(s) = line.strip_suffix('\n') { s } else { line }; // TODO ensure the command whose examples we're testing is actually in the pipeline ctx.scope.enter_scope(); let (classified_block, err) = nu_parser::parse(&line, 0, &ctx.scope); if let Some(err) = err { ctx.scope.exit_scope(); return Err(err.into()); } let input_stream = InputStream::empty(); let env = ctx.get_env(); ctx.scope.add_env(env); let result = run_block(&classified_block, ctx, input_stream).await; ctx.scope.exit_scope(); result?.collect_string(Tag::unknown()).await.map(|x| x.item) } /// Process the line by parsing the text to turn it into commands, classify those commands so that we understand what is being called in the pipeline, and then run this pipeline pub async fn process_script( script_text: &str, ctx: &EvaluationContext, redirect_stdin: bool, span_offset: usize, cli_mode: bool, ) -> LineResult { if script_text.trim() == "" { LineResult::Success(script_text.to_string()) } else { let line = chomp_newline(script_text); let (block, err) = nu_parser::parse(&line, span_offset, &ctx.scope); debug!("{:#?}", block); //println!("{:#?}", pipeline); if let Some(failure) = err { return LineResult::Error(line.to_string(), failure.into()); } // There's a special case to check before we process the pipeline: // If we're giving a path by itself // ...and it's not a command in the path // ...and it doesn't have any arguments // ...and we're in the CLI // ...then change to this directory if cli_mode && block.block.len() == 1 && block.block[0].pipelines.len() == 1 && block.block[0].pipelines[0].list.len() == 1 { if let ClassifiedCommand::Internal(InternalCommand { ref name, ref args, .. }) = block.block[0].pipelines[0].list[0] { let internal_name = name; let name = args .positional .as_ref() .and_then(|potionals| { potionals.get(0).map(|e| { if let Expression::Literal(Literal::String(ref s)) = e.expr { &s } else { "" } }) }) .unwrap_or(""); if internal_name == "run_external" && args .positional .as_ref() .map(|ref v| v.len() == 1) .unwrap_or(true) && args .named .as_ref() .map(NamedArguments::is_empty) .unwrap_or(true) && canonicalize(ctx.shell_manager.path(), name).is_ok() && Path::new(&name).is_dir() && !crate::commands::classified::external::did_find_command(&name) { // Here we work differently if we're in Windows because of the expected Windows behavior #[cfg(windows)] { if name.ends_with(':') { // This looks like a drive shortcut. We need to a) switch drives and b) go back to the previous directory we were viewing on that drive // But first, we need to save where we are now let current_path = ctx.shell_manager.path(); let split_path: Vec<_> = current_path.split(':').collect(); if split_path.len() > 1 { ctx.windows_drives_previous_cwd .lock() .insert(split_path[0].to_string(), current_path); } let name = name.to_uppercase(); let new_drive: Vec<_> = name.split(':').collect(); if let Some(val) = ctx.windows_drives_previous_cwd.lock().get(new_drive[0]) { ctx.shell_manager.set_path(val.to_string()); return LineResult::Success(line.to_string()); } else { ctx.shell_manager .set_path(format!("{}\\", name.to_string())); return LineResult::Success(line.to_string()); } } else { ctx.shell_manager.set_path(name.to_string()); return LineResult::Success(line.to_string()); } } #[cfg(not(windows))] { ctx.shell_manager.set_path(name.to_string()); return LineResult::Success(line.to_string()); } } } } let input_stream = if redirect_stdin { let file = futures::io::AllowStdIo::new(std::io::stdin()); let stream = FramedRead::new(file, MaybeTextCodec::default()).map(|line| { if let Ok(line) = line { let primitive = match line { StringOrBinary::String(s) => Primitive::String(s), StringOrBinary::Binary(b) => Primitive::Binary(b.into_iter().collect()), }; Ok(Value { value: UntaggedValue::Primitive(primitive), tag: Tag::unknown(), }) } else { panic!("Internal error: could not read lines of text from stdin") } }); stream.to_input_stream() } else { InputStream::empty() }; trace!("{:#?}", block); let env = ctx.get_env(); ctx.scope.add_env(env); let result = run_block(&block, ctx, input_stream).await; match result { Ok(input) => { // Running a pipeline gives us back a stream that we can then // work through. At the top level, we just want to pull on the // values to compute them. use futures::stream::TryStreamExt; let context = RunnableContext { input, shell_manager: ctx.shell_manager.clone(), host: ctx.host.clone(), ctrl_c: ctx.ctrl_c.clone(), current_errors: ctx.current_errors.clone(), scope: ctx.scope.clone(), name: Tag::unknown(), }; if let Ok(mut output_stream) = crate::commands::autoview::command::autoview(context).await { loop { match output_stream.try_next().await { Ok(Some(ReturnSuccess::Value(Value { value: UntaggedValue::Error(e), .. }))) => return LineResult::Error(line.to_string(), e), Ok(Some(_item)) => { if ctx.ctrl_c.load(Ordering::SeqCst) { break; } } Ok(None) => break, Err(e) => return LineResult::Error(line.to_string(), e), } } } LineResult::Success(line.to_string()) } Err(err) => LineResult::Error(line.to_string(), err), } } } pub fn print_err(err: ShellError, source: &Text) { if let Some(diag) = err.into_diagnostic() { let source = source.to_string(); let mut files = codespan_reporting::files::SimpleFiles::new(); files.add("shell", source); let writer = codespan_reporting::term::termcolor::StandardStream::stderr( codespan_reporting::term::termcolor::ColorChoice::Always, ); let config = codespan_reporting::term::Config::default(); let _ = std::panic::catch_unwind(move || { let _ = codespan_reporting::term::emit(&mut writer.lock(), &config, &files, &diag); }); } } #[cfg(test)] mod tests { #[quickcheck] fn quickcheck_parse(data: String) -> bool { let (tokens, err) = nu_parser::lex(&data, 0); let (lite_block, err2) = nu_parser::group(tokens); if err.is_none() && err2.is_none() { let context = crate::evaluation_context::EvaluationContext::basic().unwrap(); let _ = nu_parser::classify_block(&lite_block, &context.scope); } true } }