Introduced conversion to csv command.

This commit is contained in:
Andrés N. Robalino 2019-07-21 02:08:05 -05:00
parent a86a11413f
commit 191dacdd8b
12 changed files with 145 additions and 27 deletions

View File

@ -172,6 +172,7 @@ pub async fn cli() -> Result<(), Box<dyn Error>> {
command("reject", Box::new(reject::reject)), command("reject", Box::new(reject::reject)),
command("trim", Box::new(trim::trim)), command("trim", Box::new(trim::trim)),
command("to-array", Box::new(to_array::to_array)), command("to-array", Box::new(to_array::to_array)),
command("to-csv", Box::new(to_csv::to_csv)),
command("to-json", Box::new(to_json::to_json)), command("to-json", Box::new(to_json::to_json)),
command("to-toml", Box::new(to_toml::to_toml)), command("to-toml", Box::new(to_toml::to_toml)),
command("to-yaml", Box::new(to_yaml::to_yaml)), command("to-yaml", Box::new(to_yaml::to_yaml)),

View File

@ -34,6 +34,7 @@ crate mod split_row;
crate mod sysinfo; crate mod sysinfo;
crate mod table; crate mod table;
crate mod to_array; crate mod to_array;
crate mod to_csv;
crate mod to_json; crate mod to_json;
crate mod to_toml; crate mod to_toml;
crate mod to_yaml; crate mod to_yaml;

View File

@ -6,6 +6,7 @@ pub fn from_csv_string_to_value(
s: String, s: String,
span: impl Into<Span>, span: impl Into<Span>,
) -> Result<Spanned<Value>, Box<dyn std::error::Error>> { ) -> Result<Spanned<Value>, Box<dyn std::error::Error>> {
let mut reader = ReaderBuilder::new() let mut reader = ReaderBuilder::new()
.has_headers(false) .has_headers(false)
.from_reader(s.as_bytes()); .from_reader(s.as_bytes());

View File

@ -189,7 +189,7 @@ pub fn fetch(
}, },
Err(_) => { Err(_) => {
return Err(ShellError::labeled_error( return Err(ShellError::labeled_error(
"File cound not be opened", "File could not be opened",
"file not found", "file not found",
span, span,
)); ));

View File

@ -1,4 +1,5 @@
use crate::commands::command::SinkCommandArgs; use crate::commands::command::SinkCommandArgs;
use crate::commands::to_csv::{to_string as to_csv_to_string, value_to_csv_value};
use crate::commands::to_json::value_to_json_value; use crate::commands::to_json::value_to_json_value;
use crate::commands::to_toml::value_to_toml_value; use crate::commands::to_toml::value_to_toml_value;
use crate::commands::to_yaml::value_to_yaml_value; use crate::commands::to_yaml::value_to_yaml_value;
@ -37,6 +38,14 @@ pub fn save(args: SinkCommandArgs) -> Result<(), ShellError> {
}; };
let contents = match full_path.extension() { let contents = match full_path.extension() {
Some(x) if x == "csv" && !save_raw => {
if args.input.len() != 1 {
return Err(ShellError::string(
"saving to csv requires a single object (or use --raw)",
));
}
to_csv_to_string(&value_to_csv_value(&args.input[0])).unwrap()
}
Some(x) if x == "toml" && !save_raw => { Some(x) if x == "toml" && !save_raw => {
if args.input.len() != 1 { if args.input.len() != 1 {
return Err(ShellError::string( return Err(ShellError::string(

65
src/commands/to_csv.rs Normal file
View File

@ -0,0 +1,65 @@
use crate::object::{Primitive, Value};
use crate::prelude::*;
use log::debug;
use csv::WriterBuilder;
pub fn value_to_csv_value(v: &Value) -> Value {
debug!("value_to_csv_value(Value::Object(v)) where v = {:?}", v);
match v {
Value::Primitive(Primitive::String(s)) => Value::Primitive(Primitive::String(s.clone())),
Value::Primitive(Primitive::Nothing) => Value::Primitive(Primitive::Nothing),
Value::Object(o) => Value::Object(o.clone()),
Value::List(l) => Value::List(l.clone()),
Value::Block(_) => Value::Primitive(Primitive::Nothing),
_ => Value::Primitive(Primitive::Nothing)
}
}
pub fn to_string(v: &Value) -> Result<String, Box<dyn std::error::Error>> {
match v {
Value::List(_l) => return Ok(String::from("[list list]")),
Value::Object(o) => {
debug!("to_csv:to_string(Value::Object(v)) where v = {:?}", v);
let mut wtr = WriterBuilder::new().from_writer(vec![]);
let mut fields: VecDeque<String> = VecDeque::new();
let mut values: VecDeque<String> = VecDeque::new();
for (k, v) in o.entries.iter() {
fields.push_back(k.clone());
values.push_back(to_string(&v)?);
}
wtr.write_record(fields).expect("can not write.");
wtr.write_record(values).expect("can not write.");
return Ok(String::from_utf8(wtr.into_inner()?)?)
},
Value::Primitive(Primitive::String(s)) => return Ok(s.to_string()),
_ => return Err("Bad input".into())
}
}
pub fn to_csv(args: CommandArgs) -> Result<OutputStream, ShellError> {
let out = args.input;
let name_span = args.call_info.name_span;
Ok(out
.values
.map(
move |a| match to_string(&value_to_csv_value(&a.item)) {
Ok(x) => {
ReturnSuccess::value(Value::Primitive(Primitive::String(x)).spanned(name_span))
}
Err(_) => Err(ShellError::maybe_labeled_error(
"Can not convert to CSV string",
"can not convert piped data to CSV string",
name_span,
)),
},
)
.to_output_stream())
}

View File

@ -1,5 +1,6 @@
use crate::object::{Primitive, Value}; use crate::object::{Primitive, Value};
use crate::prelude::*; use crate::prelude::*;
use log::trace;
pub fn value_to_json_value(v: &Value) -> serde_json::Value { pub fn value_to_json_value(v: &Value) -> serde_json::Value {
match v { match v {

View File

@ -13,7 +13,7 @@ fn lines() {
} }
#[test] #[test]
fn open_csv() { fn open_can_parse_csv() {
nu!( nu!(
output, output,
cwd("tests/fixtures/formats"), cwd("tests/fixtures/formats"),
@ -24,7 +24,7 @@ fn open_csv() {
} }
#[test] #[test]
fn open_toml() { fn open_can_parse_toml() {
nu!( nu!(
output, output,
cwd("tests/fixtures/formats"), cwd("tests/fixtures/formats"),
@ -35,7 +35,7 @@ fn open_toml() {
} }
#[test] #[test]
fn open_json() { fn open_can_parse_json() {
nu!(output, nu!(output,
cwd("tests/fixtures/formats"), cwd("tests/fixtures/formats"),
"open sgml_description.json | get glossary.GlossDiv.GlossList.GlossEntry.GlossSee | echo $it"); "open sgml_description.json | get glossary.GlossDiv.GlossList.GlossEntry.GlossSee | echo $it");
@ -44,7 +44,7 @@ fn open_json() {
} }
#[test] #[test]
fn open_xml() { fn open_can_parse_xml() {
nu!( nu!(
output, output,
cwd("tests/fixtures/formats"), cwd("tests/fixtures/formats"),
@ -58,7 +58,7 @@ fn open_xml() {
} }
#[test] #[test]
fn open_ini() { fn open_can_parse_ini() {
nu!( nu!(
output, output,
cwd("tests/fixtures/formats"), cwd("tests/fixtures/formats"),
@ -76,11 +76,28 @@ fn open_error_if_file_not_found() {
"open i_dont_exist.txt | echo $it" "open i_dont_exist.txt | echo $it"
); );
assert!(output.contains("File cound not be opened")); assert!(output.contains("File could not be opened"));
} }
#[test] #[test]
fn rm() { fn save_can_write_out_csv() {
let (playground_path, tests_dir) = h::setup_playground_for("save_test");
let full_path = format!("{}/{}", playground_path, tests_dir );
let expected_file = format!("{}/{}", full_path , "cargo_sample.csv");
nu!(
_output,
cwd(&playground_path),
"open ../formats/cargo_sample.toml | inc package.version --minor | get package | save save_test/cargo_sample.csv"
);
let actual = h::file_contents(&expected_file);
assert!(actual.contains("[list list],A shell for the GitHub era,2018,ISC,nu,0.2.0"));
}
#[test]
fn rm_can_remove_a_file() {
let directory = "tests/fixtures/nuplayground"; let directory = "tests/fixtures/nuplayground";
let file = format!("{}/rm_test.txt", directory); let file = format!("{}/rm_test.txt", directory);
@ -92,16 +109,11 @@ fn rm() {
} }
#[test] #[test]
fn can_remove_directory_contents_with_recursive_flag() { fn rm_can_remove_directory_contents_with_recursive_flag() {
let path = "tests/fixtures/nuplayground/rm_test"; let (playground_path, tests_dir) = h::setup_playground_for("rm_test");
if h::file_exists_at(&path) {
h::delete_directory_at(path)
}
h::create_directory_at(path);
for f in ["yehuda.txt", "jonathan.txt", "andres.txt"].iter() { for f in ["yehuda.txt", "jonathan.txt", "andres.txt"].iter() {
h::create_file_at(&format!("{}/{}", path, f)); h::create_file_at(&format!("{}/{}/{}", playground_path, tests_dir, f));
} }
nu!( nu!(
@ -110,23 +122,19 @@ fn can_remove_directory_contents_with_recursive_flag() {
"rm rm_test --recursive" "rm rm_test --recursive"
); );
assert!(!h::file_exists_at(&path)); assert!(!h::file_exists_at(&format!("{}/{}", playground_path, tests_dir)));
} }
#[test] #[test]
fn rm_error_if_attempting_to_delete_a_directory_without_recursive_flag() { fn rm_error_if_attempting_to_delete_a_directory_without_recursive_flag() {
let path = "tests/fixtures/nuplayground/rm_test_2"; let (playground_path, tests_dir) = h::setup_playground_for("rm_test_2");
let full_path = format!("{}/{}", playground_path, tests_dir);
if h::file_exists_at(&path) {
h::delete_directory_at(path)
}
h::create_directory_at(path);
nu_error!(output, cwd("tests/fixtures/nuplayground"), "rm rm_test_2"); nu_error!(output, cwd("tests/fixtures/nuplayground"), "rm rm_test_2");
assert!(h::file_exists_at(&path)); assert!(h::file_exists_at(&full_path));
assert!(output.contains("is a directory")); assert!(output.contains("is a directory"));
h::delete_directory_at(path); h::delete_directory_at(&full_path);
} }
#[test] #[test]

View File

@ -2,6 +2,16 @@ mod helpers;
use helpers::in_directory as cwd; use helpers::in_directory as cwd;
#[test]
fn can_convert_table_to_csv_text_and_from_csv_text_back_into_table() {
nu!(output,
cwd("tests/fixtures/formats"),
"open caco3_plastics.csv | to-csv | from-csv | first 1 | get origin | echo $it");
assert_eq!(output, "SPAIN");
}
#[test] #[test]
fn can_convert_table_to_json_text_and_from_json_text_back_into_table() { fn can_convert_table_to_json_text_and_from_json_text_back_into_table() {
nu!(output, nu!(output,

View File

@ -1,2 +1,2 @@
rm_test *_test
*.txt *.txt

View File

@ -2,6 +2,8 @@
pub use std::path::PathBuf; pub use std::path::PathBuf;
use std::io::Read;
#[macro_export] #[macro_export]
macro_rules! nu { macro_rules! nu {
($out:ident, $cwd:expr, $commands:expr) => { ($out:ident, $cwd:expr, $commands:expr) => {
@ -77,6 +79,26 @@ macro_rules! nu_error {
}; };
} }
pub fn setup_playground_for(topic: &str) -> (String, String) {
let home = "tests/fixtures/nuplayground";
let full_path = format!("{}/{}", home, topic);
if file_exists_at(&full_path) {
delete_directory_at(&full_path);
}
create_directory_at(&full_path);
(home.to_string(), topic.to_string())
}
pub fn file_contents(full_path: &str) -> String {
let mut file = std::fs::File::open(full_path).expect("can not open file");
let mut contents = String::new();
file.read_to_string(&mut contents).expect("can not read file");
contents
}
pub fn create_file_at(full_path: &str) { pub fn create_file_at(full_path: &str) {
std::fs::write(PathBuf::from(full_path), "fake data".as_bytes()).expect("can not create file"); std::fs::write(PathBuf::from(full_path), "fake data".as_bytes()).expect("can not create file");
} }