|  |  |  | @ -153,25 +153,28 @@ impl<W: Write> TsvSolutionsWriter<W> { | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | fn write_tsv_term<'a>(term: impl Into<TermRef<'a>>, sink: &mut impl Write) -> io::Result<()> { | 
			
		
	
		
			
				
					|  |  |  |  |     //TODO: full Turtle serialization
 | 
			
		
	
		
			
				
					|  |  |  |  |     match term.into() { | 
			
		
	
		
			
				
					|  |  |  |  |         TermRef::NamedNode(node) => write!(sink, "<{}>", node.as_str()), | 
			
		
	
		
			
				
					|  |  |  |  |         TermRef::BlankNode(node) => write!(sink, "_:{}", node.as_str()), | 
			
		
	
		
			
				
					|  |  |  |  |         TermRef::Literal(literal) => match literal.datatype() { | 
			
		
	
		
			
				
					|  |  |  |  |             xsd::BOOLEAN => match literal.value() { | 
			
		
	
		
			
				
					|  |  |  |  |                 "true" | "1" => sink.write_all(b"true"), | 
			
		
	
		
			
				
					|  |  |  |  |                 "false" | "0" => sink.write_all(b"false"), | 
			
		
	
		
			
				
					|  |  |  |  |                 _ => sink.write_all(literal.to_string().as_bytes()), | 
			
		
	
		
			
				
					|  |  |  |  |             }, | 
			
		
	
		
			
				
					|  |  |  |  |             xsd::INTEGER => { | 
			
		
	
		
			
				
					|  |  |  |  |                 if literal.value().bytes().all(|c| c.is_ascii_digit()) { | 
			
		
	
		
			
				
					|  |  |  |  |                     sink.write_all(literal.value().as_bytes()) | 
			
		
	
		
			
				
					|  |  |  |  |         TermRef::Literal(literal) => { | 
			
		
	
		
			
				
					|  |  |  |  |             let value = literal.value(); | 
			
		
	
		
			
				
					|  |  |  |  |             if let Some(language) = literal.language() { | 
			
		
	
		
			
				
					|  |  |  |  |                 write_tsv_quoted_str(value, sink)?; | 
			
		
	
		
			
				
					|  |  |  |  |                 write!(sink, "@{}", language) | 
			
		
	
		
			
				
					|  |  |  |  |             } else { | 
			
		
	
		
			
				
					|  |  |  |  |                     sink.write_all(literal.to_string().as_bytes()) | 
			
		
	
		
			
				
					|  |  |  |  |                 match literal.datatype() { | 
			
		
	
		
			
				
					|  |  |  |  |                     xsd::BOOLEAN if is_turtle_boolean(value) => sink.write_all(value.as_bytes()), | 
			
		
	
		
			
				
					|  |  |  |  |                     xsd::INTEGER if is_turtle_integer(value) => sink.write_all(value.as_bytes()), | 
			
		
	
		
			
				
					|  |  |  |  |                     xsd::DECIMAL if is_turtle_decimal(value) => sink.write_all(value.as_bytes()), | 
			
		
	
		
			
				
					|  |  |  |  |                     xsd::DOUBLE if is_turtle_double(value) => sink.write_all(value.as_bytes()), | 
			
		
	
		
			
				
					|  |  |  |  |                     xsd::STRING => write_tsv_quoted_str(value, sink), | 
			
		
	
		
			
				
					|  |  |  |  |                     datatype => { | 
			
		
	
		
			
				
					|  |  |  |  |                         write_tsv_quoted_str(value, sink)?; | 
			
		
	
		
			
				
					|  |  |  |  |                         write!(sink, "^^<{}>", datatype.as_str()) | 
			
		
	
		
			
				
					|  |  |  |  |                     } | 
			
		
	
		
			
				
					|  |  |  |  |                 } | 
			
		
	
		
			
				
					|  |  |  |  |             } | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  |             _ => sink.write_all(literal.to_string().as_bytes()), | 
			
		
	
		
			
				
					|  |  |  |  |         }, | 
			
		
	
		
			
				
					|  |  |  |  |         #[cfg(feature = "rdf-star")] | 
			
		
	
		
			
				
					|  |  |  |  |         TermRef::Triple(triple) => { | 
			
		
	
		
			
				
					|  |  |  |  |             sink.write_all(b"<<")?; | 
			
		
	
	
		
			
				
					|  |  |  | @ -186,6 +189,92 @@ fn write_tsv_term<'a>(term: impl Into<TermRef<'a>>, sink: &mut impl Write) -> io | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | fn write_tsv_quoted_str(string: &str, f: &mut impl Write) -> io::Result<()> { | 
			
		
	
		
			
				
					|  |  |  |  |     f.write_all(b"\"")?; | 
			
		
	
		
			
				
					|  |  |  |  |     for c in string.bytes() { | 
			
		
	
		
			
				
					|  |  |  |  |         match c { | 
			
		
	
		
			
				
					|  |  |  |  |             b'\t' => f.write_all(b"\\t"), | 
			
		
	
		
			
				
					|  |  |  |  |             b'\n' => f.write_all(b"\\n"), | 
			
		
	
		
			
				
					|  |  |  |  |             b'\r' => f.write_all(b"\\r"), | 
			
		
	
		
			
				
					|  |  |  |  |             b'"' => f.write_all(b"\\\""), | 
			
		
	
		
			
				
					|  |  |  |  |             b'\\' => f.write_all(b"\\\\"), | 
			
		
	
		
			
				
					|  |  |  |  |             c => f.write_all(&[c]), | 
			
		
	
		
			
				
					|  |  |  |  |         }?; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     f.write_all(b"\"") | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | fn is_turtle_boolean(value: &str) -> bool { | 
			
		
	
		
			
				
					|  |  |  |  |     matches!(value, "true" | "false") | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | fn is_turtle_integer(value: &str) -> bool { | 
			
		
	
		
			
				
					|  |  |  |  |     // [19] 	INTEGER 	::= 	[+-]? [0-9]+
 | 
			
		
	
		
			
				
					|  |  |  |  |     let mut value = value.as_bytes(); | 
			
		
	
		
			
				
					|  |  |  |  |     if let Some(v) = value.strip_prefix(b"+") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } else if let Some(v) = value.strip_prefix(b"-") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     !value.is_empty() && value.iter().all(|c| c.is_ascii_digit()) | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | fn is_turtle_decimal(value: &str) -> bool { | 
			
		
	
		
			
				
					|  |  |  |  |     // [20] 	DECIMAL 	::= 	[+-]? [0-9]* '.' [0-9]+
 | 
			
		
	
		
			
				
					|  |  |  |  |     let mut value = value.as_bytes(); | 
			
		
	
		
			
				
					|  |  |  |  |     if let Some(v) = value.strip_prefix(b"+") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } else if let Some(v) = value.strip_prefix(b"-") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     while value.first().map_or(false, |c| c.is_ascii_digit()) { | 
			
		
	
		
			
				
					|  |  |  |  |         value = &value[1..]; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     if let Some(v) = value.strip_prefix(b".") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } else { | 
			
		
	
		
			
				
					|  |  |  |  |         return false; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     !value.is_empty() && value.iter().all(|c| c.is_ascii_digit()) | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | fn is_turtle_double(value: &str) -> bool { | 
			
		
	
		
			
				
					|  |  |  |  |     // [21] 	DOUBLE 	::= 	[+-]? ([0-9]+ '.' [0-9]* EXPONENT | '.' [0-9]+ EXPONENT | [0-9]+ EXPONENT)
 | 
			
		
	
		
			
				
					|  |  |  |  |     // [154s] 	EXPONENT 	::= 	[eE] [+-]? [0-9]+
 | 
			
		
	
		
			
				
					|  |  |  |  |     let mut value = value.as_bytes(); | 
			
		
	
		
			
				
					|  |  |  |  |     if let Some(v) = value.strip_prefix(b"+") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } else if let Some(v) = value.strip_prefix(b"-") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     let mut with_before = false; | 
			
		
	
		
			
				
					|  |  |  |  |     while value.first().map_or(false, |c| c.is_ascii_digit()) { | 
			
		
	
		
			
				
					|  |  |  |  |         value = &value[1..]; | 
			
		
	
		
			
				
					|  |  |  |  |         with_before = true; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     let mut with_after = false; | 
			
		
	
		
			
				
					|  |  |  |  |     if let Some(v) = value.strip_prefix(b".") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |         while value.first().map_or(false, |c| c.is_ascii_digit()) { | 
			
		
	
		
			
				
					|  |  |  |  |             value = &value[1..]; | 
			
		
	
		
			
				
					|  |  |  |  |             with_after = true; | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     if let Some(v) = value.strip_prefix(b"e") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } else if let Some(v) = value.strip_prefix(b"E") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } else { | 
			
		
	
		
			
				
					|  |  |  |  |         return false; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     if let Some(v) = value.strip_prefix(b"+") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } else if let Some(v) = value.strip_prefix(b"-") { | 
			
		
	
		
			
				
					|  |  |  |  |         value = v; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     (with_before || with_after) && !value.is_empty() && value.iter().all(|c| c.is_ascii_digit()) | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | pub enum TsvQueryResultsReader<R: BufRead> { | 
			
		
	
		
			
				
					|  |  |  |  |     Solutions { | 
			
		
	
		
			
				
					|  |  |  |  |         variables: Vec<Variable>, | 
			
		
	
	
		
			
				
					|  |  |  | @ -259,6 +348,7 @@ impl<R: BufRead> TsvSolutionsReader<R> { | 
			
		
	
		
			
				
					|  |  |  |  | #[cfg(test)] | 
			
		
	
		
			
				
					|  |  |  |  | mod tests { | 
			
		
	
		
			
				
					|  |  |  |  |     use super::*; | 
			
		
	
		
			
				
					|  |  |  |  |     use std::error::Error; | 
			
		
	
		
			
				
					|  |  |  |  |     use std::io::Cursor; | 
			
		
	
		
			
				
					|  |  |  |  |     use std::rc::Rc; | 
			
		
	
		
			
				
					|  |  |  |  |     use std::str; | 
			
		
	
	
		
			
				
					|  |  |  | @ -302,6 +392,10 @@ mod tests { | 
			
		
	
		
			
				
					|  |  |  |  |                     Some(BlankNode::new_unchecked("b1").into()), | 
			
		
	
		
			
				
					|  |  |  |  |                     Some(Literal::new_typed_literal("123", xsd::INTEGER).into()), | 
			
		
	
		
			
				
					|  |  |  |  |                 ], | 
			
		
	
		
			
				
					|  |  |  |  |                 vec![ | 
			
		
	
		
			
				
					|  |  |  |  |                     None, | 
			
		
	
		
			
				
					|  |  |  |  |                     Some(Literal::new_simple_literal("escape,\t\r\n").into()), | 
			
		
	
		
			
				
					|  |  |  |  |                 ], | 
			
		
	
		
			
				
					|  |  |  |  |             ], | 
			
		
	
		
			
				
					|  |  |  |  |         ) | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
	
		
			
				
					|  |  |  | @ -320,25 +414,44 @@ mod tests { | 
			
		
	
		
			
				
					|  |  |  |  |             )?; | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  |         let result = writer.finish()?; | 
			
		
	
		
			
				
					|  |  |  |  |         assert_eq!(str::from_utf8(&result).unwrap(), "x,literal\r\nhttp://example/x,String\r\nhttp://example/x,\"String-with-dquote\"\"\"\r\n_:b0,Blank node\r\n,Missing 'x'\r\n,\r\nhttp://example/x,\r\n_:b1,String-with-lang\r\n_:b1,123"); | 
			
		
	
		
			
				
					|  |  |  |  |         assert_eq!(str::from_utf8(&result).unwrap(), "x,literal\r\nhttp://example/x,String\r\nhttp://example/x,\"String-with-dquote\"\"\"\r\n_:b0,Blank node\r\n,Missing 'x'\r\n,\r\nhttp://example/x,\r\n_:b1,String-with-lang\r\n_:b1,123\r\n,\"escape,\t\r\n\""); | 
			
		
	
		
			
				
					|  |  |  |  |         Ok(()) | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     #[test] | 
			
		
	
		
			
				
					|  |  |  |  |     fn test_tsv_serialization() -> io::Result<()> { | 
			
		
	
		
			
				
					|  |  |  |  |     fn test_tsv_roundtrip() -> Result<(), Box<dyn Error>> { | 
			
		
	
		
			
				
					|  |  |  |  |         let (variables, solutions) = build_example(); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         // Write
 | 
			
		
	
		
			
				
					|  |  |  |  |         let mut writer = TsvSolutionsWriter::start(Vec::new(), variables.clone())?; | 
			
		
	
		
			
				
					|  |  |  |  |         let variables = Rc::new(variables); | 
			
		
	
		
			
				
					|  |  |  |  |         for solution in solutions { | 
			
		
	
		
			
				
					|  |  |  |  |         for solution in &solutions { | 
			
		
	
		
			
				
					|  |  |  |  |             writer.write( | 
			
		
	
		
			
				
					|  |  |  |  |                 variables | 
			
		
	
		
			
				
					|  |  |  |  |                     .iter() | 
			
		
	
		
			
				
					|  |  |  |  |                     .zip(&solution) | 
			
		
	
		
			
				
					|  |  |  |  |                     .zip(solution) | 
			
		
	
		
			
				
					|  |  |  |  |                     .filter_map(|(v, s)| s.as_ref().map(|s| (v.as_ref(), s.as_ref()))), | 
			
		
	
		
			
				
					|  |  |  |  |             )?; | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  |         let result = writer.finish()?; | 
			
		
	
		
			
				
					|  |  |  |  |         assert_eq!(str::from_utf8(&result).unwrap(), "?x\t?literal\n<http://example/x>\t\"String\"\n<http://example/x>\t\"String-with-dquote\\\"\"\n_:b0\t\"Blank node\"\n\t\"Missing 'x'\"\n\t\n<http://example/x>\t\n_:b1\t\"String-with-lang\"@en\n_:b1\t123"); | 
			
		
	
		
			
				
					|  |  |  |  |         assert_eq!(str::from_utf8(&result).unwrap(), "?x\t?literal\n<http://example/x>\t\"String\"\n<http://example/x>\t\"String-with-dquote\\\"\"\n_:b0\t\"Blank node\"\n\t\"Missing 'x'\"\n\t\n<http://example/x>\t\n_:b1\t\"String-with-lang\"@en\n_:b1\t123\n\t\"escape,\\t\\r\\n\""); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         // Read
 | 
			
		
	
		
			
				
					|  |  |  |  |         if let TsvQueryResultsReader::Solutions { | 
			
		
	
		
			
				
					|  |  |  |  |             solutions: mut solutions_iter, | 
			
		
	
		
			
				
					|  |  |  |  |             variables: actual_variables, | 
			
		
	
		
			
				
					|  |  |  |  |         } = TsvQueryResultsReader::read(Cursor::new(result))? | 
			
		
	
		
			
				
					|  |  |  |  |         { | 
			
		
	
		
			
				
					|  |  |  |  |             assert_eq!(actual_variables.as_slice(), variables.as_slice()); | 
			
		
	
		
			
				
					|  |  |  |  |             let mut rows = Vec::new(); | 
			
		
	
		
			
				
					|  |  |  |  |             while let Some(row) = solutions_iter.read_next()? { | 
			
		
	
		
			
				
					|  |  |  |  |                 rows.push(row); | 
			
		
	
		
			
				
					|  |  |  |  |             } | 
			
		
	
		
			
				
					|  |  |  |  |             assert_eq!(rows, solutions); | 
			
		
	
		
			
				
					|  |  |  |  |         } else { | 
			
		
	
		
			
				
					|  |  |  |  |             unreachable!() | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         Ok(()) | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  |  | 
 |