nushell/src/commands/classified.rs

143 lines
3.9 KiB
Rust
Raw Normal View History

2019-05-22 09:12:03 +02:00
use crate::prelude::*;
use std::sync::Arc;
use subprocess::Exec;
2019-05-24 09:29:16 +02:00
crate struct ClassifiedInputStream {
crate objects: InputStream,
crate stdin: Option<std::fs::File>,
}
impl ClassifiedInputStream {
crate fn new() -> ClassifiedInputStream {
ClassifiedInputStream {
objects: VecDeque::new().boxed(),
stdin: None,
}
}
crate fn from_input_stream(stream: InputStream) -> ClassifiedInputStream {
ClassifiedInputStream {
objects: stream,
stdin: None,
}
}
crate fn from_stdout(stdout: std::fs::File) -> ClassifiedInputStream {
ClassifiedInputStream {
objects: VecDeque::new().boxed(),
stdin: Some(stdout),
}
}
}
2019-05-22 09:12:03 +02:00
crate enum ClassifiedCommand {
Internal(InternalCommand),
External(ExternalCommand),
}
impl ClassifiedCommand {
crate async fn run(
2019-05-22 09:12:03 +02:00
self,
2019-05-23 06:30:43 +02:00
context: &mut Context,
2019-05-24 09:29:16 +02:00
input: ClassifiedInputStream,
) -> Result<InputStream, ShellError> {
2019-05-22 09:12:03 +02:00
match self {
ClassifiedCommand::Internal(internal) => {
2019-05-24 09:29:16 +02:00
let result = context.run_command(internal.command, internal.args, input.objects)?;
let env = context.env.clone();
2019-05-22 09:12:03 +02:00
let stream = result.filter_map(move |v| match v {
ReturnValue::Action(action) => match action {
CommandAction::ChangeCwd(cwd) => {
env.lock().unwrap().cwd = cwd;
futures::future::ready(None)
}
},
2019-05-22 09:12:03 +02:00
ReturnValue::Value(v) => futures::future::ready(Some(v)),
});
2019-05-22 09:12:03 +02:00
Ok(stream.boxed() as InputStream)
2019-05-22 09:12:03 +02:00
}
ClassifiedCommand::External(external) => {
Exec::shell(&external.name)
.args(&external.args)
.cwd(context.env.lock().unwrap().cwd())
2019-05-22 09:12:03 +02:00
.join()
.unwrap();
Ok(VecDeque::new().boxed() as InputStream)
2019-05-22 09:12:03 +02:00
}
}
}
}
crate struct InternalCommand {
crate command: Arc<dyn Command>,
crate args: Vec<Value>,
}
2019-05-24 09:29:16 +02:00
impl InternalCommand {
crate async fn run(
self,
context: &mut Context,
input: ClassifiedInputStream,
) -> Result<InputStream, ShellError> {
let result = context.run_command(self.command, self.args, input.objects)?;
let env = context.env.clone();
let stream = result.filter_map(move |v| match v {
ReturnValue::Action(action) => match action {
CommandAction::ChangeCwd(cwd) => {
env.lock().unwrap().cwd = cwd;
futures::future::ready(None)
}
},
ReturnValue::Value(v) => futures::future::ready(Some(v)),
});
Ok(stream.boxed() as InputStream)
}
}
2019-05-22 09:12:03 +02:00
crate struct ExternalCommand {
crate name: String,
crate args: Vec<String>,
}
2019-05-24 09:29:16 +02:00
impl ExternalCommand {
crate async fn run(
self,
context: &mut Context,
mut input: ClassifiedInputStream,
stream_next: bool,
) -> Result<ClassifiedInputStream, ShellError> {
let mut process = Exec::shell(&self.name)
.args(&self.args)
.cwd(context.env.lock().unwrap().cwd());
if stream_next {
process = process.stdout(subprocess::Redirection::Pipe)
}
if let Some(stdin) = input.stdin {
process = process.stdin(stdin);
}
let mut popen = process.popen().unwrap();
if stream_next {
match &popen.stdout {
None => unreachable!(),
Some(stdout) => Ok(ClassifiedInputStream::from_stdout(stdout.try_clone()?)),
}
} else {
popen.stdin.take();
popen.wait()?;
Ok(ClassifiedInputStream::new())
}
}
}