nushell/src/commands/command.rs

524 lines
13 KiB
Rust
Raw Normal View History

2019-07-23 22:22:11 +00:00
use crate::context::{SourceMap, SpanSource};
2019-05-10 16:59:12 +00:00
use crate::errors::ShellError;
2019-07-23 22:22:11 +00:00
use crate::evaluate::Scope;
2019-05-10 16:59:12 +00:00
use crate::object::Value;
2019-07-23 22:22:11 +00:00
use crate::parser::hir;
2019-08-02 19:15:07 +00:00
use crate::parser::{registry, ConfigDeserializer, Span, Spanned};
use crate::prelude::*;
2019-08-02 19:15:07 +00:00
use derive_new::new;
2019-06-22 03:43:37 +00:00
use getset::Getters;
2019-06-27 04:56:48 +00:00
use serde::{Deserialize, Serialize};
2019-07-23 22:22:11 +00:00
use std::fmt;
use std::ops::Deref;
2019-07-08 16:44:53 +00:00
use std::path::PathBuf;
use uuid::Uuid;
2019-05-10 16:59:12 +00:00
2019-07-23 22:22:11 +00:00
#[derive(Deserialize, Serialize, Debug, Clone)]
pub struct UnevaluatedCallInfo {
pub args: hir::Call,
pub source: Text,
pub source_map: SourceMap,
pub name_span: Option<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 {
fn evaluate(
self,
registry: &registry::CommandRegistry,
scope: &Scope,
) -> Result<CallInfo, ShellError> {
let args = self.args.evaluate(registry, scope, &self.source)?;
Ok(CallInfo {
args,
source_map: self.source_map,
name_span: self.name_span,
})
}
}
#[derive(Deserialize, Serialize, Debug)]
pub struct CallInfo {
2019-07-23 22:22:11 +00:00
pub args: registry::EvaluatedArgs,
pub source_map: SourceMap,
pub name_span: Option<Span>,
}
2019-06-22 03:43:37 +00:00
#[derive(Getters)]
#[get = "crate"]
pub struct CommandArgs {
2019-07-23 22:22:11 +00:00
pub host: Arc<Mutex<dyn Host>>,
2019-07-16 19:10:25 +00:00
pub env: Arc<Mutex<Environment>>,
2019-07-23 22:22:11 +00:00
pub call_info: UnevaluatedCallInfo,
pub input: InputStream,
2019-05-16 00:21:46 +00:00
}
2019-08-03 02:17:28 +00:00
#[derive(Getters)]
#[get = "crate"]
pub struct RawCommandArgs {
pub host: Arc<Mutex<dyn Host>>,
pub env: Arc<Mutex<Environment>>,
pub call_info: UnevaluatedCallInfo,
}
impl RawCommandArgs {
pub fn with_input(self, input: Vec<Spanned<Value>>) -> CommandArgs {
CommandArgs {
host: self.host,
env: self.env,
call_info: self.call_info,
input: input.into(),
}
}
}
2019-07-23 22:22:11 +00:00
impl ToDebug for CommandArgs {
fn fmt_debug(&self, f: &mut fmt::Formatter, source: &str) -> fmt::Result {
self.call_info.fmt_debug(f, source)
}
}
2019-06-22 03:43:37 +00:00
impl CommandArgs {
2019-07-23 22:22:11 +00:00
pub fn evaluate_once(
self,
registry: &registry::CommandRegistry,
) -> Result<EvaluatedStaticCommandArgs, ShellError> {
let host = self.host.clone();
let env = self.env.clone();
let input = self.input;
let call_info = self.call_info.evaluate(registry, &Scope::empty())?;
Ok(EvaluatedStaticCommandArgs::new(host, env, call_info, input))
}
pub fn name_span(&self) -> Option<Span> {
self.call_info.name_span
}
2019-08-02 19:15:07 +00:00
pub fn process<'de, T: Deserialize<'de>>(
self,
registry: &CommandRegistry,
callback: fn(T, RunnableContext) -> Result<OutputStream, ShellError>,
) -> Result<RunnableArgs<T>, ShellError> {
let env = self.env.clone();
2019-08-03 02:17:28 +00:00
let host = self.host.clone();
2019-08-02 19:15:07 +00:00
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,
2019-08-03 02:17:28 +00:00
commands: registry.clone(),
env,
name: name_span,
host,
},
callback,
})
}
pub fn process_raw<'de, T: Deserialize<'de>>(
self,
registry: &CommandRegistry,
callback: fn(T, RunnableContext, RawCommandArgs) -> Result<OutputStream, ShellError>,
) -> Result<RunnableRawArgs<T>, ShellError> {
let raw_args = RawCommandArgs {
host: self.host.clone(),
env: self.env.clone(),
call_info: self.call_info.clone(),
};
let env = self.env.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(),
2019-08-02 19:15:07 +00:00
env,
name: name_span,
2019-08-03 02:17:28 +00:00
host,
2019-08-02 19:15:07 +00:00
},
2019-08-03 02:17:28 +00:00
raw_args,
2019-08-02 19:15:07 +00:00
callback,
})
}
}
pub struct RunnableContext {
pub input: InputStream,
pub env: Arc<Mutex<Environment>>,
2019-08-03 02:17:28 +00:00
pub host: Arc<Mutex<dyn Host>>,
pub commands: CommandRegistry,
2019-08-02 19:15:07 +00:00
pub name: Option<Span>,
}
impl RunnableContext {
pub fn cwd(&self) -> PathBuf {
let env = self.env.clone();
let env = env.lock().unwrap();
env.path.clone()
}
2019-08-03 02:17:28 +00:00
pub fn expect_command(&self, name: &str) -> Arc<Command> {
self.commands
.get_command(name)
.expect(&format!("Expected command {}", name))
}
2019-08-02 19:15:07 +00:00
}
pub struct RunnableArgs<T> {
args: T,
context: RunnableContext,
callback: fn(T, RunnableContext) -> Result<OutputStream, ShellError>,
}
impl<T> RunnableArgs<T> {
pub fn run(self) -> Result<OutputStream, ShellError> {
(self.callback)(self.args, self.context)
}
2019-07-23 22:22:11 +00:00
}
2019-08-03 02:17:28 +00:00
pub struct RunnableRawArgs<T> {
args: T,
raw_args: RawCommandArgs,
context: RunnableContext,
callback: fn(T, RunnableContext, RawCommandArgs) -> Result<OutputStream, ShellError>,
}
impl<T> RunnableRawArgs<T> {
pub fn run(self) -> Result<OutputStream, ShellError> {
(self.callback)(self.args, self.context, self.raw_args)
}
}
2019-07-23 22:22:11 +00:00
pub struct EvaluatedStaticCommandArgs {
pub args: EvaluatedCommandArgs,
pub input: InputStream,
}
impl Deref for EvaluatedStaticCommandArgs {
type Target = EvaluatedCommandArgs;
fn deref(&self) -> &Self::Target {
&self.args
}
}
impl EvaluatedStaticCommandArgs {
pub fn new(
host: Arc<Mutex<dyn Host>>,
env: Arc<Mutex<Environment>>,
call_info: CallInfo,
input: impl Into<InputStream>,
) -> EvaluatedStaticCommandArgs {
EvaluatedStaticCommandArgs {
args: EvaluatedCommandArgs {
host,
env,
call_info,
},
input: input.into(),
}
}
pub fn name_span(&self) -> Option<Span> {
self.args.call_info.name_span
}
pub fn parts(self) -> (InputStream, registry::EvaluatedArgs) {
let EvaluatedStaticCommandArgs { args, input } = self;
(input, args.call_info.args)
}
2019-08-02 19:15:07 +00:00
pub fn split(self) -> (InputStream, EvaluatedCommandArgs) {
let EvaluatedStaticCommandArgs { args, input } = self;
(input, args)
}
2019-07-23 22:22:11 +00:00
}
#[derive(Getters)]
#[get = "pub"]
pub struct EvaluatedFilterCommandArgs {
args: EvaluatedCommandArgs,
2019-07-24 04:10:48 +00:00
#[allow(unused)]
2019-07-23 22:22:11 +00:00
input: Spanned<Value>,
}
impl Deref for EvaluatedFilterCommandArgs {
type Target = EvaluatedCommandArgs;
fn deref(&self) -> &Self::Target {
&self.args
}
}
impl EvaluatedFilterCommandArgs {
pub fn new(
host: Arc<Mutex<dyn Host>>,
env: Arc<Mutex<Environment>>,
call_info: CallInfo,
input: Spanned<Value>,
) -> EvaluatedFilterCommandArgs {
EvaluatedFilterCommandArgs {
args: EvaluatedCommandArgs {
host,
env,
call_info,
},
input,
}
}
}
2019-08-02 19:15:07 +00:00
#[derive(Getters, new)]
2019-07-23 22:22:11 +00:00
#[get = "crate"]
pub struct EvaluatedCommandArgs {
pub host: Arc<Mutex<dyn Host>>,
pub env: Arc<Mutex<Environment>>,
pub call_info: CallInfo,
}
impl EvaluatedCommandArgs {
pub fn call_args(&self) -> &registry::EvaluatedArgs {
&self.call_info.args
}
2019-06-22 03:43:37 +00:00
pub fn nth(&self, pos: usize) -> Option<&Spanned<Value>> {
self.call_info.args.nth(pos)
2019-06-22 03:43:37 +00:00
}
pub fn expect_nth(&self, pos: usize) -> Result<&Spanned<Value>, ShellError> {
self.call_info.args.expect_nth(pos)
2019-06-22 03:43:37 +00:00
}
pub fn len(&self) -> usize {
self.call_info.args.len()
2019-06-22 03:43:37 +00:00
}
pub fn get(&self, name: &str) -> Option<&Spanned<Value>> {
self.call_info.args.get(name)
2019-06-22 03:43:37 +00:00
}
2019-08-02 19:15:07 +00:00
pub fn slice_from(&self, from: usize) -> Vec<Spanned<Value>> {
let positional = &self.call_info.args.positional;
match positional {
None => vec![],
Some(list) => list[from..].to_vec(),
}
}
2019-07-16 19:10:25 +00:00
#[allow(unused)]
2019-06-22 03:43:37 +00:00
pub fn has(&self, name: &str) -> bool {
self.call_info.args.has(name)
2019-06-22 03:43:37 +00:00
}
}
2019-06-27 04:56:48 +00:00
#[derive(Debug, Serialize, Deserialize)]
pub enum CommandAction {
2019-06-13 21:47:25 +00:00
ChangePath(PathBuf),
AddSpanSource(Uuid, SpanSource),
2019-06-13 21:47:25 +00:00
Exit,
2019-05-11 08:08:21 +00:00
}
2019-06-27 04:56:48 +00:00
#[derive(Debug, Serialize, Deserialize)]
pub enum ReturnSuccess {
2019-07-08 16:44:53 +00:00
Value(Spanned<Value>),
Action(CommandAction),
2019-05-11 08:08:21 +00:00
}
pub type ReturnValue = Result<ReturnSuccess, ShellError>;
2019-07-08 16:44:53 +00:00
impl From<Spanned<Value>> for ReturnValue {
fn from(input: Spanned<Value>) -> ReturnValue {
Ok(ReturnSuccess::Value(input))
}
}
impl ReturnSuccess {
pub fn change_cwd(path: PathBuf) -> ReturnValue {
Ok(ReturnSuccess::Action(CommandAction::ChangePath(path)))
}
2019-07-08 16:44:53 +00:00
pub fn value(input: impl Into<Spanned<Value>>) -> ReturnValue {
Ok(ReturnSuccess::Value(input.into()))
}
pub fn action(input: CommandAction) -> ReturnValue {
Ok(ReturnSuccess::Action(input))
}
2019-07-08 16:44:53 +00:00
pub fn spanned_value(input: Value, span: Span) -> ReturnValue {
Ok(ReturnSuccess::Value(Spanned::from_item(input, span)))
2019-05-11 08:08:21 +00:00
}
2019-05-10 16:59:12 +00:00
}
2019-08-02 19:15:07 +00:00
pub trait StaticCommand: Send + Sync {
fn name(&self) -> &str;
2019-07-23 22:22:11 +00:00
fn run(
&self,
args: CommandArgs,
registry: &registry::CommandRegistry,
) -> Result<OutputStream, ShellError>;
2019-05-28 06:45:18 +00:00
2019-08-02 19:15:07 +00:00
fn signature(&self) -> Signature {
Signature {
2019-05-28 06:45:18 +00:00
name: self.name().to_string(),
positional: vec![],
2019-05-28 06:45:18 +00:00
rest_positional: true,
named: indexmap::IndexMap::new(),
2019-07-02 07:56:20 +00:00
is_filter: true,
2019-05-28 06:45:18 +00:00
}
}
}
2019-08-02 19:15:07 +00:00
pub enum Command {
Static(Arc<dyn StaticCommand>),
}
2019-06-07 06:34:42 +00:00
2019-08-02 19:15:07 +00:00
impl Command {
pub fn name(&self) -> &str {
match self {
Command::Static(command) => command.name(),
}
}
pub fn signature(&self) -> Signature {
match self {
Command::Static(command) => command.signature(),
}
}
pub async fn run(
&self,
args: CommandArgs,
registry: &registry::CommandRegistry,
) -> Result<OutputStream, ShellError> {
match self {
Command::Static(command) => command.run(args, registry),
2019-06-07 06:34:42 +00:00
}
}
}
2019-07-24 04:10:48 +00:00
#[allow(unused)]
2019-07-23 22:22:11 +00:00
pub struct FnFilterCommand {
2019-05-28 06:45:18 +00:00
name: String,
2019-07-23 22:22:11 +00:00
func: fn(EvaluatedFilterCommandArgs) -> Result<OutputStream, ShellError>,
}
2019-05-22 07:12:03 +00:00
2019-08-02 19:15:07 +00:00
impl StaticCommand for FnFilterCommand {
2019-07-23 22:22:11 +00:00
fn name(&self) -> &str {
&self.name
}
fn run(
&self,
args: CommandArgs,
registry: &registry::CommandRegistry,
) -> Result<OutputStream, ShellError> {
let CommandArgs {
host,
env,
call_info,
input,
} = args;
let host: Arc<Mutex<dyn Host>> = host.clone();
let env: Arc<Mutex<Environment>> = env.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(&registry, &Scope::it_value(it.clone()))
{
Err(err) => return OutputStream::from(vec![Err(err)]).values,
Ok(args) => args,
};
let args = EvaluatedFilterCommandArgs::new(host.clone(), env.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<ReturnValue> = result.boxed();
Ok(result.into())
}
}
pub struct FnRawCommand {
name: String,
func: Box<
dyn Fn(CommandArgs, &registry::CommandRegistry) -> Result<OutputStream, ShellError>
+ Send
+ Sync,
>,
}
2019-08-02 19:15:07 +00:00
impl StaticCommand for FnRawCommand {
fn name(&self) -> &str {
&self.name
}
2019-07-23 22:22:11 +00:00
fn run(
&self,
args: CommandArgs,
registry: &registry::CommandRegistry,
) -> Result<OutputStream, ShellError> {
(self.func)(args, registry)
2019-05-22 07:12:03 +00:00
}
2019-05-28 06:45:18 +00:00
}
pub fn command(
name: &str,
2019-07-23 22:22:11 +00:00
func: Box<
dyn Fn(CommandArgs, &registry::CommandRegistry) -> Result<OutputStream, ShellError>
+ Send
+ Sync,
>,
2019-08-02 19:15:07 +00:00
) -> Arc<Command> {
Arc::new(Command::Static(Arc::new(FnRawCommand {
2019-07-23 22:22:11 +00:00
name: name.to_string(),
func,
2019-08-02 19:15:07 +00:00
})))
}
pub fn static_command(command: impl StaticCommand + 'static) -> Arc<Command> {
Arc::new(Command::Static(Arc::new(command)))
2019-07-23 22:22:11 +00:00
}
2019-07-24 04:10:48 +00:00
#[allow(unused)]
2019-07-23 22:22:11 +00:00
pub fn filter(
name: &str,
func: fn(EvaluatedFilterCommandArgs) -> Result<OutputStream, ShellError>,
2019-08-02 19:15:07 +00:00
) -> Arc<Command> {
Arc::new(Command::Static(Arc::new(FnFilterCommand {
2019-06-07 06:34:42 +00:00
name: name.to_string(),
func,
2019-08-02 19:15:07 +00:00
})))
2019-06-07 06:34:42 +00:00
}