use crate::context::{SourceMap, SpanSource}; use crate::errors::ShellError; use crate::evaluate::Scope; use crate::object::Value; use crate::parser::hir; use crate::parser::{registry, ConfigDeserializer}; use crate::prelude::*; use derive_new::new; use getset::Getters; use serde::{Deserialize, Serialize}; use std::fmt; use std::ops::Deref; use std::path::PathBuf; use uuid::Uuid; #[derive(Deserialize, Serialize, Debug, Clone)] pub struct UnevaluatedCallInfo { pub args: hir::Call, pub source: Text, pub source_map: SourceMap, pub name_span: Span, } impl ToDebug for UnevaluatedCallInfo { fn fmt_debug(&self, f: &mut fmt::Formatter, source: &str) -> fmt::Result { self.args.fmt_debug(f, source) } } impl UnevaluatedCallInfo { pub fn evaluate( self, registry: ®istry::CommandRegistry, scope: &Scope, ) -> Result { let args = self.args.evaluate(registry, scope, &self.source)?; Ok(CallInfo { args, source_map: self.source_map, name_span: self.name_span, }) } pub fn has_it_or_block(&self) -> bool { use hir::RawExpression; use hir::Variable; if let Some(positional) = &self.args.positional() { for pos in positional { match pos { Tagged { item: RawExpression::Variable(Variable::It(_)), .. } => { return true; } Tagged { item: RawExpression::Block(_), .. } => { return true; } _ => {} } } } false } } #[derive(Deserialize, Serialize, Debug, Clone)] pub struct CallInfo { pub args: registry::EvaluatedArgs, pub source_map: SourceMap, pub name_span: Span, } #[derive(Getters)] #[get = "crate"] pub struct CommandArgs { pub host: Arc>, pub shell_manager: ShellManager, pub call_info: UnevaluatedCallInfo, // pub host: Arc>, // pub shell_manager: ShellManager, // pub call_info: CallInfo, pub input: InputStream, } #[derive(Getters, Clone)] #[get = "crate"] pub struct RawCommandArgs { pub host: Arc>, pub shell_manager: ShellManager, pub call_info: UnevaluatedCallInfo, } impl RawCommandArgs { pub fn with_input(self, input: Vec>) -> CommandArgs { CommandArgs { host: self.host, shell_manager: self.shell_manager, call_info: self.call_info, input: input.into(), } } } impl ToDebug for CommandArgs { fn fmt_debug(&self, f: &mut fmt::Formatter, source: &str) -> fmt::Result { self.call_info.fmt_debug(f, source) } } impl CommandArgs { pub fn evaluate_once( self, registry: ®istry::CommandRegistry, ) -> Result { let host = self.host.clone(); let shell_manager = self.shell_manager.clone(); let input = self.input; let call_info = self.call_info.evaluate(registry, &Scope::empty())?; Ok(EvaluatedWholeStreamCommandArgs::new( host, shell_manager, call_info, input, )) } pub fn name_span(&self) -> Span { self.call_info.name_span } pub fn process<'de, T: Deserialize<'de>>( self, registry: &CommandRegistry, callback: fn(T, RunnableContext) -> Result, ) -> Result, ShellError> { let shell_manager = self.shell_manager.clone(); let source_map = self.call_info.source_map.clone(); let host = self.host.clone(); let args = self.evaluate_once(registry)?; let (input, args) = args.split(); let name_span = args.call_info.name_span; let mut deserializer = ConfigDeserializer::from_call_node(args); Ok(RunnableArgs { args: T::deserialize(&mut deserializer)?, context: RunnableContext { input: input, commands: registry.clone(), shell_manager, name: name_span, source_map, host, }, callback, }) } pub fn process_raw<'de, T: Deserialize<'de>>( self, registry: &CommandRegistry, callback: fn(T, RunnableContext, RawCommandArgs) -> Result, ) -> Result, ShellError> { let raw_args = RawCommandArgs { host: self.host.clone(), shell_manager: self.shell_manager.clone(), call_info: self.call_info.clone(), }; let shell_manager = self.shell_manager.clone(); let source_map = self.call_info.source_map.clone(); let host = self.host.clone(); let args = self.evaluate_once(registry)?; let (input, args) = args.split(); let name_span = args.call_info.name_span; let mut deserializer = ConfigDeserializer::from_call_node(args); Ok(RunnableRawArgs { args: T::deserialize(&mut deserializer)?, context: RunnableContext { input: input, commands: registry.clone(), shell_manager, name: name_span, source_map, host, }, raw_args, callback, }) } } pub struct RunnableContext { pub input: InputStream, pub shell_manager: ShellManager, pub host: Arc>, pub commands: CommandRegistry, pub source_map: SourceMap, pub name: Span, } impl RunnableContext { pub fn cwd(&self) -> PathBuf { PathBuf::from(self.shell_manager.path()) } pub fn expect_command(&self, name: &str) -> Arc { self.commands .get_command(name) .expect(&format!("Expected command {}", name)) } } pub struct RunnableArgs { args: T, context: RunnableContext, callback: fn(T, RunnableContext) -> Result, } impl RunnableArgs { pub fn run(self) -> Result { (self.callback)(self.args, self.context) } } pub struct RunnableRawArgs { args: T, raw_args: RawCommandArgs, context: RunnableContext, callback: fn(T, RunnableContext, RawCommandArgs) -> Result, } impl RunnableRawArgs { pub fn run(self) -> Result { (self.callback)(self.args, self.context, self.raw_args) } } pub struct EvaluatedWholeStreamCommandArgs { pub args: EvaluatedCommandArgs, pub input: InputStream, } impl Deref for EvaluatedWholeStreamCommandArgs { type Target = EvaluatedCommandArgs; fn deref(&self) -> &Self::Target { &self.args } } impl EvaluatedWholeStreamCommandArgs { pub fn new( host: Arc>, shell_manager: ShellManager, call_info: CallInfo, input: impl Into, ) -> EvaluatedWholeStreamCommandArgs { EvaluatedWholeStreamCommandArgs { args: EvaluatedCommandArgs { host, shell_manager, call_info, }, input: input.into(), } } pub fn name_span(&self) -> Span { self.args.call_info.name_span } pub fn parts(self) -> (InputStream, registry::EvaluatedArgs) { let EvaluatedWholeStreamCommandArgs { args, input } = self; (input, args.call_info.args) } pub fn split(self) -> (InputStream, EvaluatedCommandArgs) { let EvaluatedWholeStreamCommandArgs { args, input } = self; (input, args) } } #[derive(Getters)] #[get = "pub"] pub struct EvaluatedFilterCommandArgs { args: EvaluatedCommandArgs, #[allow(unused)] input: Tagged, } impl Deref for EvaluatedFilterCommandArgs { type Target = EvaluatedCommandArgs; fn deref(&self) -> &Self::Target { &self.args } } impl EvaluatedFilterCommandArgs { pub fn new( host: Arc>, shell_manager: ShellManager, call_info: CallInfo, input: Tagged, ) -> EvaluatedFilterCommandArgs { EvaluatedFilterCommandArgs { args: EvaluatedCommandArgs { host, shell_manager, call_info, }, input, } } } #[derive(Getters, new)] #[get = "crate"] pub struct EvaluatedCommandArgs { pub host: Arc>, pub shell_manager: ShellManager, pub call_info: CallInfo, } impl EvaluatedCommandArgs { pub fn call_args(&self) -> ®istry::EvaluatedArgs { &self.call_info.args } pub fn nth(&self, pos: usize) -> Option<&Tagged> { self.call_info.args.nth(pos) } pub fn expect_nth(&self, pos: usize) -> Result<&Tagged, ShellError> { self.call_info.args.expect_nth(pos) } pub fn len(&self) -> usize { self.call_info.args.len() } pub fn get(&self, name: &str) -> Option<&Tagged> { self.call_info.args.get(name) } pub fn slice_from(&self, from: usize) -> Vec> { let positional = &self.call_info.args.positional; match positional { None => vec![], Some(list) => list[from..].to_vec(), } } #[allow(unused)] pub fn has(&self, name: &str) -> bool { self.call_info.args.has(name) } } #[derive(Debug, Serialize, Deserialize)] pub enum CommandAction { ChangePath(String), AddSpanSource(Uuid, SpanSource), Exit, EnterShell(String), EnterValueShell(Tagged), PreviousShell, NextShell, LeaveShell, } #[derive(Debug, Serialize, Deserialize)] pub enum ReturnSuccess { Value(Tagged), Action(CommandAction), } pub type ReturnValue = Result; impl From> for ReturnValue { fn from(input: Tagged) -> ReturnValue { Ok(ReturnSuccess::Value(input)) } } impl ReturnSuccess { pub fn change_cwd(path: String) -> ReturnValue { Ok(ReturnSuccess::Action(CommandAction::ChangePath(path))) } pub fn value(input: impl Into>) -> ReturnValue { Ok(ReturnSuccess::Value(input.into())) } pub fn action(input: CommandAction) -> ReturnValue { Ok(ReturnSuccess::Action(input)) } pub fn spanned_value(input: Value, span: Span) -> ReturnValue { Ok(ReturnSuccess::Value(Tagged::from_simple_spanned_item( input, span, ))) } } pub trait WholeStreamCommand: Send + Sync { fn name(&self) -> &str; fn run( &self, args: CommandArgs, registry: ®istry::CommandRegistry, ) -> Result; fn signature(&self) -> Signature { Signature { name: self.name().to_string(), positional: vec![], rest_positional: true, named: indexmap::IndexMap::new(), is_filter: true, } } } pub trait PerItemCommand: Send + Sync { fn name(&self) -> &str; fn run( &self, call_info: &CallInfo, registry: &CommandRegistry, shell_manager: &ShellManager, input: Tagged, ) -> Result, ShellError>; fn signature(&self) -> Signature { Signature { name: self.name().to_string(), positional: vec![], rest_positional: true, named: indexmap::IndexMap::new(), is_filter: true, } } } pub enum Command { WholeStream(Arc), PerItem(Arc), } impl Command { pub fn name(&self) -> &str { match self { Command::WholeStream(command) => command.name(), Command::PerItem(command) => command.name(), } } pub fn signature(&self) -> Signature { match self { Command::WholeStream(command) => command.signature(), Command::PerItem(command) => command.signature(), } } pub async fn run( &self, args: CommandArgs, registry: ®istry::CommandRegistry, ) -> Result { match self { Command::WholeStream(command) => command.run(args, registry), Command::PerItem(command) => self.run_helper(command.clone(), args, registry.clone()), } } fn run_helper( &self, command: Arc, args: CommandArgs, registry: CommandRegistry, ) -> Result { let raw_args = RawCommandArgs { host: args.host, shell_manager: args.shell_manager, call_info: args.call_info, }; if raw_args.call_info.has_it_or_block() { let out = args .input .values .map(move |x| { let call_info = raw_args .clone() .call_info .evaluate(®istry, &Scope::it_value(x.clone())) .unwrap(); match command.run(&call_info, ®istry, &raw_args.shell_manager, x) { Ok(o) => o, Err(e) => VecDeque::from(vec![ReturnValue::Err(e)]), } }) .flatten(); Ok(out.to_output_stream()) } else { let nothing = Value::nothing().tagged(Tag::unknown()); let call_info = raw_args .clone() .call_info .evaluate(®istry, &Scope::it_value(nothing.clone())) .unwrap(); // We don't have an $it or block, so just execute what we have let out = match command.run(&call_info, ®istry, &raw_args.shell_manager, nothing) { Ok(o) => o, Err(e) => VecDeque::from(vec![ReturnValue::Err(e)]), }; Ok(out.to_output_stream()) } } } #[allow(unused)] pub struct FnFilterCommand { name: String, func: fn(EvaluatedFilterCommandArgs) -> Result, } impl WholeStreamCommand for FnFilterCommand { fn name(&self) -> &str { &self.name } fn run( &self, args: CommandArgs, registry: ®istry::CommandRegistry, ) -> Result { let CommandArgs { host, shell_manager, call_info, input, } = args; let host: Arc> = host.clone(); let shell_manager = shell_manager.clone(); let registry: registry::CommandRegistry = registry.clone(); let func = self.func; let result = input.values.map(move |it| { let registry = registry.clone(); let call_info = match call_info .clone() .evaluate(®istry, &Scope::it_value(it.clone())) { Err(err) => return OutputStream::from(vec![Err(err)]).values, Ok(args) => args, }; let args = EvaluatedFilterCommandArgs::new(host.clone(), shell_manager.clone(), call_info, it); match func(args) { Err(err) => return OutputStream::from(vec![Err(err)]).values, Ok(stream) => stream.values, } }); let result = result.flatten(); let result: BoxStream = result.boxed(); Ok(result.into()) } } pub struct FnRawCommand { name: String, func: Box< dyn Fn(CommandArgs, ®istry::CommandRegistry) -> Result + Send + Sync, >, } impl WholeStreamCommand for FnRawCommand { fn name(&self) -> &str { &self.name } fn run( &self, args: CommandArgs, registry: ®istry::CommandRegistry, ) -> Result { (self.func)(args, registry) } } pub fn command( name: &str, func: Box< dyn Fn(CommandArgs, ®istry::CommandRegistry) -> Result + Send + Sync, >, ) -> Arc { Arc::new(Command::WholeStream(Arc::new(FnRawCommand { name: name.to_string(), func, }))) } pub fn whole_stream_command(command: impl WholeStreamCommand + 'static) -> Arc { Arc::new(Command::WholeStream(Arc::new(command))) } pub fn per_item_command(command: impl PerItemCommand + 'static) -> Arc { Arc::new(Command::PerItem(Arc::new(command))) } #[allow(unused)] pub fn filter( name: &str, func: fn(EvaluatedFilterCommandArgs) -> Result, ) -> Arc { Arc::new(Command::WholeStream(Arc::new(FnFilterCommand { name: name.to_string(), func, }))) }