use calamine::*; use indexmap::map::IndexMap; use nu_engine::CallExt; use nu_protocol::ast::Call; use nu_protocol::engine::{Command, EngineState, Stack}; use nu_protocol::{ Category, Example, PipelineData, ShellError, Signature, Span, SyntaxShape, Value, }; use std::io::Cursor; #[derive(Clone)] pub struct FromOds; impl Command for FromOds { fn name(&self) -> &str { "from ods" } fn signature(&self) -> Signature { Signature::build("from ods") .named( "sheets", SyntaxShape::List(Box::new(SyntaxShape::String)), "Only convert specified sheets", Some('s'), ) .category(Category::Formats) } fn usage(&self) -> &str { "Parse OpenDocument Spreadsheet(.ods) data and create table." } fn run( &self, engine_state: &EngineState, stack: &mut Stack, call: &Call, input: PipelineData, ) -> Result { let head = call.head; let sel_sheets = if let Some(Value::List { vals: columns, .. }) = call.get_flag(engine_state, stack, "sheets")? { convert_columns(columns.as_slice())? } else { vec![] }; from_ods(input, head, sel_sheets) } fn examples(&self) -> Vec { vec![ Example { description: "Convert binary .ods data to a table", example: "open test.txt | from ods", result: None, }, Example { description: "Convert binary .ods data to a table, specifying the tables", example: "open test.txt | from ods -s [Spreadsheet1]", result: None, }, ] } } fn convert_columns(columns: &[Value]) -> Result, ShellError> { let res = columns .iter() .map(|value| match &value { Value::String { val: s, .. } => Ok(s.clone()), _ => Err(ShellError::IncompatibleParametersSingle( "Incorrect column format, Only string as column name".to_string(), value.span().unwrap_or_else(|_| Span::unknown()), )), }) .collect::, _>>()?; Ok(res) } fn collect_binary(input: PipelineData) -> Result, ShellError> { let mut bytes = vec![]; let mut values = input.into_iter(); loop { match values.next() { Some(Value::Binary { val: b, .. }) => { bytes.extend_from_slice(&b); } Some(x) => { return Err(ShellError::UnsupportedInput( "Expected binary from pipeline".to_string(), x.span().unwrap_or_else(|_| Span::unknown()), )) } None => break, } } Ok(bytes) } fn from_ods( input: PipelineData, head: Span, sel_sheets: Vec, ) -> Result { let bytes = collect_binary(input)?; let buf: Cursor> = Cursor::new(bytes); let mut ods = Ods::<_>::new(buf) .map_err(|_| ShellError::UnsupportedInput("Could not load ods file".to_string(), head))?; let mut dict = IndexMap::new(); let mut sheet_names = ods.sheet_names().to_owned(); if !sel_sheets.is_empty() { sheet_names.retain(|e| sel_sheets.contains(e)); } for sheet_name in &sheet_names { let mut sheet_output = vec![]; if let Some(Ok(current_sheet)) = ods.worksheet_range(sheet_name) { for row in current_sheet.rows() { let mut row_output = IndexMap::new(); for (i, cell) in row.iter().enumerate() { let value = match cell { DataType::Empty => Value::nothing(head), DataType::String(s) => Value::string(s, head), DataType::Float(f) => Value::Float { val: *f, span: head, }, DataType::Int(i) => Value::Int { val: *i, span: head, }, DataType::Bool(b) => Value::Bool { val: *b, span: head, }, _ => Value::nothing(head), }; row_output.insert(format!("Column{}", i), value); } let (cols, vals) = row_output .into_iter() .fold((vec![], vec![]), |mut acc, (k, v)| { acc.0.push(k); acc.1.push(v); acc }); let record = Value::Record { cols, vals, span: head, }; sheet_output.push(record); } dict.insert( sheet_name, Value::List { vals: sheet_output, span: head, }, ); } else { return Err(ShellError::UnsupportedInput( "Could not load sheet".to_string(), head, )); } } let (cols, vals) = dict.into_iter().fold((vec![], vec![]), |mut acc, (k, v)| { acc.0.push(k.clone()); acc.1.push(v); acc }); let record = Value::Record { cols, vals, span: head, }; Ok(PipelineData::Value(record, None)) } #[cfg(test)] mod tests { use super::*; #[test] fn test_examples() { use crate::test_examples; test_examples(FromOds {}) } }