use crate::Signature; use crate::Tagged; use crate::{CallInfo, ReturnValue, ShellError, Value}; use serde::{Deserialize, Serialize}; use std::io; pub trait Plugin { fn config(&mut self) -> Result; fn begin_filter(&mut self, _call_info: CallInfo) -> Result, ShellError> { Ok(vec![]) } fn filter(&mut self, _input: Tagged) -> Result, ShellError> { Ok(vec![]) } fn end_filter(&mut self) -> Result, ShellError> { Ok(vec![]) } fn sink(&mut self, _call_info: CallInfo, _input: Vec>) {} fn quit(&mut self) {} } 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::(&input); match command { Ok(NuCommand::config) => { send_response(plugin.config()); return; } Ok(NuCommand::begin_filter { params }) => { send_response(plugin.begin_filter(params)); } Ok(NuCommand::filter { params }) => { send_response(plugin.filter(params)); } Ok(NuCommand::end_filter) => { send_response(plugin.end_filter()); return; } 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; } } } } else { loop { let mut input = String::new(); match io::stdin().read_line(&mut input) { Ok(_) => { let command = serde_json::from_str::(&input); match command { Ok(NuCommand::config) => { send_response(plugin.config()); break; } Ok(NuCommand::begin_filter { params }) => { send_response(plugin.begin_filter(params)); } Ok(NuCommand::filter { params }) => { send_response(plugin.filter(params)); } Ok(NuCommand::end_filter) => { send_response(plugin.end_filter()); break; } 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; } } } } } #[derive(Debug, Serialize, Deserialize)] pub struct JsonRpc { jsonrpc: String, pub method: String, pub params: T, } impl JsonRpc { pub fn new>(method: U, params: T) -> Self { JsonRpc { jsonrpc: "2.0".into(), method: method.into(), params, } } } fn send_response(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), } } #[derive(Debug, Serialize, Deserialize)] #[serde(tag = "method")] #[allow(non_camel_case_types)] pub enum NuCommand { config, begin_filter { params: CallInfo, }, filter { params: Tagged, }, end_filter, sink { params: (CallInfo, Vec>), }, quit, }