use model::*; use std::collections::BTreeMap; use std::collections::BTreeSet; use std::fmt; use std::ops::Add; use utils::Escaper; use uuid::Uuid; use Result; #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum Variable { Variable { name: String }, BlankNode { id: Uuid }, Internal { id: Uuid }, } impl Variable { pub fn new(name: impl Into) -> Self { Variable::Variable { name: name.into() } } pub fn has_name(&self) -> bool { match self { Variable::Variable { .. } => true, _ => false, } } pub fn name(&self) -> Result<&str> { match self { Variable::Variable { name } => Ok(name), _ => Err(format_err!("The variable {} has no name", self)), } } } impl fmt::Display for Variable { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { Variable::Variable { name } => write!(f, "?{}", name), Variable::BlankNode { id } => write!(f, "_:{}", id.to_simple()), Variable::Internal { id } => write!(f, "?{}", id.to_simple()), } } } impl Default for Variable { fn default() -> Self { Variable::Internal { id: Uuid::new_v4() } } } impl From for Variable { fn from(blank_node: BlankNode) -> Self { Variable::BlankNode { id: *blank_node.as_uuid(), } } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum NamedNodeOrVariable { NamedNode(NamedNode), Variable(Variable), } impl fmt::Display for NamedNodeOrVariable { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { NamedNodeOrVariable::NamedNode(node) => write!(f, "{}", node), NamedNodeOrVariable::Variable(var) => write!(f, "{}", var), } } } impl From for NamedNodeOrVariable { fn from(node: NamedNode) -> Self { NamedNodeOrVariable::NamedNode(node) } } impl From for NamedNodeOrVariable { fn from(var: Variable) -> Self { NamedNodeOrVariable::Variable(var) } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum TermOrVariable { Term(Term), Variable(Variable), } impl fmt::Display for TermOrVariable { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { TermOrVariable::Term(term) => write!(f, "{}", term), TermOrVariable::Variable(var) => write!(f, "{}", var), } } } impl From for TermOrVariable { fn from(node: NamedNode) -> Self { TermOrVariable::Term(node.into()) } } impl From for TermOrVariable { fn from(node: BlankNode) -> Self { TermOrVariable::Variable(node.into()) } } impl From for TermOrVariable { fn from(literal: Literal) -> Self { TermOrVariable::Term(literal.into()) } } impl From for TermOrVariable { fn from(var: Variable) -> Self { TermOrVariable::Variable(var) } } impl From for TermOrVariable { fn from(term: Term) -> Self { match term { Term::NamedNode(node) => TermOrVariable::Term(node.into()), Term::BlankNode(node) => TermOrVariable::Variable(node.into()), Term::Literal(literal) => TermOrVariable::Term(literal.into()), } } } impl From for TermOrVariable { fn from(element: NamedNodeOrVariable) -> Self { match element { NamedNodeOrVariable::NamedNode(node) => TermOrVariable::Term(node.into()), NamedNodeOrVariable::Variable(var) => TermOrVariable::Variable(var), } } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub struct StaticBindings { variables: Vec, values: Vec>>, } impl StaticBindings { pub fn new(variables: Vec, values: Vec>>) -> Self { Self { variables, values } } pub fn variables(&self) -> &[Variable] { &*self.variables } pub fn variables_iter(&self) -> impl Iterator { self.variables.iter() } pub fn values_iter(&self) -> impl Iterator>> { self.values.iter() } pub fn into_iterator(self) -> BindingsIterator<'static> { BindingsIterator { variables: self.variables, iter: Box::new(self.values.into_iter().map(Ok)), } } pub fn is_empty(&self) -> bool { self.values.is_empty() } } impl Default for StaticBindings { fn default() -> Self { Self { variables: Vec::default(), values: Vec::default(), } } } pub struct BindingsIterator<'a> { variables: Vec, iter: Box>>> + 'a>, } impl<'a> BindingsIterator<'a> { pub fn new( variables: Vec, iter: Box>>> + 'a>, ) -> Self { Self { variables, iter } } pub fn variables(&self) -> &[Variable] { &*self.variables } pub fn into_values_iter(self) -> Box>>> + 'a> { self.iter } pub fn destruct( self, ) -> ( Vec, Box>>> + 'a>, ) { (self.variables, self.iter) } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub struct TriplePattern { pub subject: TermOrVariable, pub predicate: NamedNodeOrVariable, pub object: TermOrVariable, } impl TriplePattern { pub fn new( subject: impl Into, predicate: impl Into, object: impl Into, ) -> Self { Self { subject: subject.into(), predicate: predicate.into(), object: object.into(), } } } impl fmt::Display for TriplePattern { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "{} {} {}", self.subject, self.predicate, self.object) } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum PropertyPath { PredicatePath(NamedNode), InversePath(Box), SequencePath(Box, Box), AlternativePath(Box, Box), ZeroOrMorePath(Box), OneOrMorePath(Box), ZeroOrOnePath(Box), NegatedPropertySet(Vec), } impl fmt::Display for PropertyPath { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { PropertyPath::PredicatePath(p) => write!(f, "link({})", p), PropertyPath::InversePath(p) => write!(f, "inv({})", p), PropertyPath::AlternativePath(a, b) => write!(f, "alt({}, {})", a, b), PropertyPath::SequencePath(a, b) => write!(f, "seq({}, {})", a, b), PropertyPath::ZeroOrMorePath(p) => write!(f, "ZeroOrMorePath({})", p), PropertyPath::OneOrMorePath(p) => write!(f, "OneOrMorePath({})", p), PropertyPath::ZeroOrOnePath(p) => write!(f, "ZeroOrOnePath({})", p), PropertyPath::NegatedPropertySet(p) => write!( f, "NPS({{ {} }})", p.iter() .map(|v| v.to_string()) .collect::>() .join(" ") ), } } } struct SparqlPropertyPath<'a>(&'a PropertyPath); impl<'a> fmt::Display for SparqlPropertyPath<'a> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self.0 { PropertyPath::PredicatePath(p) => write!(f, "{}", p), PropertyPath::InversePath(p) => write!(f, "^{}", SparqlPropertyPath(&*p)), PropertyPath::SequencePath(a, b) => write!( f, "({} / {})", SparqlPropertyPath(&*a), SparqlPropertyPath(&*b) ), PropertyPath::AlternativePath(a, b) => write!( f, "({} | {})", SparqlPropertyPath(&*a), SparqlPropertyPath(&*b) ), PropertyPath::ZeroOrMorePath(p) => write!(f, "{}*", SparqlPropertyPath(&*p)), PropertyPath::OneOrMorePath(p) => write!(f, "{}+", SparqlPropertyPath(&*p)), PropertyPath::ZeroOrOnePath(p) => write!(f, "{}?", SparqlPropertyPath(&*p)), PropertyPath::NegatedPropertySet(p) => write!( f, "!({})", p.iter() .map(|v| v.to_string()) .collect::>() .join(" | ") ), } } } impl From for PropertyPath { fn from(p: NamedNode) -> Self { PropertyPath::PredicatePath(p) } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub struct PathPattern { pub subject: TermOrVariable, pub path: PropertyPath, pub object: TermOrVariable, } impl fmt::Display for PathPattern { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "Path({} {} {})", self.subject, self.path, self.object) } } impl PathPattern { pub fn new( subject: impl Into, path: impl Into, object: impl Into, ) -> Self { Self { subject: subject.into(), path: path.into(), object: object.into(), } } } struct SparqlPathPattern<'a>(&'a PathPattern); impl<'a> fmt::Display for SparqlPathPattern<'a> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!( f, "{} {} {}", self.0.subject, SparqlPropertyPath(&self.0.path), self.0.object ) } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum TripleOrPathPattern { Triple(TriplePattern), Path(PathPattern), } impl<'a> fmt::Display for TripleOrPathPattern { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { TripleOrPathPattern::Triple(tp) => write!(f, "{}", tp), TripleOrPathPattern::Path(ppp) => write!(f, "{}", ppp), } } } impl From for TripleOrPathPattern { fn from(tp: TriplePattern) -> Self { TripleOrPathPattern::Triple(tp) } } impl From for TripleOrPathPattern { fn from(ppp: PathPattern) -> Self { TripleOrPathPattern::Path(ppp) } } struct SparqlTripleOrPathPattern<'a>(&'a TripleOrPathPattern); impl<'a> fmt::Display for SparqlTripleOrPathPattern<'a> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self.0 { TripleOrPathPattern::Triple(tp) => write!(f, "{}", tp), TripleOrPathPattern::Path(ppp) => write!(f, "{}", SparqlPathPattern(&ppp)), } } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum Expression { Constant(TermOrVariable), Or(Box, Box), And(Box, Box), Equal(Box, Box), NotEqual(Box, Box), Greater(Box, Box), GreaterOrEq(Box, Box), Lower(Box, Box), LowerOrEq(Box, Box), In(Box, Vec), NotIn(Box, Vec), Add(Box, Box), Sub(Box, Box), Mul(Box, Box), Div(Box, Box), UnaryPlus(Box), UnaryMinus(Box), UnaryNot(Box), StrFunctionCall(Box), LangFunctionCall(Box), LangMatchesFunctionCall(Box, Box), DatatypeFunctionCall(Box), BoundFunctionCall(Variable), IRIFunctionCall(Box), BNodeFunctionCall(Option>), RandFunctionCall(), AbsFunctionCall(Box), CeilFunctionCall(Box), FloorFunctionCall(Box), RoundFunctionCall(Box), ConcatFunctionCall(Vec), SubStrFunctionCall(Box, Box, Option>), StrLenFunctionCall(Box), ReplaceFunctionCall( Box, Box, Box, Option>, ), UCaseFunctionCall(Box), LCaseFunctionCall(Box), EncodeForURIFunctionCall(Box), ContainsFunctionCall(Box, Box), StrStartsFunctionCall(Box, Box), StrEndsFunctionCall(Box, Box), StrBeforeFunctionCall(Box, Box), StrAfterFunctionCall(Box, Box), YearFunctionCall(Box), MonthFunctionCall(Box), DayFunctionCall(Box), HoursFunctionCall(Box), MinutesFunctionCall(Box), SecondsFunctionCall(Box), TimezoneFunctionCall(Box), NowFunctionCall(), UUIDFunctionCall(), StrUUIDFunctionCall(), MD5FunctionCall(Box), SHA1FunctionCall(Box), SHA256FunctionCall(Box), SHA384FunctionCall(Box), SHA512FunctionCall(Box), CoalesceFunctionCall(Vec), IfFunctionCall(Box, Box, Box), StrLangFunctionCall(Box, Box), StrDTFunctionCall(Box, Box), SameTermFunctionCall(Box, Box), IsIRIFunctionCall(Box), IsBlankFunctionCall(Box), IsLiteralFunctionCall(Box), IsNumericFunctionCall(Box), RegexFunctionCall(Box, Box, Option>), CustomFunctionCall(NamedNode, Vec), ExistsFunctionCall(Box), } impl fmt::Display for Expression { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { Expression::Constant(t) => write!(f, "{}", t), Expression::Or(a, b) => write!(f, "({} || {})", a, b), Expression::And(a, b) => write!(f, "({} && {})", a, b), Expression::Equal(a, b) => write!(f, "({} = {})", a, b), Expression::NotEqual(a, b) => write!(f, "({} != {})", a, b), Expression::Greater(a, b) => write!(f, "({} > {})", a, b), Expression::GreaterOrEq(a, b) => write!(f, "({} >= {})", a, b), Expression::Lower(a, b) => write!(f, "({} < {})", a, b), Expression::LowerOrEq(a, b) => write!(f, "({} <= {})", a, b), Expression::In(a, b) => write!( f, "({} IN ({}))", a, b.iter() .map(|v| v.to_string()) .collect::>() .join(", ") ), Expression::NotIn(a, b) => write!( f, "({} NOT IN ({}))", a, b.iter() .map(|v| v.to_string()) .collect::>() .join(", ") ), Expression::Add(a, b) => write!(f, "{} + {}", a, b), Expression::Sub(a, b) => write!(f, "{} - {}", a, b), Expression::Mul(a, b) => write!(f, "{} * {}", a, b), Expression::Div(a, b) => write!(f, "{} / {}", a, b), Expression::UnaryPlus(e) => write!(f, "+{}", e), Expression::UnaryMinus(e) => write!(f, "-{}", e), Expression::UnaryNot(e) => write!(f, "!{}", e), Expression::StrFunctionCall(e) => write!(f, "STR({})", e), Expression::LangFunctionCall(e) => write!(f, "LANG({})", e), Expression::LangMatchesFunctionCall(a, b) => write!(f, "LANGMATCHES({}, {})", a, b), Expression::DatatypeFunctionCall(e) => write!(f, "DATATYPE({})", e), Expression::BoundFunctionCall(v) => write!(f, "BOUND({})", v), Expression::IRIFunctionCall(e) => write!(f, "IRI({})", e), Expression::BNodeFunctionCall(v) => v .as_ref() .map(|id| write!(f, "BNODE({})", id)) .unwrap_or_else(|| write!(f, "BNODE()")), Expression::RandFunctionCall() => write!(f, "RAND()"), Expression::AbsFunctionCall(e) => write!(f, "ABS({})", e), Expression::CeilFunctionCall(e) => write!(f, "CEIL({})", e), Expression::FloorFunctionCall(e) => write!(f, "FLOOR({})", e), Expression::RoundFunctionCall(e) => write!(f, "ROUND({})", e), Expression::ConcatFunctionCall(e) => write!( f, "CONCAT({})", e.iter() .map(|v| v.to_string()) .collect::>() .join(", ") ), Expression::SubStrFunctionCall(a, b, c) => c .as_ref() .map(|cv| write!(f, "SUBSTR({}, {}, {})", a, b, cv)) .unwrap_or_else(|| write!(f, "SUBSTR({}, {})", a, b)), Expression::StrLenFunctionCall(e) => write!(f, "STRLEN({})", e), Expression::ReplaceFunctionCall(arg, pattern, replacement, flags) => match flags { Some(flags) => write!( f, "REPLACE({}, {}, {}, {})", arg, pattern, replacement, flags ), None => write!(f, "REPLACE({}, {}, {})", arg, pattern, replacement), }, Expression::UCaseFunctionCall(e) => write!(f, "UCASE({})", e), Expression::LCaseFunctionCall(e) => write!(f, "LCASE({})", e), Expression::EncodeForURIFunctionCall(e) => write!(f, "ENCODE_FOR_URI({})", e), Expression::ContainsFunctionCall(a, b) => write!(f, "CONTAINS({}, {})", a, b), Expression::StrStartsFunctionCall(a, b) => write!(f, "STRSTATS({}, {})", a, b), Expression::StrEndsFunctionCall(a, b) => write!(f, "STRENDS({}, {})", a, b), Expression::StrBeforeFunctionCall(a, b) => write!(f, "STRBEFORE({}, {})", a, b), Expression::StrAfterFunctionCall(a, b) => write!(f, "STRAFTER({}, {})", a, b), Expression::YearFunctionCall(e) => write!(f, "YEAR({})", e), Expression::MonthFunctionCall(e) => write!(f, "MONTH({})", e), Expression::DayFunctionCall(e) => write!(f, "DAY({})", e), Expression::HoursFunctionCall(e) => write!(f, "HOURS({})", e), Expression::MinutesFunctionCall(e) => write!(f, "MINUTES({})", e), Expression::SecondsFunctionCall(e) => write!(f, "SECONDS({})", e), Expression::TimezoneFunctionCall(e) => write!(f, "TIMEZONE({})", e), Expression::NowFunctionCall() => write!(f, "NOW()"), Expression::UUIDFunctionCall() => write!(f, "UUID()"), Expression::StrUUIDFunctionCall() => write!(f, "STRUUID()"), Expression::MD5FunctionCall(e) => write!(f, "MD5({})", e), Expression::SHA1FunctionCall(e) => write!(f, "SHA1({})", e), Expression::SHA256FunctionCall(e) => write!(f, "SHA256({})", e), Expression::SHA384FunctionCall(e) => write!(f, "SHA384({})", e), Expression::SHA512FunctionCall(e) => write!(f, "SHA512({})", e), Expression::CoalesceFunctionCall(e) => write!( f, "COALESCE({})", e.iter() .map(|v| v.to_string()) .collect::>() .join(", ") ), Expression::IfFunctionCall(a, b, c) => write!(f, "IF({}, {}, {})", a, b, c), Expression::StrLangFunctionCall(a, b) => write!(f, "STRLANG({}, {})", a, b), Expression::StrDTFunctionCall(a, b) => write!(f, "STRDT({}, {})", a, b), Expression::SameTermFunctionCall(a, b) => write!(f, "sameTerm({}, {})", a, b), Expression::IsIRIFunctionCall(e) => write!(f, "isIRI({})", e), Expression::IsBlankFunctionCall(e) => write!(f, "isBLANK({})", e), Expression::IsLiteralFunctionCall(e) => write!(f, "isLITERAL({})", e), Expression::IsNumericFunctionCall(e) => write!(f, "isNUMERIC({})", e), Expression::RegexFunctionCall(text, pattern, flags) => match flags { Some(flags) => write!(f, "REGEX({}, {}, {})", text, pattern, flags), None => write!(f, "REGEX({}, {})", text, pattern), }, Expression::CustomFunctionCall(iri, args) => write!( f, "{}({})", iri, args.iter() .map(|v| v.to_string()) .collect::>() .join(", ") ), Expression::ExistsFunctionCall(p) => write!(f, "EXISTS {{ {} }}", p), } } } impl From for Expression { fn from(p: NamedNode) -> Self { Expression::Constant(p.into()) } } impl From for Expression { fn from(p: Literal) -> Self { Expression::Constant(p.into()) } } impl From for Expression { fn from(v: Variable) -> Self { Expression::Constant(v.into()) } } struct SparqlExpression<'a>(&'a Expression); impl<'a> fmt::Display for SparqlExpression<'a> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self.0 { Expression::Constant(t) => write!(f, "{}", t), Expression::Or(a, b) => write!( f, "({} || {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::And(a, b) => write!( f, "({} && {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::Equal(a, b) => { write!(f, "({} = {})", SparqlExpression(&*a), SparqlExpression(&*b)) } Expression::NotEqual(a, b) => write!( f, "({} != {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::Greater(a, b) => { write!(f, "({} > {})", SparqlExpression(&*a), SparqlExpression(&*b)) } Expression::GreaterOrEq(a, b) => write!( f, "({} >= {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::Lower(a, b) => { write!(f, "({} < {})", SparqlExpression(&*a), SparqlExpression(&*b)) } Expression::LowerOrEq(a, b) => write!( f, "({} <= {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::In(a, b) => write!( f, "({} IN ({}))", a, b.iter() .map(|v| SparqlExpression(v).to_string()) .collect::>() .join(", ") ), Expression::NotIn(a, b) => write!( f, "({} NOT IN ({}))", a, b.iter() .map(|v| SparqlExpression(v).to_string()) .collect::>() .join(", ") ), Expression::Add(a, b) => { write!(f, "{} + {}", SparqlExpression(&*a), SparqlExpression(&*b)) } Expression::Sub(a, b) => { write!(f, "{} - {}", SparqlExpression(&*a), SparqlExpression(&*b)) } Expression::Mul(a, b) => { write!(f, "{} * {}", SparqlExpression(&*a), SparqlExpression(&*b)) } Expression::Div(a, b) => { write!(f, "{} / {}", SparqlExpression(&*a), SparqlExpression(&*b)) } Expression::UnaryPlus(e) => write!(f, "+{}", SparqlExpression(&*e)), Expression::UnaryMinus(e) => write!(f, "-{}", SparqlExpression(&*e)), Expression::UnaryNot(e) => match e.as_ref() { Expression::ExistsFunctionCall(p) => { write!(f, "NOT EXISTS {{ {} }}", SparqlGraphPattern(&*p)) } e => write!(f, "!{}", e), }, Expression::StrFunctionCall(e) => write!(f, "STR({})", SparqlExpression(&*e)), Expression::LangFunctionCall(e) => write!(f, "LANG({})", SparqlExpression(&*e)), Expression::LangMatchesFunctionCall(a, b) => write!( f, "LANGMATCHES({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::DatatypeFunctionCall(e) => write!(f, "DATATYPE({})", SparqlExpression(&*e)), Expression::BoundFunctionCall(v) => write!(f, "BOUND({})", v), Expression::IRIFunctionCall(e) => write!(f, "IRI({})", SparqlExpression(&*e)), Expression::BNodeFunctionCall(v) => v .as_ref() .map(|id| write!(f, "BOUND({})", SparqlExpression(&*id))) .unwrap_or_else(|| write!(f, "BOUND()")), Expression::RandFunctionCall() => write!(f, "RAND()"), Expression::AbsFunctionCall(e) => write!(f, "ABS({})", SparqlExpression(&*e)), Expression::CeilFunctionCall(e) => write!(f, "CEIL({})", SparqlExpression(&*e)), Expression::FloorFunctionCall(e) => write!(f, "FLOOR({})", SparqlExpression(&*e)), Expression::RoundFunctionCall(e) => write!(f, "ROUND({})", SparqlExpression(&*e)), Expression::ConcatFunctionCall(e) => write!( f, "CONCAT({})", e.iter() .map(|v| SparqlExpression(v).to_string()) .collect::>() .join(", ") ), Expression::SubStrFunctionCall(a, b, c) => c .as_ref() .map(|cv| { write!( f, "SUBSTR({}, {}, {})", SparqlExpression(&*a), SparqlExpression(&*b), SparqlExpression(cv) ) }).unwrap_or_else(|| { write!( f, "SUBSTR({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ) }), Expression::StrLenFunctionCall(e) => write!(f, "STRLEN({})", SparqlExpression(&*e)), Expression::ReplaceFunctionCall(arg, pattern, replacement, flags) => match flags { Some(flags) => write!( f, "REPLACE({}, {}, {}, {})", SparqlExpression(&*arg), SparqlExpression(&*pattern), SparqlExpression(&*replacement), flags ), None => write!( f, "REPLACE({}, {}, {})", SparqlExpression(&*arg), SparqlExpression(&*pattern), SparqlExpression(&*replacement) ), }, Expression::UCaseFunctionCall(e) => write!(f, "UCASE({})", SparqlExpression(&*e)), Expression::LCaseFunctionCall(e) => write!(f, "LCASE({})", SparqlExpression(&*e)), Expression::EncodeForURIFunctionCall(e) => { write!(f, "ENCODE_FOR_URI({})", SparqlExpression(&*e)) } Expression::ContainsFunctionCall(a, b) => write!( f, "CONTAINS({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::StrStartsFunctionCall(a, b) => write!( f, "STRSTATS({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::StrEndsFunctionCall(a, b) => write!( f, "STRENDS({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::StrBeforeFunctionCall(a, b) => write!( f, "STRBEFORE({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::StrAfterFunctionCall(a, b) => write!( f, "STRAFTER({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::YearFunctionCall(e) => write!(f, "YEAR({})", SparqlExpression(&*e)), Expression::MonthFunctionCall(e) => write!(f, "MONTH({})", SparqlExpression(&*e)), Expression::DayFunctionCall(e) => write!(f, "DAY({})", SparqlExpression(&*e)), Expression::HoursFunctionCall(e) => write!(f, "HOURS({})", SparqlExpression(&*e)), Expression::MinutesFunctionCall(e) => write!(f, "MINUTES({})", SparqlExpression(&*e)), Expression::SecondsFunctionCall(e) => write!(f, "SECONDS({})", SparqlExpression(&*e)), Expression::TimezoneFunctionCall(e) => write!(f, "TIMEZONE({})", SparqlExpression(&*e)), Expression::NowFunctionCall() => write!(f, "NOW()"), Expression::UUIDFunctionCall() => write!(f, "UUID()"), Expression::StrUUIDFunctionCall() => write!(f, "STRUUID()"), Expression::MD5FunctionCall(e) => write!(f, "MD5({})", SparqlExpression(&*e)), Expression::SHA1FunctionCall(e) => write!(f, "SHA1({})", SparqlExpression(&*e)), Expression::SHA256FunctionCall(e) => write!(f, "SHA256({})", SparqlExpression(&*e)), Expression::SHA384FunctionCall(e) => write!(f, "SHA384({})", SparqlExpression(&*e)), Expression::SHA512FunctionCall(e) => write!(f, "SHA512({})", SparqlExpression(&*e)), Expression::CoalesceFunctionCall(e) => write!( f, "COALESCE({})", e.iter() .map(|v| SparqlExpression(&*v).to_string()) .collect::>() .join(", ") ), Expression::IfFunctionCall(a, b, c) => write!( f, "IF({}, {}, {})", SparqlExpression(&*a), SparqlExpression(&*b), SparqlExpression(&*c) ), Expression::StrLangFunctionCall(a, b) => write!( f, "STRLANG({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::StrDTFunctionCall(a, b) => write!( f, "STRDT({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::SameTermFunctionCall(a, b) => write!( f, "sameTerm({}, {})", SparqlExpression(&*a), SparqlExpression(&*b) ), Expression::IsIRIFunctionCall(e) => write!(f, "isIRI({})", SparqlExpression(&*e)), Expression::IsBlankFunctionCall(e) => write!(f, "isBLANK({})", SparqlExpression(&*e)), Expression::IsLiteralFunctionCall(e) => { write!(f, "isLITERAL({})", SparqlExpression(&*e)) } Expression::IsNumericFunctionCall(e) => { write!(f, "isNUMERIC({})", SparqlExpression(&*e)) } Expression::RegexFunctionCall(text, pattern, flags) => match flags { Some(flags) => write!( f, "REGEX({}, {}, {})", SparqlExpression(&*text), SparqlExpression(&*pattern), flags ), None => write!( f, "REGEX({}, {})", SparqlExpression(&*text), SparqlExpression(&*pattern) ), }, Expression::CustomFunctionCall(iri, args) => write!( f, "{}({})", iri, args.iter() .map(|v| SparqlExpression(v).to_string()) .collect::>() .join(", ") ), Expression::ExistsFunctionCall(p) => { write!(f, "EXISTS {{ {} }}", SparqlGraphPattern(&*p)) } } } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum GraphPattern { BGP(Vec), Join(Box, Box), LeftJoin(Box, Box, Expression), Filter(Expression, Box), Union(Box, Box), Graph(NamedNodeOrVariable, Box), Extend(Box, Variable, Expression), Minus(Box, Box), Service(NamedNodeOrVariable, Box, bool), AggregateJoin(GroupPattern, BTreeMap), Data(StaticBindings), OrderBy(Box, Vec), Project(Box, Vec), Distinct(Box), Reduced(Box), Slice(Box, usize, Option), } impl fmt::Display for GraphPattern { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { GraphPattern::BGP(p) => write!( f, "BGP({})", p.iter() .map(|v| v.to_string()) .collect::>() .join(" . ") ), GraphPattern::Join(a, b) => write!(f, "Join({}, {})", a, b), GraphPattern::LeftJoin(a, b, e) => write!(f, "LeftJoin({}, {}, {})", a, b, e), GraphPattern::Filter(e, p) => write!(f, "Filter({}, {})", e, p), GraphPattern::Union(a, b) => write!(f, "Union({}, {})", a, b), GraphPattern::Graph(g, p) => write!(f, "Graph({}, {})", g, p), GraphPattern::Extend(p, v, e) => write!(f, "Extend({}), {}, {})", p, v, e), GraphPattern::Minus(a, b) => write!(f, "Minus({}, {})", a, b), GraphPattern::Service(n, p, s) => write!(f, "Service({}, {}, {})", n, p, s), GraphPattern::AggregateJoin(g, a) => write!( f, "AggregateJoin({}, {})", g, a.iter() .map(|(a, v)| format!("{}: {}", v, a)) .collect::>() .join(", ") ), GraphPattern::Data(bs) => { let variables = bs.variables(); write!(f, "{{ ")?; for values in bs.values_iter() { write!(f, "{{")?; for i in 0..values.len() { if let Some(ref val) = values[i] { write!(f, " {} → {} ", variables[i], val)?; } } write!(f, "}}")?; } write!(f, "}}") } GraphPattern::OrderBy(l, o) => write!( f, "OrderBy({}, ({}))", l, o.iter() .map(|c| c.to_string()) .collect::>() .join(", ") ), GraphPattern::Project(l, pv) => write!( f, "Project({}, ({}))", l, pv.iter() .map(|v| v.to_string()) .collect::>() .join(", ") ), GraphPattern::Distinct(l) => write!(f, "Distinct({})", l), GraphPattern::Reduced(l) => write!(f, "Reduce({})", l), GraphPattern::Slice(l, start, length) => write!( f, "Slice({}, {}, {})", l, start, length .map(|l| l.to_string()) .unwrap_or_else(|| '?'.to_string()) ), } } } impl Default for GraphPattern { fn default() -> Self { GraphPattern::BGP(Vec::default()) } } impl From for GraphPattern { fn from(p: TripleOrPathPattern) -> Self { GraphPattern::BGP(vec![p]) } } impl GraphPattern { pub fn visible_variables(&self) -> BTreeSet<&Variable> { let mut vars = BTreeSet::default(); self.add_visible_variables(&mut vars); vars } fn add_visible_variables<'a>(&'a self, vars: &mut BTreeSet<&'a Variable>) { match self { GraphPattern::BGP(p) => { for pattern in p { match pattern { TripleOrPathPattern::Triple(tp) => { if let TermOrVariable::Variable(ref s) = tp.subject { adds_if_has_name(vars, s); } if let NamedNodeOrVariable::Variable(ref p) = tp.predicate { adds_if_has_name(vars, p); } if let TermOrVariable::Variable(ref o) = tp.object { adds_if_has_name(vars, o); } } TripleOrPathPattern::Path(ppp) => { if let TermOrVariable::Variable(ref s) = ppp.subject { adds_if_has_name(vars, s); } if let TermOrVariable::Variable(ref o) = ppp.object { adds_if_has_name(vars, o); } } } } } GraphPattern::Join(a, b) => { a.add_visible_variables(vars); b.add_visible_variables(vars); } GraphPattern::LeftJoin(a, b, _) => { a.add_visible_variables(vars); b.add_visible_variables(vars); } GraphPattern::Filter(_, p) => p.add_visible_variables(vars), GraphPattern::Union(a, b) => { a.add_visible_variables(vars); b.add_visible_variables(vars); } GraphPattern::Graph(_, p) => p.add_visible_variables(vars), GraphPattern::Extend(p, v, _) => { p.add_visible_variables(vars); adds_if_has_name(vars, &v); } GraphPattern::Minus(a, _) => a.add_visible_variables(vars), GraphPattern::Service(_, p, _) => p.add_visible_variables(vars), GraphPattern::AggregateJoin(_, a) => vars.extend(a.iter().map(|(_, v)| v)), GraphPattern::Data(b) => vars.extend(b.variables_iter()), GraphPattern::OrderBy(l, _) => l.add_visible_variables(vars), GraphPattern::Project(_, pv) => vars.extend(pv.iter()), GraphPattern::Distinct(l) => l.add_visible_variables(vars), GraphPattern::Reduced(l) => l.add_visible_variables(vars), GraphPattern::Slice(l, _, _) => l.add_visible_variables(vars), } } } fn adds_if_has_name<'a>(vars: &mut BTreeSet<&'a Variable>, var: &'a Variable) { if var.has_name() { vars.insert(var); } } struct SparqlGraphPattern<'a>(&'a GraphPattern); impl<'a> fmt::Display for SparqlGraphPattern<'a> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self.0 { GraphPattern::BGP(p) => { for pattern in p { write!(f, "{} .", SparqlTripleOrPathPattern(pattern))? } Ok(()) } GraphPattern::Join(a, b) => write!( f, "{{ {} }} {{ {} }}", SparqlGraphPattern(&*a), SparqlGraphPattern(&*b) ), GraphPattern::LeftJoin(a, b, e) => write!( f, "{} OPTIONAL {{ {} FILTER({}) }}", SparqlGraphPattern(&*a), SparqlGraphPattern(&*b), SparqlExpression(e) ), GraphPattern::Filter(e, p) => write!( f, "{} FILTER({})", SparqlGraphPattern(&*p), SparqlExpression(e) ), GraphPattern::Union(a, b) => write!( f, "{{ {} }} UNION {{ {} }}", SparqlGraphPattern(&*a), SparqlGraphPattern(&*b), ), GraphPattern::Graph(g, p) => { write!(f, "GRAPH {} {{ {} }}", g, SparqlGraphPattern(&*p),) } GraphPattern::Extend(p, v, e) => write!( f, "{} BIND({} AS {})", SparqlGraphPattern(&*p), SparqlExpression(e), v ), GraphPattern::Minus(a, b) => write!( f, "{} MINUS {{ {} }}", SparqlGraphPattern(&*a), SparqlGraphPattern(&*b) ), GraphPattern::Service(n, p, s) => if *s { write!(f, "SERVICE SILENT {} {{ {} }}", n, SparqlGraphPattern(&*p)) } else { write!(f, "SERVICE {} {{ {} }}", n, SparqlGraphPattern(&*p)) }, GraphPattern::Data(bs) => if bs.is_empty() { Ok(()) } else { write!(f, "VALUES ( ")?; for var in bs.variables() { write!(f, "{} ", var)?; } write!(f, ") {{ ")?; for values in bs.values_iter() { write!(f, "( ")?; for val in values { match val { Some(val) => write!(f, "{} ", val), None => write!(f, "UNDEF "), }?; } write!(f, ") ")?; } write!(f, " }}") }, GraphPattern::AggregateJoin(GroupPattern(group, p), agg) => write!( f, "{{ SELECT {} WHERE {{ {} }} GROUP BY {} }}", agg.iter() .map(|(a, v)| format!("({} AS {})", SparqlAggregation(&a), v)) .collect::>() .join(" "), SparqlGraphPattern(&*p), group .iter() .map(|e| format!("({})", e.to_string())) .collect::>() .join(" ") ), p => write!( f, "{{ {} }}", SparqlGraphRootPattern { algebra: p, dataset: &EMPTY_DATASET } ), } } } struct SparqlGraphRootPattern<'a> { algebra: &'a GraphPattern, dataset: &'a DatasetSpec, } impl<'a> fmt::Display for SparqlGraphRootPattern<'a> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { let mut distinct = false; let mut reduced = false; let mut order = None; let mut start = 0; let mut length = None; let mut project: &[Variable] = &[]; let mut child = self.algebra; loop { match child { GraphPattern::OrderBy(l, o) => { order = Some(o); child = &*l; } GraphPattern::Project(l, pv) if project.is_empty() => { project = pv; child = &*l; } GraphPattern::Distinct(l) => { distinct = true; child = &*l; } GraphPattern::Reduced(l) => { reduced = true; child = &*l; } GraphPattern::Slice(l, s, len) => { start = *s; length = *len; child = l; } p => { write!(f, "SELECT ")?; if distinct { write!(f, "DISTINCT ")?; } if reduced { write!(f, "REDUCED ")?; } write!( f, "{} {} WHERE {{ {} }}", build_sparql_select_arguments(project), self.dataset, SparqlGraphPattern(p) )?; if let Some(order) = order { write!( f, " ORDER BY {}", order .iter() .map(|c| SparqlOrderComparator(c).to_string()) .collect::>() .join(" ") )?; } if start > 0 { write!(f, " OFFSET {}", start)?; } if let Some(length) = length { write!(f, " LIMIT {}", length)?; } return Ok(()); } } } } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub struct GroupPattern(pub Vec, pub Box); impl fmt::Display for GroupPattern { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!( f, "Group(({}), {})", self.0 .iter() .map(|c| c.to_string()) .collect::>() .join(", "), self.1 ) } } fn build_sparql_select_arguments(args: &[Variable]) -> String { if args.is_empty() { "*".to_owned() } else { args.iter() .map(|v| v.to_string()) .collect::>() .join(" ") } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum Aggregation { Count(Option>, bool), Sum(Box, bool), Min(Box, bool), Max(Box, bool), Avg(Box, bool), Sample(Box, bool), GroupConcat(Box, bool, Option), } impl fmt::Display for Aggregation { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { Aggregation::Count(e, distinct) => if *distinct { e.as_ref() .map(|ex| write!(f, "COUNT(DISTINCT {})", ex)) .unwrap_or_else(|| write!(f, "COUNT(DISTINCT *)")) } else { e.as_ref() .map(|ex| write!(f, "COUNT({})", ex)) .unwrap_or_else(|| write!(f, "COUNT(*)")) }, Aggregation::Sum(e, distinct) => if *distinct { write!(f, "Aggregation(Distinct({}), Sum, {{}})", e) } else { write!(f, "Aggregation({}, Sum, {{}})", e) }, Aggregation::Min(e, distinct) => if *distinct { write!(f, "Aggregation(Distinct({}), Min, {{}})", e) } else { write!(f, "Aggregation({}, Min, {{}})", e) }, Aggregation::Max(e, distinct) => if *distinct { write!(f, "Aggregation(Distinct({}), Max, {{}})", e) } else { write!(f, "Aggregation({}, Max, {{}})", e) }, Aggregation::Avg(e, distinct) => if *distinct { write!(f, "Aggregation(Distinct({}), Avg, {{}})", e) } else { write!(f, "Aggregation({}, Avg, {{}})", e) }, Aggregation::Sample(e, distinct) => if *distinct { write!(f, "Aggregation(Distinct({}), Sum, {{}})", e) } else { write!(f, "Aggregation({}, Sample, {{}})", e) }, Aggregation::GroupConcat(e, distinct, sep) => if *distinct { sep.as_ref() .map(|s| { write!( f, "Aggregation(Distinct({}), GroupConcat, {{\"separator\" → \"{}\"}})", e, s.escape() ) }) .unwrap_or_else(|| write!(f, "Aggregation(Distinct({}), GroupConcat, {{}})", e)) } else { sep.as_ref() .map(|s| { write!( f, "Aggregation({}, GroupConcat, {{\"separator\" → \"{}\"}})", e, s.escape() ) }) .unwrap_or_else(|| write!(f, "Aggregation(Distinct({}), GroupConcat, {{}})", e)) }, } } } struct SparqlAggregation<'a>(&'a Aggregation); impl<'a> fmt::Display for SparqlAggregation<'a> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self.0 { Aggregation::Count(e, distinct) => if *distinct { if let Some(e) = e { write!(f, "COUNT(DISTINCT {})", SparqlExpression(e)) } else { write!(f, "COUNT(DISTINCT *)") } } else if let Some(e) = e { write!(f, "COUNT({})", SparqlExpression(e)) } else { write!(f, "COUNT(*)") }, Aggregation::Sum(e, distinct) => if *distinct { write!(f, "SUM(DISTINCT {})", SparqlExpression(e)) } else { write!(f, "SUM({})", SparqlExpression(e)) }, Aggregation::Min(e, distinct) => if *distinct { write!(f, "MIN(DISTINCT {})", SparqlExpression(e)) } else { write!(f, "MIN({})", SparqlExpression(e)) }, Aggregation::Max(e, distinct) => if *distinct { write!(f, "MAX(DISTINCT {})", SparqlExpression(e)) } else { write!(f, "MAX({})", SparqlExpression(e)) }, Aggregation::Avg(e, distinct) => if *distinct { write!(f, "AVG(DISTINCT {})", SparqlExpression(e)) } else { write!(f, "AVG({})", SparqlExpression(e)) }, Aggregation::Sample(e, distinct) => if *distinct { write!(f, "SAMPLE(DISTINCT {})", SparqlExpression(e)) } else { write!(f, "SAMPLE({})", SparqlExpression(e)) }, Aggregation::GroupConcat(e, distinct, sep) => if *distinct { if let Some(sep) = sep { write!( f, "GROUP_CONCAT(DISTINCT {}; SEPARATOR = \"{}\")", SparqlExpression(e), sep.escape() ) } else { write!(f, "GROUP_CONCAT(DISTINCT {})", SparqlExpression(e)) } } else if let Some(sep) = sep { write!( f, "GROUP_CONCAT({}; SEPARATOR = \"{}\")", SparqlExpression(e), sep.escape() ) } else { write!(f, "GROUP_CONCAT({})", SparqlExpression(e)) }, } } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum OrderComparator { Asc(Expression), Desc(Expression), } impl fmt::Display for OrderComparator { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { OrderComparator::Asc(e) => write!(f, "ASC({})", e), OrderComparator::Desc(e) => write!(f, "DESC({})", e), } } } impl From for OrderComparator { fn from(e: Expression) -> Self { OrderComparator::Asc(e) } } struct SparqlOrderComparator<'a>(&'a OrderComparator); impl<'a> fmt::Display for SparqlOrderComparator<'a> { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self.0 { OrderComparator::Asc(e) => write!(f, "ASC({})", SparqlExpression(e)), OrderComparator::Desc(e) => write!(f, "DESC({})", SparqlExpression(e)), } } } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash, Default)] pub struct DatasetSpec { pub default: Vec, pub named: Vec, } impl DatasetSpec { pub fn new_with_default(graph: NamedNode) -> Self { Self { default: vec![graph], named: Vec::default(), } } pub fn new_with_named(graph: NamedNode) -> Self { Self { default: Vec::default(), named: vec![graph], } } } impl Add for DatasetSpec { type Output = Self; fn add(mut self, rhs: Self) -> Self { self.default.extend_from_slice(&rhs.default); self.named.extend_from_slice(&rhs.named); self } } impl fmt::Display for DatasetSpec { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { for g in &self.default { write!(f, "FROM {} ", g)?; } for g in &self.named { write!(f, "FROM NAMED {} ", g)?; } Ok(()) } } lazy_static! { static ref EMPTY_DATASET: DatasetSpec = DatasetSpec::default(); } #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Hash)] pub enum Query { Select { dataset: DatasetSpec, algebra: GraphPattern, }, Construct { construct: Vec, dataset: DatasetSpec, algebra: GraphPattern, }, Describe { dataset: DatasetSpec, algebra: GraphPattern, }, Ask { dataset: DatasetSpec, algebra: GraphPattern, }, } impl fmt::Display for Query { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { Query::Select { dataset, algebra } => write!( f, "{}", SparqlGraphRootPattern { algebra: &algebra, dataset: &dataset } ), Query::Construct { construct, dataset, algebra, } => write!( f, "CONSTRUCT {{ {} }} {} WHERE {{ {} }}", construct .iter() .map(|t| t.to_string()) .collect::>() .join(" . "), dataset, SparqlGraphRootPattern { algebra: &algebra, dataset: &EMPTY_DATASET } ), Query::Describe { dataset, algebra } => write!( f, "DESCRIBE * {} WHERE {{ {} }}", dataset, SparqlGraphRootPattern { algebra: &algebra, dataset: &EMPTY_DATASET } ), Query::Ask { dataset, algebra } => write!( f, "ASK {} WHERE {{ {} }}", dataset, SparqlGraphRootPattern { algebra: &algebra, dataset: &EMPTY_DATASET } ), } } } pub enum QueryResult<'a> { Bindings(BindingsIterator<'a>), Boolean(bool), Graph(Box> + 'a>), }