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;
|
|
|
|
use crate::parser::{registry, Span, Spanned};
|
2019-05-13 17:30:51 +00:00
|
|
|
use crate::prelude::*;
|
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;
|
2019-07-19 19:48:14 +00:00
|
|
|
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 name(&self) -> Result<&str, ShellError> {
|
|
|
|
let head = &self.args.head();
|
|
|
|
match head.item() {
|
|
|
|
hir::RawExpression::Literal(hir::Literal::Bare) => Ok(head.span.slice(&self.source)),
|
|
|
|
hir::RawExpression::Literal(hir::Literal::String(span)) => Ok(span.slice(&self.source)),
|
|
|
|
other => Err(ShellError::type_error(
|
|
|
|
"Command name",
|
|
|
|
head.type_name().spanned(head.span),
|
|
|
|
)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn evaluate(
|
|
|
|
self,
|
|
|
|
registry: ®istry::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,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-20 02:27:10 +00:00
|
|
|
#[derive(Deserialize, Serialize, Debug)]
|
|
|
|
pub struct CallInfo {
|
2019-07-23 22:22:11 +00:00
|
|
|
pub args: registry::EvaluatedArgs,
|
2019-07-20 02:27:10 +00:00
|
|
|
pub source_map: SourceMap,
|
|
|
|
pub name_span: Option<Span>,
|
|
|
|
}
|
|
|
|
|
2019-06-22 03:43:37 +00:00
|
|
|
#[derive(Getters)]
|
|
|
|
#[get = "crate"]
|
2019-05-23 07:23:06 +00:00
|
|
|
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,
|
2019-05-23 07:23:06 +00:00
|
|
|
pub input: InputStream,
|
2019-05-16 00:21:46 +00:00
|
|
|
}
|
|
|
|
|
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: ®istry::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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum EvaluatedInput {
|
|
|
|
Static(InputStream),
|
|
|
|
Filter(Spanned<Value>),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl EvaluatedInput {
|
|
|
|
pub fn stream(self) -> InputStream {
|
|
|
|
match self {
|
|
|
|
EvaluatedInput::Static(stream) => stream,
|
|
|
|
EvaluatedInput::Filter(value) => vec![value].into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Getters)]
|
|
|
|
#[get = "pub"]
|
|
|
|
pub struct EvaluatedFilterCommandArgs {
|
|
|
|
args: EvaluatedCommandArgs,
|
|
|
|
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,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Getters)]
|
|
|
|
#[get = "crate"]
|
|
|
|
pub struct EvaluatedCommandArgs {
|
|
|
|
pub host: Arc<Mutex<dyn Host>>,
|
|
|
|
pub env: Arc<Mutex<Environment>>,
|
|
|
|
pub call_info: CallInfo,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl EvaluatedCommandArgs {
|
|
|
|
pub fn parts(self) -> () {}
|
|
|
|
|
|
|
|
pub fn call_args(&self) -> ®istry::EvaluatedArgs {
|
|
|
|
&self.call_info.args
|
|
|
|
}
|
|
|
|
|
2019-06-22 03:43:37 +00:00
|
|
|
pub fn nth(&self, pos: usize) -> Option<&Spanned<Value>> {
|
2019-07-20 02:27:10 +00:00
|
|
|
self.call_info.args.nth(pos)
|
2019-06-22 03:43:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn positional_iter(&self) -> impl Iterator<Item = &Spanned<Value>> {
|
2019-07-20 02:27:10 +00:00
|
|
|
self.call_info.args.positional_iter()
|
2019-06-22 03:43:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn expect_nth(&self, pos: usize) -> Result<&Spanned<Value>, ShellError> {
|
2019-07-20 02:27:10 +00:00
|
|
|
self.call_info.args.expect_nth(pos)
|
2019-06-22 03:43:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn len(&self) -> usize {
|
2019-07-20 02:27:10 +00:00
|
|
|
self.call_info.args.len()
|
2019-06-22 03:43:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get(&self, name: &str) -> Option<&Spanned<Value>> {
|
2019-07-20 02:27:10 +00:00
|
|
|
self.call_info.args.get(name)
|
2019-06-22 03:43:37 +00:00
|
|
|
}
|
|
|
|
|
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 {
|
2019-07-20 02:27:10 +00:00
|
|
|
self.call_info.args.has(name)
|
2019-06-22 03:43:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-07 06:34:42 +00:00
|
|
|
pub struct SinkCommandArgs {
|
|
|
|
pub ctx: Context,
|
2019-07-20 02:27:10 +00:00
|
|
|
pub call_info: CallInfo,
|
2019-07-08 16:44:53 +00:00
|
|
|
pub input: Vec<Spanned<Value>>,
|
2019-06-07 06:34:42 +00:00
|
|
|
}
|
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
impl SinkCommandArgs {
|
|
|
|
pub fn name_span(&self) -> Option<Span> {
|
|
|
|
self.call_info.name_span
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-27 04:56:48 +00:00
|
|
|
#[derive(Debug, Serialize, Deserialize)]
|
2019-05-13 17:30:51 +00:00
|
|
|
pub enum CommandAction {
|
2019-06-13 21:47:25 +00:00
|
|
|
ChangePath(PathBuf),
|
2019-07-19 19:48:14 +00:00
|
|
|
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)]
|
2019-07-03 20:31:15 +00:00
|
|
|
pub enum ReturnSuccess {
|
2019-07-08 16:44:53 +00:00
|
|
|
Value(Spanned<Value>),
|
2019-05-13 17:30:51 +00:00
|
|
|
Action(CommandAction),
|
2019-05-11 08:08:21 +00:00
|
|
|
}
|
|
|
|
|
2019-07-03 20:31:15 +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 {
|
2019-07-03 20:31:15 +00:00
|
|
|
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()))
|
|
|
|
}
|
|
|
|
|
2019-07-19 19:48:14 +00:00
|
|
|
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-05-13 17:30:51 +00:00
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
pub trait Command: Send + Sync {
|
|
|
|
fn run(
|
|
|
|
&self,
|
|
|
|
args: CommandArgs,
|
|
|
|
registry: ®istry::CommandRegistry,
|
|
|
|
) -> Result<OutputStream, ShellError>;
|
2019-05-28 06:45:18 +00:00
|
|
|
fn name(&self) -> &str;
|
|
|
|
|
2019-06-22 03:43:37 +00:00
|
|
|
fn config(&self) -> registry::CommandConfig {
|
|
|
|
registry::CommandConfig {
|
2019-05-28 06:45:18 +00:00
|
|
|
name: self.name().to_string(),
|
2019-07-03 20:31:15 +00:00
|
|
|
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,
|
|
|
|
is_sink: false,
|
2019-05-28 06:45:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-07 06:34:42 +00:00
|
|
|
pub trait Sink {
|
|
|
|
fn run(&self, args: SinkCommandArgs) -> Result<(), ShellError>;
|
|
|
|
fn name(&self) -> &str;
|
|
|
|
|
2019-06-22 03:43:37 +00:00
|
|
|
fn config(&self) -> registry::CommandConfig {
|
|
|
|
registry::CommandConfig {
|
2019-06-07 06:34:42 +00:00
|
|
|
name: self.name().to_string(),
|
2019-07-03 20:31:15 +00:00
|
|
|
positional: vec![],
|
2019-06-07 06:34:42 +00:00
|
|
|
rest_positional: true,
|
|
|
|
named: indexmap::IndexMap::new(),
|
2019-07-02 07:56:20 +00:00
|
|
|
is_filter: false,
|
|
|
|
is_sink: true,
|
2019-06-07 06:34:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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-13 17:30:51 +00:00
|
|
|
}
|
2019-05-22 07:12:03 +00:00
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
impl Command for FnFilterCommand {
|
|
|
|
fn name(&self) -> &str {
|
|
|
|
&self.name
|
|
|
|
}
|
|
|
|
|
|
|
|
fn run(
|
|
|
|
&self,
|
|
|
|
args: CommandArgs,
|
|
|
|
registry: ®istry::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(®istry, &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, ®istry::CommandRegistry) -> Result<OutputStream, ShellError>
|
|
|
|
+ Send
|
|
|
|
+ Sync,
|
|
|
|
>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Command for FnRawCommand {
|
|
|
|
fn run(
|
|
|
|
&self,
|
|
|
|
args: CommandArgs,
|
|
|
|
registry: ®istry::CommandRegistry,
|
|
|
|
) -> Result<OutputStream, ShellError> {
|
|
|
|
(self.func)(args, registry)
|
2019-05-22 07:12:03 +00:00
|
|
|
}
|
2019-05-28 06:45:18 +00:00
|
|
|
|
|
|
|
fn name(&self) -> &str {
|
|
|
|
&self.name
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn command(
|
|
|
|
name: &str,
|
2019-07-23 22:22:11 +00:00
|
|
|
func: Box<
|
|
|
|
dyn Fn(CommandArgs, ®istry::CommandRegistry) -> Result<OutputStream, ShellError>
|
|
|
|
+ Send
|
|
|
|
+ Sync,
|
|
|
|
>,
|
|
|
|
) -> Arc<dyn Command> {
|
|
|
|
Arc::new(FnRawCommand {
|
|
|
|
name: name.to_string(),
|
|
|
|
func,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn filter(
|
|
|
|
name: &str,
|
|
|
|
func: fn(EvaluatedFilterCommandArgs) -> Result<OutputStream, ShellError>,
|
2019-05-28 06:45:18 +00:00
|
|
|
) -> Arc<dyn Command> {
|
2019-07-23 22:22:11 +00:00
|
|
|
Arc::new(FnFilterCommand {
|
2019-05-28 06:45:18 +00:00
|
|
|
name: name.to_string(),
|
|
|
|
func,
|
|
|
|
})
|
2019-05-22 07:12:03 +00:00
|
|
|
}
|
2019-06-07 06:34:42 +00:00
|
|
|
|
|
|
|
pub struct FnSink {
|
|
|
|
name: String,
|
2019-07-03 17:37:09 +00:00
|
|
|
func: Box<dyn Fn(SinkCommandArgs) -> Result<(), ShellError>>,
|
2019-06-07 06:34:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Sink for FnSink {
|
|
|
|
fn run(&self, args: SinkCommandArgs) -> Result<(), ShellError> {
|
|
|
|
(self.func)(args)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn name(&self) -> &str {
|
|
|
|
&self.name
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-03 17:37:09 +00:00
|
|
|
pub fn sink(
|
|
|
|
name: &str,
|
|
|
|
func: Box<dyn Fn(SinkCommandArgs) -> Result<(), ShellError>>,
|
|
|
|
) -> Arc<dyn Sink> {
|
2019-06-07 06:34:42 +00:00
|
|
|
Arc::new(FnSink {
|
|
|
|
name: name.to_string(),
|
|
|
|
func,
|
|
|
|
})
|
|
|
|
}
|