Add a simple read/parse plugin to better handle text data

This commit is contained in:
Jonathan Turner 2019-10-30 11:33:36 +13:00
parent b6824d8b88
commit 3820fef801
5 changed files with 184 additions and 1 deletions

View file

@ -74,8 +74,8 @@ natural = "0.3.0"
serde_urlencoded = "0.6.1" serde_urlencoded = "0.6.1"
sublime_fuzzy = "0.5" sublime_fuzzy = "0.5"
trash = "1.0.0" trash = "1.0.0"
regex = "1"
regex = {version = "1", optional = true }
neso = { version = "0.5.0", optional = true } neso = { version = "0.5.0", optional = true }
crossterm = { version = "0.10.2", optional = true } crossterm = { version = "0.10.2", optional = true }
syntect = {version = "3.2.0", optional = true } syntect = {version = "3.2.0", optional = true }
@ -136,6 +136,10 @@ path = "src/plugins/add.rs"
name = "nu_plugin_edit" name = "nu_plugin_edit"
path = "src/plugins/edit.rs" path = "src/plugins/edit.rs"
[[bin]]
name = "nu_plugin_read"
path = "src/plugins/read.rs"
[[bin]] [[bin]]
name = "nu_plugin_str" name = "nu_plugin_str"
path = "src/plugins/str.rs" path = "src/plugins/str.rs"

156
src/plugins/read.rs Normal file
View file

@ -0,0 +1,156 @@
use nu::{
serve_plugin, CallInfo, Plugin, Primitive, ReturnSuccess, ReturnValue, ShellError, Signature,
SyntaxShape, Tagged, TaggedDictBuilder, Value,
};
use nom::{
bytes::complete::{tag, take_while},
IResult,
};
use regex::Regex;
#[derive(Debug)]
enum ReadCommand {
Text(String),
Column(String),
}
fn read(input: &str) -> IResult<&str, Vec<ReadCommand>> {
let mut output = vec![];
let mut loop_input = input;
loop {
let (input, before) = take_while(|c| c != '{')(loop_input)?;
if before.len() > 0 {
output.push(ReadCommand::Text(before.to_string()));
}
if input != "" {
// Look for column as we're now at one
let (input, _) = tag("{")(input)?;
let (input, column) = take_while(|c| c != '}')(input)?;
let (input, _) = tag("}")(input)?;
output.push(ReadCommand::Column(column.to_string()));
loop_input = input;
} else {
loop_input = input;
}
if loop_input == "" {
break;
}
}
Ok((loop_input, output))
}
fn column_names(commands: &[ReadCommand]) -> Vec<String> {
let mut output = vec![];
for command in commands {
match command {
ReadCommand::Column(c) => {
output.push(c.clone());
}
_ => {}
}
}
output
}
fn build_regex(commands: &[ReadCommand]) -> String {
let mut output = String::new();
for command in commands {
match command {
ReadCommand::Text(s) => {
output.push_str(&s.replace("(", "\\("));
}
ReadCommand::Column(_) => {
output.push_str("(.*)");
}
}
}
return output;
}
struct Read {
regex: Regex,
column_names: Vec<String>,
}
impl Read {
fn new() -> Self {
Read {
regex: Regex::new("").unwrap(),
column_names: vec![],
}
}
}
impl Plugin for Read {
fn config(&mut self) -> Result<Signature, ShellError> {
Ok(Signature::build("read")
.desc("Parse columns from string data using a simple pattern")
.required(
"pattern",
SyntaxShape::Any,
"the pattern to match. Eg) \"{foo}: {bar}\"",
)
.filter())
}
fn begin_filter(&mut self, call_info: CallInfo) -> Result<Vec<ReturnValue>, ShellError> {
if let Some(args) = call_info.args.positional {
match &args[0] {
Tagged {
item: Value::Primitive(Primitive::String(pattern)),
..
} => {
//self.pattern = s.clone();
let read_pattern = read(&pattern).unwrap();
let read_regex = build_regex(&read_pattern.1);
self.column_names = column_names(&read_pattern.1);
self.regex = Regex::new(&read_regex).unwrap();
}
Tagged { tag, .. } => {
return Err(ShellError::labeled_error(
"Unrecognized type in params",
"value",
tag,
));
}
}
}
Ok(vec![])
}
fn filter(&mut self, input: Tagged<Value>) -> Result<Vec<ReturnValue>, ShellError> {
let mut results = vec![];
match &input {
Tagged {
tag,
item: Value::Primitive(Primitive::String(s)),
} => {
//self.full_input.push_str(&s);
for cap in self.regex.captures_iter(&s) {
let mut dict = TaggedDictBuilder::new(tag);
for (idx, column_name) in self.column_names.iter().enumerate() {
dict.insert(column_name, Value::string(&cap[idx + 1].to_string()));
}
results.push(ReturnSuccess::value(dict.into_tagged_value()));
}
}
_ => {}
}
Ok(results)
}
}
fn main() {
serve_plugin(&mut Read::new());
}

View file

@ -448,6 +448,10 @@ mod tests {
loc: fixtures().join("cargo_sample.toml"), loc: fixtures().join("cargo_sample.toml"),
at: 0 at: 0
}, },
Res {
loc: fixtures().join("fileA.txt"),
at: 0
},
Res { Res {
loc: fixtures().join("jonathan.xml"), loc: fixtures().join("jonathan.xml"),
at: 0 at: 0

3
tests/fixtures/formats/fileA.txt vendored Normal file
View file

@ -0,0 +1,3 @@
VAR1=Chill
VAR2=StupidLongName
VAR3=AlsoChill

View file

@ -56,6 +56,22 @@ fn add_plugin() {
assert_eq!(actual, "1"); assert_eq!(actual, "1");
} }
#[test]
fn read_plugin() {
let actual = nu!(
cwd: "tests/fixtures/formats", h::pipeline(
r#"
open fileA.txt
| read "{Name}={Value}"
| nth 1
| get Value
| echo $it
"#
));
assert_eq!(actual, "StupidLongName");
}
#[test] #[test]
fn edit_plugin() { fn edit_plugin() {
let actual = nu!( let actual = nu!(