nushell/src/plugin.rs

167 lines
5 KiB
Rust
Raw Normal View History

2019-08-09 04:51:21 +00:00
use crate::Signature;
2019-08-01 01:58:42 +00:00
use crate::Tagged;
2019-08-09 04:51:21 +00:00
use crate::{CallInfo, ReturnValue, ShellError, Value};
2019-07-02 07:56:20 +00:00
use serde::{Deserialize, Serialize};
use std::io;
pub trait Plugin {
2019-08-02 19:15:07 +00:00
fn config(&mut self) -> Result<Signature, ShellError>;
2019-07-28 23:34:37 +00:00
2019-09-02 06:11:05 +00:00
fn begin_filter(&mut self, _call_info: CallInfo) -> Result<Vec<ReturnValue>, ShellError> {
Ok(vec![])
2019-07-02 07:56:20 +00:00
}
2019-09-02 06:11:05 +00:00
fn filter(&mut self, _input: Tagged<Value>) -> Result<Vec<ReturnValue>, ShellError> {
Ok(vec![])
2019-07-02 07:56:20 +00:00
}
2019-09-02 06:11:05 +00:00
2019-07-26 18:40:00 +00:00
fn end_filter(&mut self) -> Result<Vec<ReturnValue>, ShellError> {
Ok(vec![])
2019-07-02 07:56:20 +00:00
}
2019-09-02 06:11:05 +00:00
fn sink(&mut self, _call_info: CallInfo, _input: Vec<Tagged<Value>>) {}
fn quit(&mut self) {}
2019-07-02 07:56:20 +00:00
}
pub fn serve_plugin(plugin: &mut dyn Plugin) {
let args = std::env::args();
if args.len() > 1 {
let input = args.skip(1).next();
let input = match input {
Some(arg) => std::fs::read_to_string(arg),
None => {
send_response(ShellError::string(format!("No input given.")));
return;
}
};
if let Ok(input) = input {
let command = serde_json::from_str::<NuCommand>(&input);
match command {
Ok(NuCommand::config) => {
send_response(plugin.config());
2019-08-31 21:19:59 +00:00
return;
}
Ok(NuCommand::begin_filter { params }) => {
send_response(plugin.begin_filter(params));
}
Ok(NuCommand::filter { params }) => {
send_response(plugin.filter(params));
}
2019-07-26 18:40:00 +00:00
Ok(NuCommand::end_filter) => {
send_response(plugin.end_filter());
2019-08-31 21:19:59 +00:00
return;
2019-07-26 18:40:00 +00:00
}
Ok(NuCommand::sink { params }) => {
plugin.sink(params.0, params.1);
return;
}
Ok(NuCommand::quit) => {
plugin.quit();
return;
}
e => {
send_response(ShellError::string(format!(
"Could not handle plugin message: {} {:?}",
input, e
)));
return;
2019-07-02 07:56:20 +00:00
}
}
}
} else {
loop {
let mut input = String::new();
match io::stdin().read_line(&mut input) {
Ok(_) => {
let command = serde_json::from_str::<NuCommand>(&input);
match command {
Ok(NuCommand::config) => {
send_response(plugin.config());
2019-08-31 21:19:59 +00:00
break;
}
Ok(NuCommand::begin_filter { params }) => {
send_response(plugin.begin_filter(params));
}
Ok(NuCommand::filter { params }) => {
send_response(plugin.filter(params));
}
2019-07-26 18:40:00 +00:00
Ok(NuCommand::end_filter) => {
send_response(plugin.end_filter());
2019-08-31 21:19:59 +00:00
break;
2019-07-26 18:40:00 +00:00
}
Ok(NuCommand::sink { params }) => {
plugin.sink(params.0, params.1);
break;
}
Ok(NuCommand::quit) => {
plugin.quit();
break;
}
e => {
send_response(ShellError::string(format!(
"Could not handle plugin message: {} {:?}",
input, e
)));
break;
}
}
}
e => {
send_response(ShellError::string(format!(
"Could not handle plugin message: {:?}",
e,
)));
break;
}
2019-07-02 07:56:20 +00:00
}
}
}
}
#[derive(Debug, Serialize, Deserialize)]
pub struct JsonRpc<T> {
jsonrpc: String,
pub method: String,
pub params: T,
}
impl<T> JsonRpc<T> {
pub fn new<U: Into<String>>(method: U, params: T) -> Self {
JsonRpc {
jsonrpc: "2.0".into(),
method: method.into(),
params,
}
}
}
fn send_response<T: Serialize>(result: T) {
let response = JsonRpc::new("response", result);
let response_raw = serde_json::to_string(&response);
match response_raw {
Ok(response) => println!("{}", response),
Err(err) => println!("{}", err),
}
2019-07-02 07:56:20 +00:00
}
#[derive(Debug, Serialize, Deserialize)]
#[serde(tag = "method")]
#[allow(non_camel_case_types)]
pub enum NuCommand {
config,
begin_filter {
params: CallInfo,
},
filter {
2019-08-01 01:58:42 +00:00
params: Tagged<Value>,
},
2019-07-26 18:40:00 +00:00
end_filter,
sink {
2019-08-01 01:58:42 +00:00
params: (CallInfo, Vec<Tagged<Value>>),
},
2019-07-02 07:56:20 +00:00
quit,
}