parent
329a80bc8c
commit
f553a28f0b
@ -0,0 +1,325 @@ |
|||||||
|
use criterion::{criterion_group, criterion_main, Criterion}; |
||||||
|
use failure::format_err; |
||||||
|
use oxigraph::model::vocab::rdf; |
||||||
|
use oxigraph::model::*; |
||||||
|
use oxigraph::sparql::*; |
||||||
|
use oxigraph::*; |
||||||
|
use std::fs::File; |
||||||
|
use std::io::{BufRead, BufReader, Read}; |
||||||
|
use std::path::PathBuf; |
||||||
|
|
||||||
|
criterion_group!(sparql, sparql_w3c_syntax_bench); |
||||||
|
|
||||||
|
criterion_main!(sparql); |
||||||
|
|
||||||
|
fn sparql_w3c_syntax_bench(c: &mut Criterion) { |
||||||
|
let manifest_urls = vec![ |
||||||
|
"http://www.w3.org/2001/sw/DataAccess/tests/data-r2/manifest-syntax.ttl", |
||||||
|
"http://www.w3.org/2009/sparql/docs/tests/data-sparql11/syntax-query/manifest.ttl", |
||||||
|
"http://www.w3.org/2009/sparql/docs/tests/data-sparql11/syntax-fed/manifest.ttl", |
||||||
|
"http://www.w3.org/2009/sparql/docs/tests/data-sparql11/construct/manifest.ttl", |
||||||
|
"http://www.w3.org/2009/sparql/docs/tests/data-sparql11/grouping/manifest.ttl", |
||||||
|
"http://www.w3.org/2009/sparql/docs/tests/data-sparql11/aggregates/manifest.ttl", |
||||||
|
]; |
||||||
|
let queries: Vec<_> = manifest_urls |
||||||
|
.into_iter() |
||||||
|
.flat_map(TestManifest::new) |
||||||
|
.flat_map(|test| { |
||||||
|
let test = test.unwrap(); |
||||||
|
if test.kind == "PositiveSyntaxTest" || test.kind == "PositiveSyntaxTest11" { |
||||||
|
Some((read_file_to_string(&test.query).unwrap(), test.query)) |
||||||
|
} else { |
||||||
|
None |
||||||
|
} |
||||||
|
}) |
||||||
|
.collect(); |
||||||
|
|
||||||
|
c.bench_function("query parser", |b| { |
||||||
|
b.iter(|| { |
||||||
|
for (query, base) in &queries { |
||||||
|
Query::parse(query, Some(base)).unwrap(); |
||||||
|
} |
||||||
|
}) |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
fn to_relative_path(url: &str) -> Result<String> { |
||||||
|
if url.starts_with("http://www.w3.org/2001/sw/DataAccess/tests/data-r2/") { |
||||||
|
Ok(url.replace( |
||||||
|
"http://www.w3.org/2001/sw/DataAccess/tests/", |
||||||
|
"rdf-tests/sparql11/", |
||||||
|
)) |
||||||
|
} else if url.starts_with("http://www.w3.org/2009/sparql/docs/tests/data-sparql11/") { |
||||||
|
Ok(url.replace( |
||||||
|
"http://www.w3.org/2009/sparql/docs/tests/", |
||||||
|
"rdf-tests/sparql11/", |
||||||
|
)) |
||||||
|
} else { |
||||||
|
Err(format_err!("Not supported url for file: {}", url)) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
fn read_file(url: &str) -> Result<impl BufRead> { |
||||||
|
let mut base_path = PathBuf::from(env!("CARGO_MANIFEST_DIR")); |
||||||
|
base_path.push("tests"); |
||||||
|
base_path.push(to_relative_path(url)?); |
||||||
|
|
||||||
|
Ok(BufReader::new(File::open(&base_path).map_err(|e| { |
||||||
|
format_err!("Opening file {} failed with {}", base_path.display(), e) |
||||||
|
})?)) |
||||||
|
} |
||||||
|
|
||||||
|
fn read_file_to_string(url: &str) -> Result<String> { |
||||||
|
let mut string = String::default(); |
||||||
|
read_file(url)?.read_to_string(&mut string)?; |
||||||
|
Ok(string) |
||||||
|
} |
||||||
|
|
||||||
|
fn load_graph(url: &str) -> Result<SimpleGraph> { |
||||||
|
let repository = MemoryRepository::default(); |
||||||
|
load_graph_to_repository(url, &mut repository.connection().unwrap(), None)?; |
||||||
|
Ok(repository |
||||||
|
.connection() |
||||||
|
.unwrap() |
||||||
|
.quads_for_pattern(None, None, None, Some(None)) |
||||||
|
.map(|q| q.unwrap().into_triple()) |
||||||
|
.collect()) |
||||||
|
} |
||||||
|
|
||||||
|
fn load_graph_to_repository( |
||||||
|
url: &str, |
||||||
|
connection: &mut <&MemoryRepository as Repository>::Connection, |
||||||
|
to_graph_name: Option<&NamedOrBlankNode>, |
||||||
|
) -> Result<()> { |
||||||
|
let syntax = if url.ends_with(".nt") { |
||||||
|
GraphSyntax::NTriples |
||||||
|
} else if url.ends_with(".ttl") { |
||||||
|
GraphSyntax::Turtle |
||||||
|
} else if url.ends_with(".rdf") { |
||||||
|
GraphSyntax::RdfXml |
||||||
|
} else { |
||||||
|
return Err(format_err!("Serialization type not found for {}", url)); |
||||||
|
}; |
||||||
|
connection.load_graph(read_file(url)?, syntax, to_graph_name, Some(url)) |
||||||
|
} |
||||||
|
|
||||||
|
mod mf { |
||||||
|
use lazy_static::lazy_static; |
||||||
|
use oxigraph::model::NamedNode; |
||||||
|
|
||||||
|
lazy_static! { |
||||||
|
pub static ref INCLUDE: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/test-manifest#include") |
||||||
|
.unwrap(); |
||||||
|
pub static ref ENTRIES: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/test-manifest#entries") |
||||||
|
.unwrap(); |
||||||
|
pub static ref ACTION: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/test-manifest#action") |
||||||
|
.unwrap(); |
||||||
|
pub static ref RESULT: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/test-manifest#result") |
||||||
|
.unwrap(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
mod qt { |
||||||
|
use lazy_static::lazy_static; |
||||||
|
use oxigraph::model::NamedNode; |
||||||
|
|
||||||
|
lazy_static! { |
||||||
|
pub static ref QUERY: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/test-query#query") |
||||||
|
.unwrap(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
mod rs { |
||||||
|
use lazy_static::lazy_static; |
||||||
|
use oxigraph::model::NamedNode; |
||||||
|
|
||||||
|
lazy_static! { |
||||||
|
pub static ref RESULT_SET: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/result-set#ResultSet") |
||||||
|
.unwrap(); |
||||||
|
pub static ref RESULT_VARIABLE: NamedNode = NamedNode::parse( |
||||||
|
"http://www.w3.org/2001/sw/DataAccess/tests/result-set#resultVariable" |
||||||
|
) |
||||||
|
.unwrap(); |
||||||
|
pub static ref SOLUTION: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/result-set#solution") |
||||||
|
.unwrap(); |
||||||
|
pub static ref BINDING: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/result-set#binding") |
||||||
|
.unwrap(); |
||||||
|
pub static ref VALUE: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/result-set#value") |
||||||
|
.unwrap(); |
||||||
|
pub static ref VARIABLE: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/result-set#variable") |
||||||
|
.unwrap(); |
||||||
|
pub static ref INDEX: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/result-set#index") |
||||||
|
.unwrap(); |
||||||
|
pub static ref BOOLEAN: NamedNode = |
||||||
|
NamedNode::parse("http://www.w3.org/2001/sw/DataAccess/tests/result-set#boolean") |
||||||
|
.unwrap(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
struct Test { |
||||||
|
kind: String, |
||||||
|
query: String, |
||||||
|
} |
||||||
|
|
||||||
|
struct TestManifest { |
||||||
|
graph: SimpleGraph, |
||||||
|
tests_to_do: Vec<Term>, |
||||||
|
manifests_to_do: Vec<String>, |
||||||
|
} |
||||||
|
|
||||||
|
impl TestManifest { |
||||||
|
fn new(url: impl Into<String>) -> TestManifest { |
||||||
|
Self { |
||||||
|
graph: SimpleGraph::default(), |
||||||
|
tests_to_do: Vec::default(), |
||||||
|
manifests_to_do: vec![url.into()], |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
impl Iterator for TestManifest { |
||||||
|
type Item = Result<Test>; |
||||||
|
|
||||||
|
fn next(&mut self) -> Option<Result<Test>> { |
||||||
|
match self.tests_to_do.pop() { |
||||||
|
Some(Term::NamedNode(test_node)) => { |
||||||
|
let test_subject = NamedOrBlankNode::from(test_node.clone()); |
||||||
|
let kind = match self |
||||||
|
.graph |
||||||
|
.object_for_subject_predicate(&test_subject, &rdf::TYPE) |
||||||
|
{ |
||||||
|
Some(Term::NamedNode(c)) => match c.as_str().split('#').last() { |
||||||
|
Some(k) => k.to_string(), |
||||||
|
None => return self.next(), //We ignore the test
|
||||||
|
}, |
||||||
|
_ => return self.next(), //We ignore the test
|
||||||
|
}; |
||||||
|
let query = match self |
||||||
|
.graph |
||||||
|
.object_for_subject_predicate(&test_subject, &*mf::ACTION) |
||||||
|
{ |
||||||
|
Some(Term::NamedNode(n)) => n.as_str().to_owned(), |
||||||
|
Some(Term::BlankNode(n)) => { |
||||||
|
let n = n.clone().into(); |
||||||
|
match self.graph.object_for_subject_predicate(&n, &qt::QUERY) { |
||||||
|
Some(Term::NamedNode(q)) => q.as_str().to_owned(), |
||||||
|
Some(_) => return Some(Err(format_err!("invalid query"))), |
||||||
|
None => return Some(Err(format_err!("query not found"))), |
||||||
|
} |
||||||
|
} |
||||||
|
Some(_) => return Some(Err(format_err!("invalid action"))), |
||||||
|
None => { |
||||||
|
return Some(Err(format_err!( |
||||||
|
"action not found for test {}", |
||||||
|
test_subject |
||||||
|
))); |
||||||
|
} |
||||||
|
}; |
||||||
|
Some(Ok(Test { kind, query })) |
||||||
|
} |
||||||
|
Some(_) => Some(Err(format_err!("invalid test list"))), |
||||||
|
None => { |
||||||
|
match self.manifests_to_do.pop() { |
||||||
|
Some(url) => { |
||||||
|
let manifest = |
||||||
|
NamedOrBlankNode::from(NamedNode::parse(url.clone()).unwrap()); |
||||||
|
match load_graph(&url) { |
||||||
|
Ok(g) => self.graph.extend(g.into_iter()), |
||||||
|
Err(e) => return Some(Err(e)), |
||||||
|
} |
||||||
|
|
||||||
|
// New manifests
|
||||||
|
match self |
||||||
|
.graph |
||||||
|
.object_for_subject_predicate(&manifest, &*mf::INCLUDE) |
||||||
|
{ |
||||||
|
Some(Term::BlankNode(list)) => { |
||||||
|
self.manifests_to_do.extend( |
||||||
|
RdfListIterator::iter(&self.graph, list.clone().into()) |
||||||
|
.filter_map(|m| match m { |
||||||
|
Term::NamedNode(nm) => Some(nm.into_string()), |
||||||
|
_ => None, |
||||||
|
}), |
||||||
|
); |
||||||
|
} |
||||||
|
Some(_) => return Some(Err(format_err!("invalid tests list"))), |
||||||
|
None => (), |
||||||
|
} |
||||||
|
|
||||||
|
// New tests
|
||||||
|
match self |
||||||
|
.graph |
||||||
|
.object_for_subject_predicate(&manifest, &*mf::ENTRIES) |
||||||
|
{ |
||||||
|
Some(Term::BlankNode(list)) => { |
||||||
|
self.tests_to_do.extend(RdfListIterator::iter( |
||||||
|
&self.graph, |
||||||
|
list.clone().into(), |
||||||
|
)); |
||||||
|
} |
||||||
|
Some(term) => { |
||||||
|
return Some(Err(format_err!( |
||||||
|
"Invalid tests list. Got term {}", |
||||||
|
term |
||||||
|
))); |
||||||
|
} |
||||||
|
None => (), |
||||||
|
} |
||||||
|
} |
||||||
|
None => return None, |
||||||
|
} |
||||||
|
self.next() |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
struct RdfListIterator<'a> { |
||||||
|
graph: &'a SimpleGraph, |
||||||
|
current_node: Option<NamedOrBlankNode>, |
||||||
|
} |
||||||
|
|
||||||
|
impl<'a> RdfListIterator<'a> { |
||||||
|
fn iter(graph: &'a SimpleGraph, root: NamedOrBlankNode) -> RdfListIterator<'a> { |
||||||
|
RdfListIterator { |
||||||
|
graph, |
||||||
|
current_node: Some(root), |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
impl<'a> Iterator for RdfListIterator<'a> { |
||||||
|
type Item = Term; |
||||||
|
|
||||||
|
fn next(&mut self) -> Option<Term> { |
||||||
|
match self.current_node.clone() { |
||||||
|
Some(current) => { |
||||||
|
let result = self |
||||||
|
.graph |
||||||
|
.object_for_subject_predicate(¤t, &rdf::FIRST); |
||||||
|
self.current_node = match self |
||||||
|
.graph |
||||||
|
.object_for_subject_predicate(¤t, &rdf::REST) |
||||||
|
{ |
||||||
|
Some(Term::NamedNode(ref n)) if *n == *rdf::NIL => None, |
||||||
|
Some(Term::NamedNode(n)) => Some(n.clone().into()), |
||||||
|
Some(Term::BlankNode(n)) => Some(n.clone().into()), |
||||||
|
_ => None, |
||||||
|
}; |
||||||
|
result.cloned() |
||||||
|
} |
||||||
|
None => None, |
||||||
|
} |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue