SPARQL UPDATE/ Avoids direct manipulation of encoded terms

pull/171/head
Tpt 4 years ago
parent 98f5f35dbb
commit ddc8eb584a
  1. 29
      lib/src/model/triple.rs
  2. 599
      lib/src/sparql/update.rs
  3. 47
      lib/src/storage/numeric_encoder.rs

@ -228,6 +228,20 @@ impl From<Triple> for Subject {
} }
} }
impl From<Arc<Triple>> for Subject {
#[inline]
fn from(node: Arc<Triple>) -> Self {
Self::Triple(node)
}
}
impl From<Box<Triple>> for Subject {
#[inline]
fn from(node: Box<Triple>) -> Self {
Self::Triple(node.into())
}
}
impl From<TripleRef<'_>> for Subject { impl From<TripleRef<'_>> for Subject {
#[inline] #[inline]
fn from(node: TripleRef<'_>) -> Self { fn from(node: TripleRef<'_>) -> Self {
@ -453,6 +467,7 @@ impl From<LiteralRef<'_>> for Term {
literal.into_owned().into() literal.into_owned().into()
} }
} }
impl From<Triple> for Term { impl From<Triple> for Term {
#[inline] #[inline]
fn from(triple: Triple) -> Self { fn from(triple: Triple) -> Self {
@ -460,6 +475,20 @@ impl From<Triple> for Term {
} }
} }
impl From<Arc<Triple>> for Term {
#[inline]
fn from(node: Arc<Triple>) -> Self {
Self::Triple(node)
}
}
impl From<Box<Triple>> for Term {
#[inline]
fn from(node: Box<Triple>) -> Self {
Self::Triple(node.into())
}
}
impl From<TripleRef<'_>> for Term { impl From<TripleRef<'_>> for Term {
#[inline] #[inline]
fn from(triple: TripleRef<'_>) -> Self { fn from(triple: TripleRef<'_>) -> Self {

@ -1,6 +1,9 @@
use crate::error::{invalid_data_error, invalid_input_error}; use crate::error::{invalid_data_error, invalid_input_error};
use crate::io::GraphFormat; use crate::io::GraphFormat;
use crate::model::{BlankNode as OxBlankNode, GraphNameRef, LiteralRef, NamedNodeRef}; use crate::model::{
BlankNode as OxBlankNode, GraphName as OxGraphName, GraphNameRef, Literal as OxLiteral,
NamedNode as OxNamedNode, NamedNodeRef, Quad as OxQuad, Term as OxTerm, Triple as OxTriple,
};
use crate::sparql::algebra::QueryDataset; use crate::sparql::algebra::QueryDataset;
use crate::sparql::dataset::DatasetView; use crate::sparql::dataset::DatasetView;
use crate::sparql::eval::SimpleEvaluator; use crate::sparql::eval::SimpleEvaluator;
@ -9,9 +12,7 @@ use crate::sparql::plan::EncodedTuple;
use crate::sparql::plan_builder::PlanBuilder; use crate::sparql::plan_builder::PlanBuilder;
use crate::sparql::{EvaluationError, UpdateOptions}; use crate::sparql::{EvaluationError, UpdateOptions};
use crate::storage::io::load_graph; use crate::storage::io::load_graph;
use crate::storage::numeric_encoder::{ use crate::storage::numeric_encoder::{Decoder, EncodedTerm, WriteEncoder};
EncodedQuad, EncodedTerm, EncodedTriple, StrLookup, WriteEncoder,
};
use crate::storage::Storage; use crate::storage::Storage;
use http::header::{ACCEPT, CONTENT_TYPE, USER_AGENT}; use http::header::{ACCEPT, CONTENT_TYPE, USER_AGENT};
use http::{Method, Request, StatusCode}; use http::{Method, Request, StatusCode};
@ -94,7 +95,8 @@ impl<'a> SimpleUpdateEvaluator<'a> {
fn eval_insert_data(&mut self, data: &[Quad]) -> Result<(), EvaluationError> { fn eval_insert_data(&mut self, data: &[Quad]) -> Result<(), EvaluationError> {
let mut bnodes = HashMap::new(); let mut bnodes = HashMap::new();
for quad in data { for quad in data {
let quad = self.encode_quad_for_insertion(quad, &mut bnodes)?; let quad = self.convert_quad(quad, &mut bnodes);
let quad = self.storage.encode_quad(quad.as_ref())?;
self.storage.insert(&quad)?; self.storage.insert(&quad)?;
} }
Ok(()) Ok(())
@ -102,8 +104,8 @@ impl<'a> SimpleUpdateEvaluator<'a> {
fn eval_delete_data(&mut self, data: &[GroundQuad]) -> Result<(), EvaluationError> { fn eval_delete_data(&mut self, data: &[GroundQuad]) -> Result<(), EvaluationError> {
for quad in data { for quad in data {
let quad = self.encode_quad_for_deletion(quad); let quad = self.convert_ground_quad(quad);
self.storage.remove(&quad)?; self.storage.remove(&quad.as_ref().into())?;
} }
Ok(()) Ok(())
} }
@ -124,38 +126,19 @@ impl<'a> SimpleUpdateEvaluator<'a> {
); );
let mut bnodes = HashMap::new(); let mut bnodes = HashMap::new();
for tuple in evaluator.eval_plan(&plan, EncodedTuple::with_capacity(variables.len())) { for tuple in evaluator.eval_plan(&plan, EncodedTuple::with_capacity(variables.len())) {
// We map the tuple to only get store strings let tuple = tuple?;
let tuple = tuple?
.into_iter()
.map(|t| {
Ok(if let Some(t) = t {
let r: Result<_, EvaluationError> = t.on_each_id(&mut |id| {
self.storage.insert_str(
id,
&dataset.get_str(id)?.ok_or_else(|| {
EvaluationError::msg("String not stored in the string store")
})?,
)?;
Ok(())
});
r?;
Some(t)
} else {
None
})
})
.collect::<Result<Vec<_>, EvaluationError>>()?;
for quad in delete { for quad in delete {
if let Some(quad) = self.encode_quad_pattern_for_deletion(quad, &variables, &tuple) if let Some(quad) =
self.convert_ground_quad_pattern(quad, &variables, &tuple, &dataset)?
{ {
self.storage.remove(&quad)?; self.storage.remove(&quad.as_ref().into())?;
} }
} }
for quad in insert { for quad in insert {
if let Some(quad) = if let Some(quad) =
self.encode_quad_pattern_for_insertion(quad, &variables, &tuple, &mut bnodes)? self.convert_quad_pattern(quad, &variables, &tuple, &dataset, &mut bnodes)?
{ {
let quad = self.storage.encode_quad(quad.as_ref())?;
self.storage.insert(&quad)?; self.storage.insert(&quad)?;
} }
} }
@ -212,15 +195,17 @@ impl<'a> SimpleUpdateEvaluator<'a> {
Ok(()) Ok(())
} }
fn eval_create(&mut self, graph: &NamedNode, silent: bool) -> Result<(), EvaluationError> { fn eval_create(&mut self, graph_name: &NamedNode, silent: bool) -> Result<(), EvaluationError> {
let encoded_graph_name = self.encode_named_node_for_insertion(graph)?; let encoded_graph_name = self
.storage
.encode_named_node(NamedNodeRef::new_unchecked(&graph_name.iri))?;
if self.storage.contains_named_graph(&encoded_graph_name)? { if self.storage.contains_named_graph(&encoded_graph_name)? {
if silent { if silent {
Ok(()) Ok(())
} else { } else {
Err(EvaluationError::msg(format!( Err(EvaluationError::msg(format!(
"The graph {} already exists", "The graph {} already exists",
graph graph_name
))) )))
} }
} else { } else {
@ -232,9 +217,11 @@ impl<'a> SimpleUpdateEvaluator<'a> {
fn eval_clear(&mut self, graph: &GraphTarget, silent: bool) -> Result<(), EvaluationError> { fn eval_clear(&mut self, graph: &GraphTarget, silent: bool) -> Result<(), EvaluationError> {
match graph { match graph {
GraphTarget::NamedNode(graph_name) => { GraphTarget::NamedNode(graph_name) => {
let graph_name = self.encode_named_node_for_deletion(graph_name); let encoded_graph_name = self
if self.storage.contains_named_graph(&graph_name)? { .storage
Ok(self.storage.clear_graph(&graph_name)?) .encode_named_node(NamedNodeRef::new_unchecked(&graph_name.iri))?;
if self.storage.contains_named_graph(&encoded_graph_name)? {
Ok(self.storage.clear_graph(&encoded_graph_name)?)
} else if silent { } else if silent {
Ok(()) Ok(())
} else { } else {
@ -267,9 +254,11 @@ impl<'a> SimpleUpdateEvaluator<'a> {
fn eval_drop(&mut self, graph: &GraphTarget, silent: bool) -> Result<(), EvaluationError> { fn eval_drop(&mut self, graph: &GraphTarget, silent: bool) -> Result<(), EvaluationError> {
match graph { match graph {
GraphTarget::NamedNode(graph_name) => { GraphTarget::NamedNode(graph_name) => {
let graph_name = self.encode_named_node_for_deletion(graph_name); let encoded_graph_name = self
if self.storage.contains_named_graph(&graph_name)? { .storage
self.storage.remove_named_graph(&graph_name)?; .encode_named_node(NamedNodeRef::new_unchecked(&graph_name.iri))?;
if self.storage.contains_named_graph(&encoded_graph_name)? {
self.storage.remove_named_graph(&encoded_graph_name)?;
Ok(()) Ok(())
} else if silent { } else if silent {
Ok(()) Ok(())
@ -294,80 +283,143 @@ impl<'a> SimpleUpdateEvaluator<'a> {
} }
} }
fn encode_quad_for_insertion( fn convert_quad(&self, quad: &Quad, bnodes: &mut HashMap<BlankNode, OxBlankNode>) -> OxQuad {
&mut self, OxQuad {
quad: &Quad, subject: match &quad.subject {
Subject::NamedNode(subject) => self.convert_named_node(subject).into(),
Subject::BlankNode(subject) => self.convert_blank_node(subject, bnodes).into(),
Subject::Triple(subject) => self.convert_triple(subject, bnodes).into(),
},
predicate: self.convert_named_node(&quad.predicate),
object: match &quad.object {
Term::NamedNode(object) => self.convert_named_node(object).into(),
Term::BlankNode(object) => self.convert_blank_node(object, bnodes).into(),
Term::Literal(object) => self.convert_literal(object).into(),
Term::Triple(subject) => self.convert_triple(subject, bnodes).into(),
},
graph_name: match &quad.graph_name {
GraphName::NamedNode(graph_name) => self.convert_named_node(graph_name).into(),
GraphName::DefaultGraph => OxGraphName::DefaultGraph,
},
}
}
fn convert_triple(
&self,
triple: &Triple,
bnodes: &mut HashMap<BlankNode, OxBlankNode>,
) -> OxTriple {
OxTriple {
subject: match &triple.subject {
Subject::NamedNode(subject) => self.convert_named_node(subject).into(),
Subject::BlankNode(subject) => self.convert_blank_node(subject, bnodes).into(),
Subject::Triple(subject) => self.convert_triple(subject, bnodes).into(),
},
predicate: self.convert_named_node(&triple.predicate),
object: match &triple.object {
Term::NamedNode(object) => self.convert_named_node(object).into(),
Term::BlankNode(object) => self.convert_blank_node(object, bnodes).into(),
Term::Literal(object) => self.convert_literal(object).into(),
Term::Triple(subject) => self.convert_triple(subject, bnodes).into(),
},
}
}
fn convert_named_node(&self, node: &NamedNode) -> OxNamedNode {
OxNamedNode::new_unchecked(&node.iri)
}
fn convert_blank_node(
&self,
node: &BlankNode,
bnodes: &mut HashMap<BlankNode, OxBlankNode>, bnodes: &mut HashMap<BlankNode, OxBlankNode>,
) -> Result<EncodedQuad, EvaluationError> { ) -> OxBlankNode {
Ok(EncodedQuad { bnodes.entry(node.clone()).or_default().clone()
}
fn convert_literal(&self, literal: &Literal) -> OxLiteral {
match literal {
Literal::Simple { value } => OxLiteral::new_simple_literal(value),
Literal::LanguageTaggedString { value, language } => {
OxLiteral::new_language_tagged_literal_unchecked(value, language)
}
Literal::Typed { value, datatype } => {
OxLiteral::new_typed_literal(value, NamedNodeRef::new_unchecked(&datatype.iri))
}
}
}
fn convert_ground_quad(&self, quad: &GroundQuad) -> OxQuad {
OxQuad {
subject: match &quad.subject { subject: match &quad.subject {
Subject::NamedNode(subject) => self.encode_named_node_for_insertion(subject)?, GroundSubject::NamedNode(subject) => self.convert_named_node(subject).into(),
Subject::BlankNode(subject) => self GroundSubject::Triple(subject) => self.convert_ground_triple(subject).into(),
.storage
.encode_blank_node(bnodes.entry(subject.clone()).or_default().as_ref())?,
Subject::Triple(subject) => {
self.encode_triple_for_insertion(subject, bnodes)?.into()
}
}, },
predicate: self predicate: self.convert_named_node(&quad.predicate),
.storage
.encode_named_node(NamedNodeRef::new_unchecked(&quad.predicate.iri))?,
object: match &quad.object { object: match &quad.object {
Term::NamedNode(object) => self.encode_named_node_for_insertion(object)?, GroundTerm::NamedNode(object) => self.convert_named_node(object).into(),
Term::BlankNode(object) => self GroundTerm::Literal(object) => self.convert_literal(object).into(),
.storage GroundTerm::Triple(subject) => self.convert_ground_triple(subject).into(),
.encode_blank_node(bnodes.entry(object.clone()).or_default().as_ref())?,
Term::Literal(object) => self.encode_literal_for_insertion(object)?,
Term::Triple(subject) => self.encode_triple_for_insertion(subject, bnodes)?.into(),
}, },
graph_name: match &quad.graph_name { graph_name: match &quad.graph_name {
GraphName::NamedNode(graph_name) => { GraphName::NamedNode(graph_name) => self.convert_named_node(graph_name).into(),
self.encode_named_node_for_insertion(graph_name)? GraphName::DefaultGraph => OxGraphName::DefaultGraph,
}
GraphName::DefaultGraph => EncodedTerm::DefaultGraph,
}, },
}) }
} }
fn encode_quad_pattern_for_insertion( fn convert_ground_triple(&self, triple: &GroundTriple) -> OxTriple {
&mut self, OxTriple {
subject: match &triple.subject {
GroundSubject::NamedNode(subject) => self.convert_named_node(subject).into(),
GroundSubject::Triple(subject) => self.convert_ground_triple(subject).into(),
},
predicate: self.convert_named_node(&triple.predicate),
object: match &triple.object {
GroundTerm::NamedNode(object) => self.convert_named_node(object).into(),
GroundTerm::Literal(object) => self.convert_literal(object).into(),
GroundTerm::Triple(subject) => self.convert_ground_triple(subject).into(),
},
}
}
fn convert_quad_pattern(
&self,
quad: &QuadPattern, quad: &QuadPattern,
variables: &[Variable], variables: &[Variable],
values: &[Option<EncodedTerm>], values: &EncodedTuple,
dataset: &DatasetView,
bnodes: &mut HashMap<BlankNode, OxBlankNode>, bnodes: &mut HashMap<BlankNode, OxBlankNode>,
) -> Result<Option<EncodedQuad>, EvaluationError> { ) -> Result<Option<OxQuad>, EvaluationError> {
Ok(Some(EncodedQuad { Ok(Some(OxQuad {
subject: if let Some(subject) = self.encode_term_or_var_for_insertion( subject: match self.convert_term_or_var(
&quad.subject, &quad.subject,
variables, variables,
values, values,
dataset,
bnodes, bnodes,
|t| t.is_named_node() || t.is_blank_node(),
)? { )? {
subject Some(OxTerm::NamedNode(node)) => node.into(),
} else { Some(OxTerm::BlankNode(node)) => node.into(),
return Ok(None); Some(OxTerm::Triple(triple)) => triple.into(),
Some(OxTerm::Literal(_)) | None => return Ok(None),
}, },
predicate: if let Some(predicate) = predicate: if let Some(predicate) =
self.encode_named_node_or_var_for_insertion(&quad.predicate, variables, values)? self.convert_named_node_or_var(&quad.predicate, variables, values, dataset)?
{ {
predicate predicate
} else { } else {
return Ok(None); return Ok(None);
}, },
object: if let Some(object) = self.encode_term_or_var_for_insertion( object: if let Some(object) =
&quad.object, self.convert_term_or_var(&quad.object, variables, values, dataset, bnodes)?
variables, {
values,
bnodes,
|t| !t.is_default_graph(),
)? {
object object
} else { } else {
return Ok(None); return Ok(None);
}, },
graph_name: if let Some(graph_name) = graph_name: if let Some(graph_name) =
self.encode_graph_name_or_var_for_insertion(&quad.graph_name, variables, values)? self.convert_graph_name_or_var(&quad.graph_name, variables, values, dataset)?
{ {
graph_name graph_name
} else { } else {
@ -376,90 +428,98 @@ impl<'a> SimpleUpdateEvaluator<'a> {
})) }))
} }
fn encode_term_or_var_for_insertion( fn convert_term_or_var(
&mut self, &self,
term: &TermPattern, term: &TermPattern,
variables: &[Variable], variables: &[Variable],
values: &[Option<EncodedTerm>], values: &EncodedTuple,
dataset: &DatasetView,
bnodes: &mut HashMap<BlankNode, OxBlankNode>, bnodes: &mut HashMap<BlankNode, OxBlankNode>,
validate: impl FnOnce(&EncodedTerm) -> bool, ) -> Result<Option<OxTerm>, EvaluationError> {
) -> Result<Option<EncodedTerm>, EvaluationError> {
Ok(match term { Ok(match term {
TermPattern::NamedNode(term) => Some(self.encode_named_node_for_insertion(term)?), TermPattern::NamedNode(term) => Some(self.convert_named_node(term).into()),
TermPattern::BlankNode(bnode) => Some( TermPattern::BlankNode(bnode) => Some(self.convert_blank_node(bnode, bnodes).into()),
self.storage TermPattern::Literal(term) => Some(self.convert_literal(term).into()),
.encode_blank_node(bnodes.entry(bnode.clone()).or_default().as_ref())?,
),
TermPattern::Literal(term) => Some(self.encode_literal_for_insertion(term)?),
TermPattern::Triple(triple) => self TermPattern::Triple(triple) => self
.encode_triple_pattern_for_insertion(triple, variables, values, bnodes)? .convert_triple_pattern(triple, variables, values, dataset, bnodes)?
.map(EncodedTerm::from), .map(|t| t.into()),
TermPattern::Variable(v) => self.lookup_variable(v, variables, values).filter(validate), TermPattern::Variable(v) => self
.lookup_variable(v, variables, values)
.map(|node| dataset.decode_term(&node))
.transpose()?,
}) })
} }
fn encode_named_node_or_var_for_insertion( fn convert_named_node_or_var(
&mut self, &self,
term: &NamedNodePattern, term: &NamedNodePattern,
variables: &[Variable], variables: &[Variable],
values: &[Option<EncodedTerm>], values: &EncodedTuple,
) -> Result<Option<EncodedTerm>, EvaluationError> { dataset: &DatasetView,
) -> Result<Option<OxNamedNode>, EvaluationError> {
Ok(match term { Ok(match term {
NamedNodePattern::NamedNode(term) => Some(self.encode_named_node_for_insertion(term)?), NamedNodePattern::NamedNode(term) => Some(self.convert_named_node(term)),
NamedNodePattern::Variable(v) => self NamedNodePattern::Variable(v) => self
.lookup_variable(v, variables, values) .lookup_variable(v, variables, values)
.filter(|value| value.is_named_node()), .map(|node| dataset.decode_named_node(&node))
.transpose()?,
}) })
} }
fn encode_graph_name_or_var_for_insertion( fn convert_graph_name_or_var(
&mut self, &self,
term: &GraphNamePattern, term: &GraphNamePattern,
variables: &[Variable], variables: &[Variable],
values: &[Option<EncodedTerm>], values: &EncodedTuple,
) -> Result<Option<EncodedTerm>, EvaluationError> { dataset: &DatasetView,
Ok(match term { ) -> Result<Option<OxGraphName>, EvaluationError> {
GraphNamePattern::NamedNode(term) => Some(self.encode_named_node_for_insertion(term)?), match term {
GraphNamePattern::DefaultGraph => Some(EncodedTerm::DefaultGraph), GraphNamePattern::NamedNode(term) => Ok(Some(self.convert_named_node(term).into())),
GraphNamePattern::DefaultGraph => Ok(Some(OxGraphName::DefaultGraph)),
GraphNamePattern::Variable(v) => self GraphNamePattern::Variable(v) => self
.lookup_variable(v, variables, values) .lookup_variable(v, variables, values)
.filter(|value| value.is_named_node()), .map(|node| {
}) Ok(if node == EncodedTerm::DefaultGraph {
OxGraphName::DefaultGraph
} else {
dataset.decode_named_node(&node)?.into()
})
})
.transpose(),
}
} }
fn encode_triple_pattern_for_insertion( fn convert_triple_pattern(
&mut self, &self,
triple: &TriplePattern, triple: &TriplePattern,
variables: &[Variable], variables: &[Variable],
values: &[Option<EncodedTerm>], values: &EncodedTuple,
dataset: &DatasetView,
bnodes: &mut HashMap<BlankNode, OxBlankNode>, bnodes: &mut HashMap<BlankNode, OxBlankNode>,
) -> Result<Option<EncodedTriple>, EvaluationError> { ) -> Result<Option<OxTriple>, EvaluationError> {
Ok(Some(EncodedTriple { Ok(Some(OxTriple {
subject: if let Some(subject) = self.encode_term_or_var_for_insertion( subject: match self.convert_term_or_var(
&triple.subject, &triple.subject,
variables, variables,
values, values,
dataset,
bnodes, bnodes,
|t| t.is_named_node() || t.is_blank_node(),
)? { )? {
subject Some(OxTerm::NamedNode(node)) => node.into(),
} else { Some(OxTerm::BlankNode(node)) => node.into(),
return Ok(None); Some(OxTerm::Triple(triple)) => triple.into(),
Some(OxTerm::Literal(_)) | None => return Ok(None),
}, },
predicate: if let Some(predicate) = predicate: if let Some(predicate) =
self.encode_named_node_or_var_for_insertion(&triple.predicate, variables, values)? self.convert_named_node_or_var(&triple.predicate, variables, values, dataset)?
{ {
predicate predicate
} else { } else {
return Ok(None); return Ok(None);
}, },
object: if let Some(object) = self.encode_term_or_var_for_insertion( object: if let Some(object) =
&triple.object, self.convert_term_or_var(&triple.object, variables, values, dataset, bnodes)?
variables, {
values,
bnodes,
|t| !t.is_default_graph(),
)? {
object object
} else { } else {
return Ok(None); return Ok(None);
@ -467,212 +527,115 @@ impl<'a> SimpleUpdateEvaluator<'a> {
})) }))
} }
fn lookup_variable( fn convert_ground_quad_pattern(
&self, &self,
v: &Variable, quad: &GroundQuadPattern,
variables: &[Variable], variables: &[Variable],
values: &[Option<EncodedTerm>], values: &EncodedTuple,
) -> Option<EncodedTerm> { dataset: &DatasetView,
{ ) -> Result<Option<OxQuad>, EvaluationError> {
if let Some(Some(term)) = variables Ok(Some(OxQuad {
.iter() subject: match self.convert_ground_term_or_var(
.position(|v2| v == v2) &quad.subject,
.and_then(|i| values.get(i)) variables,
values,
dataset,
)? {
Some(OxTerm::NamedNode(node)) => node.into(),
Some(OxTerm::BlankNode(node)) => node.into(),
Some(OxTerm::Triple(triple)) => triple.into(),
Some(OxTerm::Literal(_)) | None => return Ok(None),
},
predicate: if let Some(predicate) =
self.convert_named_node_or_var(&quad.predicate, variables, values, dataset)?
{ {
Some(term.clone()) predicate
} else { } else {
None return Ok(None);
}
}
}
fn encode_named_node_for_insertion(
&mut self,
term: &NamedNode,
) -> Result<EncodedTerm, EvaluationError> {
Ok(self
.storage
.encode_named_node(NamedNodeRef::new_unchecked(&term.iri))?)
}
fn encode_triple_for_insertion(
&mut self,
triple: &Triple,
bnodes: &mut HashMap<BlankNode, OxBlankNode>,
) -> Result<EncodedTriple, EvaluationError> {
Ok(EncodedTriple {
subject: match &triple.subject {
Subject::NamedNode(subject) => self.encode_named_node_for_insertion(subject)?,
Subject::BlankNode(subject) => self
.storage
.encode_blank_node(bnodes.entry(subject.clone()).or_default().as_ref())?,
Subject::Triple(subject) => {
self.encode_triple_for_insertion(subject, bnodes)?.into()
}
},
predicate: self
.storage
.encode_named_node(NamedNodeRef::new_unchecked(&triple.predicate.iri))?,
object: match &triple.object {
Term::NamedNode(object) => self.encode_named_node_for_insertion(object)?,
Term::BlankNode(object) => self
.storage
.encode_blank_node(bnodes.entry(object.clone()).or_default().as_ref())?,
Term::Literal(object) => self.encode_literal_for_insertion(object)?,
Term::Triple(object) => self.encode_triple_for_insertion(object, bnodes)?.into(),
},
})
}
fn encode_literal_for_insertion(
&mut self,
term: &Literal,
) -> Result<EncodedTerm, EvaluationError> {
Ok(self.storage.encode_literal(match term {
Literal::Simple { value } => LiteralRef::new_simple_literal(value),
Literal::LanguageTaggedString { value, language } => {
LiteralRef::new_language_tagged_literal_unchecked(value, language)
}
Literal::Typed { value, datatype } => {
LiteralRef::new_typed_literal(value, NamedNodeRef::new_unchecked(&datatype.iri))
}
})?)
}
fn encode_quad_for_deletion(&mut self, quad: &GroundQuad) -> EncodedQuad {
EncodedQuad {
subject: match &quad.subject {
GroundSubject::NamedNode(subject) => self.encode_named_node_for_deletion(subject),
GroundSubject::Triple(subject) => self.encode_triple_for_deletion(subject),
}, },
predicate: self.encode_named_node_for_deletion(&quad.predicate), object: if let Some(object) =
object: match &quad.object { self.convert_ground_term_or_var(&quad.object, variables, values, dataset)?
GroundTerm::NamedNode(object) => self.encode_named_node_for_deletion(object), {
GroundTerm::Literal(object) => self.encode_literal_for_deletion(object), object
GroundTerm::Triple(object) => self.encode_triple_for_deletion(object), } else {
return Ok(None);
}, },
graph_name: match &quad.graph_name { graph_name: if let Some(graph_name) =
GraphName::NamedNode(graph_name) => self.encode_named_node_for_deletion(graph_name), self.convert_graph_name_or_var(&quad.graph_name, variables, values, dataset)?
GraphName::DefaultGraph => EncodedTerm::DefaultGraph, {
graph_name
} else {
return Ok(None);
}, },
} }))
}
fn encode_quad_pattern_for_deletion(
&self,
quad: &GroundQuadPattern,
variables: &[Variable],
values: &[Option<EncodedTerm>],
) -> Option<EncodedQuad> {
Some(EncodedQuad {
subject: self.encode_term_or_var_for_deletion(&quad.subject, variables, values)?,
predicate: self.encode_named_node_or_var_for_deletion(
&quad.predicate,
variables,
values,
)?,
object: self.encode_term_or_var_for_deletion(&quad.object, variables, values)?,
graph_name: self.encode_graph_name_or_var_for_deletion(
&quad.graph_name,
variables,
values,
)?,
})
} }
fn encode_term_or_var_for_deletion( fn convert_ground_term_or_var(
&self, &self,
term: &GroundTermPattern, term: &GroundTermPattern,
variables: &[Variable], variables: &[Variable],
values: &[Option<EncodedTerm>], values: &EncodedTuple,
) -> Option<EncodedTerm> { dataset: &DatasetView,
match term { ) -> Result<Option<OxTerm>, EvaluationError> {
GroundTermPattern::NamedNode(term) => Some(self.encode_named_node_for_deletion(term)), Ok(match term {
GroundTermPattern::Literal(term) => Some(self.encode_literal_for_deletion(term)), GroundTermPattern::NamedNode(term) => Some(self.convert_named_node(term).into()),
GroundTermPattern::Variable(v) => self.lookup_variable(v, variables, values), GroundTermPattern::Literal(term) => Some(self.convert_literal(term).into()),
GroundTermPattern::Triple(triple) => Some( GroundTermPattern::Triple(triple) => self
self.encode_triple_pattern_for_deletion(triple, variables, values)? .convert_ground_triple_pattern(triple, variables, values, dataset)?
.into(), .map(|t| t.into()),
), GroundTermPattern::Variable(v) => self
}
}
fn encode_named_node_or_var_for_deletion(
&self,
term: &NamedNodePattern,
variables: &[Variable],
values: &[Option<EncodedTerm>],
) -> Option<EncodedTerm> {
match term {
NamedNodePattern::NamedNode(term) => Some(self.encode_named_node_for_deletion(term)),
NamedNodePattern::Variable(v) => self
.lookup_variable(v, variables, values)
.filter(|v| v.is_named_node()),
}
}
fn encode_graph_name_or_var_for_deletion(
&self,
graph_name: &GraphNamePattern,
variables: &[Variable],
values: &[Option<EncodedTerm>],
) -> Option<EncodedTerm> {
match graph_name {
GraphNamePattern::NamedNode(term) => Some(self.encode_named_node_for_deletion(term)),
GraphNamePattern::DefaultGraph => Some(EncodedTerm::DefaultGraph),
GraphNamePattern::Variable(v) => self
.lookup_variable(v, variables, values) .lookup_variable(v, variables, values)
.filter(|v| v.is_named_node()), .map(|node| dataset.decode_term(&node))
} .transpose()?,
})
} }
fn encode_triple_pattern_for_deletion( fn convert_ground_triple_pattern(
&self, &self,
triple: &GroundTriplePattern, triple: &GroundTriplePattern,
variables: &[Variable], variables: &[Variable],
values: &[Option<EncodedTerm>], values: &EncodedTuple,
) -> Option<EncodedTriple> { dataset: &DatasetView,
Some(EncodedTriple { ) -> Result<Option<OxTriple>, EvaluationError> {
subject: self.encode_term_or_var_for_deletion(&triple.subject, variables, values)?, Ok(Some(OxTriple {
predicate: self.encode_named_node_or_var_for_deletion( subject: match self.convert_ground_term_or_var(
&triple.predicate, &triple.subject,
variables, variables,
values, values,
)?, dataset,
object: self.encode_term_or_var_for_deletion(&triple.object, variables, values)?, )? {
}) Some(OxTerm::NamedNode(node)) => node.into(),
} Some(OxTerm::BlankNode(node)) => node.into(),
Some(OxTerm::Triple(triple)) => triple.into(),
fn encode_named_node_for_deletion(&self, term: &NamedNode) -> EncodedTerm { Some(OxTerm::Literal(_)) | None => return Ok(None),
NamedNodeRef::new_unchecked(&term.iri).into()
}
fn encode_literal_for_deletion(&self, term: &Literal) -> EncodedTerm {
match term {
Literal::Simple { value } => LiteralRef::new_simple_literal(value),
Literal::LanguageTaggedString { value, language } => {
LiteralRef::new_language_tagged_literal_unchecked(value, language)
}
Literal::Typed { value, datatype } => {
LiteralRef::new_typed_literal(value, NamedNodeRef::new_unchecked(&datatype.iri))
}
}
.into()
}
fn encode_triple_for_deletion(&self, triple: &GroundTriple) -> EncodedTerm {
EncodedTriple::new(
match &triple.subject {
GroundSubject::NamedNode(subject) => self.encode_named_node_for_deletion(subject),
GroundSubject::Triple(subject) => self.encode_triple_for_deletion(subject),
}, },
self.encode_named_node_for_deletion(&triple.predicate), predicate: if let Some(predicate) =
match &triple.object { self.convert_named_node_or_var(&triple.predicate, variables, values, dataset)?
GroundTerm::NamedNode(object) => self.encode_named_node_for_deletion(object), {
GroundTerm::Literal(object) => self.encode_literal_for_deletion(object), predicate
GroundTerm::Triple(object) => self.encode_triple_for_deletion(object), } else {
return Ok(None);
}, },
) object: if let Some(object) =
.into() self.convert_ground_term_or_var(&triple.object, variables, values, dataset)?
{
object
} else {
return Ok(None);
},
}))
}
fn lookup_variable(
&self,
v: &Variable,
variables: &[Variable],
values: &EncodedTuple,
) -> Option<EncodedTerm> {
variables
.iter()
.position(|v2| v == v2)
.and_then(|i| values.get(i))
.cloned()
} }
} }

@ -340,53 +340,6 @@ impl EncodedTerm {
pub fn is_triple(&self) -> bool { pub fn is_triple(&self) -> bool {
matches!(self, Self::Triple { .. }) matches!(self, Self::Triple { .. })
} }
pub fn on_each_id<E>(
&self,
callback: &mut impl FnMut(&StrHash) -> Result<(), E>,
) -> Result<(), E> {
match self {
Self::NamedNode { iri_id } => {
callback(iri_id)?;
}
Self::BigBlankNode { id_id } => {
callback(id_id)?;
}
Self::BigStringLiteral { value_id } => {
callback(value_id)?;
}
Self::SmallBigLangStringLiteral { language_id, .. } => {
callback(language_id)?;
}
Self::BigSmallLangStringLiteral { value_id, .. } => {
callback(value_id)?;
}
Self::BigBigLangStringLiteral {
value_id,
language_id,
} => {
callback(value_id)?;
callback(language_id)?;
}
Self::SmallTypedLiteral { datatype_id, .. } => {
callback(datatype_id)?;
}
Self::BigTypedLiteral {
value_id,
datatype_id,
} => {
callback(value_id)?;
callback(datatype_id)?;
}
Self::Triple(triple) => {
triple.subject.on_each_id(callback)?;
triple.predicate.on_each_id(callback)?;
triple.object.on_each_id(callback)?;
}
_ => (),
}
Ok(())
}
} }
impl From<bool> for EncodedTerm { impl From<bool> for EncodedTerm {

Loading…
Cancel
Save