use crate::commands::{command::CommandArgs, Command, UnevaluatedCallInfo}; use crate::env::host::Host; use crate::shell::shell_manager::ShellManager; use crate::stream::{InputStream, OutputStream}; use indexmap::IndexMap; use nu_errors::ShellError; use nu_parser::{hir, hir::syntax_shape::ExpandContext, hir::syntax_shape::SignatureRegistry}; use nu_protocol::{errln, Signature}; use nu_source::{Tag, Text}; use std::error::Error; use std::sync::atomic::AtomicBool; use std::sync::{Arc, Mutex}; #[derive(Clone)] pub struct CommandRegistry { registry: Arc>>>, } impl SignatureRegistry for CommandRegistry { fn has(&self, name: &str) -> Result { if let Ok(registry) = self.registry.lock() { Ok(registry.contains_key(name)) } else { Err(ShellError::untagged_runtime_error(format!( "Could not load from registry: {}", name ))) } } fn get(&self, name: &str) -> Result, ShellError> { if let Ok(registry) = self.registry.lock() { Ok(registry.get(name).map(|command| command.signature())) } else { Err(ShellError::untagged_runtime_error(format!( "Could not get from registry: {}", name ))) } } } impl CommandRegistry { pub fn new() -> CommandRegistry { CommandRegistry { registry: Arc::new(Mutex::new(IndexMap::default())), } } } impl CommandRegistry { pub(crate) fn empty() -> CommandRegistry { CommandRegistry { registry: Arc::new(Mutex::new(IndexMap::default())), } } pub(crate) fn get_command(&self, name: &str) -> Option> { let registry = self.registry.lock().unwrap(); registry.get(name).cloned() } pub(crate) fn expect_command(&self, name: &str) -> Result, ShellError> { self.get_command(name).ok_or_else(|| { ShellError::untagged_runtime_error(format!("Could not load command: {}", name)) }) } pub(crate) fn has(&self, name: &str) -> bool { let registry = self.registry.lock().unwrap(); registry.contains_key(name) } pub(crate) fn insert(&mut self, name: impl Into, command: Arc) { let mut registry = self.registry.lock().unwrap(); registry.insert(name.into(), command); } pub(crate) fn names(&self) -> Vec { let registry = self.registry.lock().unwrap(); registry.keys().cloned().collect() } } #[derive(Clone)] pub struct Context { pub registry: CommandRegistry, pub host: Arc>>, pub current_errors: Arc>>, pub ctrl_c: Arc, pub(crate) shell_manager: ShellManager, } impl Context { pub(crate) fn registry(&self) -> &CommandRegistry { &self.registry } pub(crate) fn expand_context<'context>( &'context self, source: &'context Text, ) -> ExpandContext<'context> { ExpandContext::new( Box::new(self.registry.clone()), source, self.shell_manager.homedir(), ) } pub(crate) fn basic() -> Result> { let registry = CommandRegistry::new(); Ok(Context { registry: registry.clone(), host: Arc::new(Mutex::new(Box::new(crate::env::host::BasicHost))), current_errors: Arc::new(Mutex::new(vec![])), ctrl_c: Arc::new(AtomicBool::new(false)), shell_manager: ShellManager::basic(registry)?, }) } pub(crate) fn error(&mut self, error: ShellError) { self.with_errors(|errors| errors.push(error)) } pub(crate) fn maybe_print_errors(&mut self, source: Text) -> bool { let errors = self.current_errors.clone(); let errors = errors.lock(); let host = self.host.clone(); let host = host.lock(); let result: bool; match (errors, host) { (Err(err), _) => { errln!( "Unexpected error attempting to acquire the lock of the current errors: {:?}", err ); result = false; } (_, Err(err)) => { errln!( "Unexpected error attempting to acquire the lock of the current errors: {:?}", err ); result = false; } (Ok(mut errors), Ok(host)) => { if errors.len() > 0 { let error = errors[0].clone(); *errors = vec![]; crate::cli::print_err(error, &*host, &source); result = true; } else { result = false; } } }; result } pub(crate) fn with_host(&mut self, block: impl FnOnce(&mut dyn Host) -> T) -> T { let mut host = self.host.lock().unwrap(); block(&mut *host) } pub(crate) fn with_errors(&mut self, block: impl FnOnce(&mut Vec) -> T) -> T { let mut errors = self.current_errors.lock().unwrap(); block(&mut *errors) } pub fn add_commands(&mut self, commands: Vec>) { for command in commands { self.registry.insert(command.name().to_string(), command); } } pub(crate) fn get_command(&self, name: &str) -> Option> { self.registry.get_command(name) } pub(crate) fn expect_command(&self, name: &str) -> Result, ShellError> { self.registry.expect_command(name) } pub(crate) fn run_command( &mut self, command: Arc, name_tag: Tag, args: hir::Call, source: &Text, input: InputStream, ) -> OutputStream { let command_args = self.command_args(args, input, source, name_tag); command.run(command_args, self.registry()) } fn call_info(&self, args: hir::Call, source: &Text, name_tag: Tag) -> UnevaluatedCallInfo { UnevaluatedCallInfo { args, source: source.clone(), name_tag, } } fn command_args( &self, args: hir::Call, input: InputStream, source: &Text, name_tag: Tag, ) -> CommandArgs { CommandArgs { host: self.host.clone(), ctrl_c: self.ctrl_c.clone(), shell_manager: self.shell_manager.clone(), call_info: self.call_info(args, source, name_tag), input, } } }