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::{ops::Try, 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) } 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), Enter(Value), Exit, } #[derive(Debug, Serialize, Deserialize)] pub enum ReturnSuccess { Value(Value), Action(CommandAction), } pub type ReturnValue = Result; impl From for ReturnValue { fn from(input: Value) -> ReturnValue { Ok(ReturnSuccess::Value(input)) } } impl ReturnSuccess { pub fn change_cwd(path: PathBuf) -> ReturnValue { Ok(ReturnSuccess::Action(CommandAction::ChangePath(path))) } pub fn value(input: Value) -> ReturnValue { Ok(ReturnSuccess::Value(input)) } } 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, can_load: vec![], can_save: vec![], } } } 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, can_load: vec![], can_save: vec![], } } } 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, }) }