use crate::errors::ShellError; use crate::object::Value; use crate::parser::{ registry::{self, Args}, Span, Spanned, }; use crate::prelude::*; use getset::Getters; use serde::{Deserialize, Serialize}; use std::path::PathBuf; #[derive(Getters)] #[get = "crate"] pub struct CommandArgs { pub host: Arc>, pub env: Arc>, pub name_span: Option, pub args: Args, pub input: InputStream, } impl CommandArgs { pub fn nth(&self, pos: usize) -> Option<&Spanned> { self.args.nth(pos) } pub fn positional_iter(&self) -> impl Iterator> { self.args.positional_iter() } pub fn expect_nth(&self, pos: usize) -> Result<&Spanned, ShellError> { self.args.expect_nth(pos) } pub fn len(&self) -> usize { self.args.len() } pub fn get(&self, name: &str) -> Option<&Spanned> { self.args.get(name) } #[allow(unused)] pub fn has(&self, name: &str) -> bool { self.args.has(name) } } pub struct SinkCommandArgs { pub ctx: Context, pub name_span: Option, pub args: Args, pub input: Vec>, } #[derive(Debug, Serialize, Deserialize)] pub enum CommandAction { ChangePath(PathBuf), Exit, } #[derive(Debug, Serialize, Deserialize)] pub enum ReturnSuccess { Value(Spanned), Action(CommandAction), } pub type ReturnValue = Result; impl From> for ReturnValue { fn from(input: Spanned) -> ReturnValue { Ok(ReturnSuccess::Value(input)) } } impl ReturnSuccess { pub fn change_cwd(path: PathBuf) -> ReturnValue { Ok(ReturnSuccess::Action(CommandAction::ChangePath(path))) } pub fn value(input: impl Into>) -> ReturnValue { Ok(ReturnSuccess::Value(input.into())) } pub fn spanned_value(input: Value, span: Span) -> ReturnValue { Ok(ReturnSuccess::Value(Spanned::from_item(input, span))) } } pub trait Command { fn run(&self, args: CommandArgs) -> Result; fn name(&self) -> &str; fn config(&self) -> registry::CommandConfig { registry::CommandConfig { name: self.name().to_string(), positional: vec![], rest_positional: true, named: indexmap::IndexMap::new(), is_filter: true, is_sink: false, } } } pub trait Sink { fn run(&self, args: SinkCommandArgs) -> Result<(), ShellError>; fn name(&self) -> &str; fn config(&self) -> registry::CommandConfig { registry::CommandConfig { name: self.name().to_string(), positional: vec![], rest_positional: true, named: indexmap::IndexMap::new(), is_filter: false, is_sink: true, } } } pub struct FnCommand { name: String, func: Box Result>, } impl Command for FnCommand { fn run(&self, args: CommandArgs) -> Result { (self.func)(args) } fn name(&self) -> &str { &self.name } } pub fn command( name: &str, func: Box Result>, ) -> Arc { Arc::new(FnCommand { name: name.to_string(), func, }) } pub struct FnSink { name: String, func: Box Result<(), ShellError>>, } impl Sink for FnSink { fn run(&self, args: SinkCommandArgs) -> Result<(), ShellError> { (self.func)(args) } fn name(&self) -> &str { &self.name } } pub fn sink( name: &str, func: Box Result<(), ShellError>>, ) -> Arc { Arc::new(FnSink { name: name.to_string(), func, }) }