org.eclipse.rdf4j.spin.SpinRenderer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of rdf4j-spin Show documentation
Show all versions of rdf4j-spin Show documentation
SPARQL input notation interfaces and implementations
/*******************************************************************************
* Copyright (c) 2015 Eclipse RDF4J contributors, Aduna, and others.
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Distribution License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/org/documents/edl-v10.php.
*
* SPDX-License-Identifier: BSD-3-Clause
*******************************************************************************/
package org.eclipse.rdf4j.spin;
import java.io.IOException;
import java.io.StringReader;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.eclipse.rdf4j.model.BNode;
import org.eclipse.rdf4j.model.IRI;
import org.eclipse.rdf4j.model.Resource;
import org.eclipse.rdf4j.model.Statement;
import org.eclipse.rdf4j.model.Value;
import org.eclipse.rdf4j.model.ValueFactory;
import org.eclipse.rdf4j.model.impl.BooleanLiteral;
import org.eclipse.rdf4j.model.impl.SimpleValueFactory;
import org.eclipse.rdf4j.model.vocabulary.AFN;
import org.eclipse.rdf4j.model.vocabulary.RDF;
import org.eclipse.rdf4j.model.vocabulary.SP;
import org.eclipse.rdf4j.model.vocabulary.XSD;
import org.eclipse.rdf4j.query.BindingSet;
import org.eclipse.rdf4j.query.Dataset;
import org.eclipse.rdf4j.query.algebra.And;
import org.eclipse.rdf4j.query.algebra.Avg;
import org.eclipse.rdf4j.query.algebra.BNodeGenerator;
import org.eclipse.rdf4j.query.algebra.BindingSetAssignment;
import org.eclipse.rdf4j.query.algebra.Bound;
import org.eclipse.rdf4j.query.algebra.Clear;
import org.eclipse.rdf4j.query.algebra.Coalesce;
import org.eclipse.rdf4j.query.algebra.Compare;
import org.eclipse.rdf4j.query.algebra.Compare.CompareOp;
import org.eclipse.rdf4j.query.algebra.Count;
import org.eclipse.rdf4j.query.algebra.Create;
import org.eclipse.rdf4j.query.algebra.Datatype;
import org.eclipse.rdf4j.query.algebra.DeleteData;
import org.eclipse.rdf4j.query.algebra.Difference;
import org.eclipse.rdf4j.query.algebra.Distinct;
import org.eclipse.rdf4j.query.algebra.Exists;
import org.eclipse.rdf4j.query.algebra.Extension;
import org.eclipse.rdf4j.query.algebra.ExtensionElem;
import org.eclipse.rdf4j.query.algebra.Filter;
import org.eclipse.rdf4j.query.algebra.FunctionCall;
import org.eclipse.rdf4j.query.algebra.Group;
import org.eclipse.rdf4j.query.algebra.GroupConcat;
import org.eclipse.rdf4j.query.algebra.IRIFunction;
import org.eclipse.rdf4j.query.algebra.If;
import org.eclipse.rdf4j.query.algebra.InsertData;
import org.eclipse.rdf4j.query.algebra.IsBNode;
import org.eclipse.rdf4j.query.algebra.IsLiteral;
import org.eclipse.rdf4j.query.algebra.IsNumeric;
import org.eclipse.rdf4j.query.algebra.IsURI;
import org.eclipse.rdf4j.query.algebra.Join;
import org.eclipse.rdf4j.query.algebra.Lang;
import org.eclipse.rdf4j.query.algebra.LeftJoin;
import org.eclipse.rdf4j.query.algebra.Load;
import org.eclipse.rdf4j.query.algebra.LocalName;
import org.eclipse.rdf4j.query.algebra.MathExpr;
import org.eclipse.rdf4j.query.algebra.MathExpr.MathOp;
import org.eclipse.rdf4j.query.algebra.Max;
import org.eclipse.rdf4j.query.algebra.Min;
import org.eclipse.rdf4j.query.algebra.Modify;
import org.eclipse.rdf4j.query.algebra.MultiProjection;
import org.eclipse.rdf4j.query.algebra.Not;
import org.eclipse.rdf4j.query.algebra.Or;
import org.eclipse.rdf4j.query.algebra.Order;
import org.eclipse.rdf4j.query.algebra.OrderElem;
import org.eclipse.rdf4j.query.algebra.Projection;
import org.eclipse.rdf4j.query.algebra.ProjectionElem;
import org.eclipse.rdf4j.query.algebra.ProjectionElemList;
import org.eclipse.rdf4j.query.algebra.QueryModelNode;
import org.eclipse.rdf4j.query.algebra.Reduced;
import org.eclipse.rdf4j.query.algebra.Regex;
import org.eclipse.rdf4j.query.algebra.Sample;
import org.eclipse.rdf4j.query.algebra.Service;
import org.eclipse.rdf4j.query.algebra.Slice;
import org.eclipse.rdf4j.query.algebra.StatementPattern;
import org.eclipse.rdf4j.query.algebra.Str;
import org.eclipse.rdf4j.query.algebra.Sum;
import org.eclipse.rdf4j.query.algebra.TupleExpr;
import org.eclipse.rdf4j.query.algebra.Union;
import org.eclipse.rdf4j.query.algebra.UpdateExpr;
import org.eclipse.rdf4j.query.algebra.ValueConstant;
import org.eclipse.rdf4j.query.algebra.ValueExpr;
import org.eclipse.rdf4j.query.algebra.Var;
import org.eclipse.rdf4j.query.algebra.helpers.AbstractQueryModelVisitor;
import org.eclipse.rdf4j.query.parser.ParsedBooleanQuery;
import org.eclipse.rdf4j.query.parser.ParsedDescribeQuery;
import org.eclipse.rdf4j.query.parser.ParsedGraphQuery;
import org.eclipse.rdf4j.query.parser.ParsedOperation;
import org.eclipse.rdf4j.query.parser.ParsedQuery;
import org.eclipse.rdf4j.query.parser.ParsedTupleQuery;
import org.eclipse.rdf4j.query.parser.ParsedUpdate;
import org.eclipse.rdf4j.query.parser.sparql.SPARQLUpdateDataBlockParser;
import org.eclipse.rdf4j.rio.RDFHandler;
import org.eclipse.rdf4j.rio.RDFHandlerException;
import org.eclipse.rdf4j.rio.RDFParseException;
import org.eclipse.rdf4j.rio.helpers.AbstractRDFHandler;
import com.google.common.base.Function;
public class SpinRenderer {
public enum Output {
TEXT_AND_RDF(true, true),
TEXT_ONLY(true, false),
RDF_ONLY(false, true);
final boolean text, rdf;
Output(boolean text, boolean rdf) {
this.text = text;
this.rdf = rdf;
}
}
private final ValueFactory valueFactory;
private final Output output;
private final Function wellKnownVars;
private final Function wellKnownFunctions;
public SpinRenderer() {
this(Output.TEXT_AND_RDF);
}
public SpinRenderer(Output output) {
this(output, SpinWellKnownVars.INSTANCE::getURI, SpinWellKnownFunctions.INSTANCE::getURI,
SimpleValueFactory.getInstance());
}
public SpinRenderer(Output output, Function wellKnownVarMapper,
Function wellKnownFuncMapper, ValueFactory vf) {
this.output = output;
this.wellKnownVars = wellKnownVarMapper;
this.wellKnownFunctions = wellKnownFuncMapper;
this.valueFactory = vf;
}
public void render(ParsedOperation operation, RDFHandler handler) throws RDFHandlerException {
if (operation instanceof ParsedQuery) {
render((ParsedQuery) operation, handler);
} else if (operation instanceof ParsedUpdate) {
render((ParsedUpdate) operation, handler);
} else {
throw new AssertionError("Unrecognised ParsedOperation: " + operation.getClass());
}
}
public void render(ParsedQuery query, RDFHandler handler) throws RDFHandlerException {
if (query instanceof ParsedBooleanQuery) {
render((ParsedBooleanQuery) query, handler);
} else if (query instanceof ParsedTupleQuery) {
render((ParsedTupleQuery) query, handler);
}
// order matters as subclass of ParsedGraphQuery
else if (query instanceof ParsedDescribeQuery) {
render((ParsedDescribeQuery) query, handler);
} else if (query instanceof ParsedGraphQuery) {
render((ParsedGraphQuery) query, handler);
} else {
throw new AssertionError("Unrecognised ParsedQuery: " + query.getClass());
}
}
public void render(ParsedBooleanQuery query, RDFHandler handler) throws RDFHandlerException {
handler.startRDF();
Resource querySubj = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(querySubj, RDF.TYPE, SP.ASK_CLASS));
if (output.text) {
handler.handleStatement(valueFactory.createStatement(querySubj, SP.TEXT_PROPERTY,
valueFactory.createLiteral(query.getSourceString())));
}
if (output.rdf) {
Resource whereBNode = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(querySubj, SP.WHERE_PROPERTY, whereBNode));
TupleExpr expr = query.getTupleExpr();
SpinVisitor visitor = new AskVisitor(handler, whereBNode, query.getDataset());
expr.visit(visitor);
visitor.end();
}
handler.endRDF();
}
public void render(ParsedTupleQuery query, RDFHandler handler) throws RDFHandlerException {
handler.startRDF();
Resource querySubj = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(querySubj, RDF.TYPE, SP.SELECT_CLASS));
if (output.text) {
handler.handleStatement(valueFactory.createStatement(querySubj, SP.TEXT_PROPERTY,
valueFactory.createLiteral(query.getSourceString())));
}
if (output.rdf) {
TupleExpr expr = query.getTupleExpr();
SpinVisitor visitor = new SpinVisitor(handler, null, querySubj, query.getDataset());
expr.visit(visitor);
visitor.end();
}
handler.endRDF();
}
public void render(ParsedDescribeQuery query, RDFHandler handler) throws RDFHandlerException {
handler.startRDF();
Resource querySubj = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(querySubj, RDF.TYPE, SP.DESCRIBE_CLASS));
if (output.text) {
handler.handleStatement(valueFactory.createStatement(querySubj, SP.TEXT_PROPERTY,
valueFactory.createLiteral(query.getSourceString())));
}
if (output.rdf) {
TupleExpr expr = query.getTupleExpr();
SpinVisitor visitor = new DescribeVisitor(handler, querySubj, query.getDataset());
expr.visit(visitor);
visitor.end();
}
handler.endRDF();
}
public void render(ParsedGraphQuery query, RDFHandler handler) throws RDFHandlerException {
handler.startRDF();
Resource querySubj = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(querySubj, RDF.TYPE, SP.CONSTRUCT_CLASS));
if (output.text) {
handler.handleStatement(valueFactory.createStatement(querySubj, SP.TEXT_PROPERTY,
valueFactory.createLiteral(query.getSourceString())));
}
if (output.rdf) {
TupleExpr expr = query.getTupleExpr();
SpinVisitor visitor = new ConstructVisitor(handler, querySubj, query.getDataset());
expr.visit(visitor);
visitor.end();
}
handler.endRDF();
}
public void render(ParsedUpdate update, RDFHandler handler) throws RDFHandlerException {
handler.startRDF();
for (Map.Entry entry : update.getNamespaces().entrySet()) {
handler.handleNamespace(entry.getKey(), entry.getValue());
}
String[] sourceStrings = update.getSourceString().split("\\s*;\\s*");
List updateExprs = update.getUpdateExprs();
Map datasets = update.getDatasetMapping();
for (int i = 0; i < updateExprs.size(); i++) {
UpdateExpr updateExpr = updateExprs.get(i);
Resource updateSubj = valueFactory.createBNode();
Dataset dataset = datasets.get(updateExpr);
IRI updateClass;
if (updateExpr instanceof Modify) {
Modify modify = (Modify) updateExpr;
if (modify.getInsertExpr() == null && modify.getWhereExpr().equals(modify.getDeleteExpr())) {
updateClass = SP.DELETE_WHERE_CLASS;
} else {
updateClass = SP.MODIFY_CLASS;
}
} else if (updateExpr instanceof InsertData) {
updateClass = SP.INSERT_DATA_CLASS;
} else if (updateExpr instanceof DeleteData) {
updateClass = SP.DELETE_DATA_CLASS;
} else if (updateExpr instanceof Load) {
updateClass = SP.LOAD_CLASS;
} else if (updateExpr instanceof Clear) {
updateClass = SP.CLEAR_CLASS;
} else if (updateExpr instanceof Create) {
updateClass = SP.CREATE_CLASS;
} else {
throw new RDFHandlerException("Unrecognised UpdateExpr: " + updateExpr.getClass());
}
handler.handleStatement(valueFactory.createStatement(updateSubj, RDF.TYPE, updateClass));
if (output.text) {
handler.handleStatement(valueFactory.createStatement(updateSubj, SP.TEXT_PROPERTY,
valueFactory.createLiteral(sourceStrings[i])));
}
if (output.rdf) {
SpinVisitor visitor = new SpinVisitor(handler, null, updateSubj, dataset);
updateExpr.visit(visitor);
visitor.end();
}
}
handler.endRDF();
}
private class AskVisitor extends SpinVisitor {
AskVisitor(RDFHandler handler, Resource list, Dataset dataset) {
super(handler, list, null, dataset);
}
@Override
public void meet(Slice node) throws RDFHandlerException {
if (!isSubQuery) { // ignore root slice
node.getArg().visit(this);
} else {
super.meet(node);
}
}
}
private class DescribeVisitor extends SpinVisitor {
DescribeVisitor(RDFHandler handler, Resource subject, Dataset dataset) {
super(handler, null, subject, dataset);
}
@Override
public void meet(ProjectionElemList node) throws RDFHandlerException {
if (isSubQuery) {
super.meet(node);
} else {
Resource elemListBNode = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.RESULT_NODES_PROPERTY, elemListBNode));
ListContext ctx = newList(elemListBNode);
meetNode(node);
endList(ctx);
}
}
}
private class ConstructVisitor extends SpinVisitor {
ConstructVisitor(RDFHandler handler, Resource subject, Dataset dataset) {
super(handler, null, subject, dataset);
}
@Override
public void meet(Reduced node) throws RDFHandlerException {
if (!isSubQuery) { // ignore root reduced
node.getArg().visit(this);
} else {
super.meet(node);
}
}
@Override
public void meet(ProjectionElemList node) throws RDFHandlerException {
if (isSubQuery) {
super.meet(node);
} else if (isMultiProjection) {
listEntry();
meetNode(node);
} else {
ListContext ctx = startTemplateList();
listEntry();
meetNode(node);
endTemplateList(ctx);
}
}
@Override
public void meet(ProjectionElem node) throws RDFHandlerException {
if (isSubQuery) {
super.meet(node);
} else {
String varName = node.getName();
ValueExpr valueExpr = inlineBindings.getValueExpr(varName);
Value value = (valueExpr instanceof ValueConstant) ? ((ValueConstant) valueExpr).getValue()
: getVar(varName);
String targetName = node.getProjectionAlias().orElse(null);
IRI pred;
if ("subject".equals(targetName)) {
pred = SP.SUBJECT_PROPERTY;
} else if ("predicate".equals(targetName)) {
pred = SP.PREDICATE_PROPERTY;
} else if ("object".equals(targetName)) {
pred = SP.OBJECT_PROPERTY;
} else {
throw new AssertionError("Unexpected ProjectionElem: " + node);
}
handler.handleStatement(valueFactory.createStatement(subject, pred, value));
}
}
}
private class SpinVisitor extends AbstractQueryModelVisitor {
final RDFHandler handler;
final Dataset dataset;
final Map varBNodes = new HashMap<>();
final Map namedGraphLists = new HashMap<>();
ExtensionContext inlineBindings;
Resource list;
Resource subject;
IRI predicate;
ListContext namedGraphContext;
boolean isMultiProjection;
boolean isSubQuery;
boolean hasGroup;
SpinVisitor(RDFHandler handler, Resource list, Resource subject, Dataset dataset) {
this.handler = handler;
this.list = list;
this.subject = subject;
this.dataset = dataset;
}
private ExtensionContext meetExtension(TupleExpr expr) {
ExtensionContext extVisitor = new ExtensionContext();
expr.visit(extVisitor);
ExtensionContext oldInlineBindings = inlineBindings;
inlineBindings = (extVisitor.extension) != null ? extVisitor : null;
return oldInlineBindings;
}
ListContext save() {
return new ListContext(list, subject);
}
void update(ListContext ctx) {
ctx.list = list;
ctx.subject = subject;
}
void restore(ListContext ctx) {
list = ctx.list;
subject = ctx.subject;
}
ListContext newList(Resource res) {
ListContext ctx = save();
list = res;
subject = null;
return ctx;
}
void listEntry() throws RDFHandlerException {
listEntry(null);
}
void listEntry(Value entry) throws RDFHandlerException {
if (list == null) {
list = valueFactory.createBNode();
}
if (subject != null) {
nextListEntry(valueFactory.createBNode());
}
if (entry == null) {
entry = valueFactory.createBNode();
}
handler.handleStatement(valueFactory.createStatement(list, RDF.FIRST, entry));
if (entry instanceof Resource) {
subject = (Resource) entry;
} else {
// in this case, actual value doesn't matter, only that it is
// not null
subject = RDF.NIL;
}
}
void endList(ListContext ctx) throws RDFHandlerException {
nextListEntry(RDF.NIL);
if (ctx != null) {
restore(ctx);
}
}
private void nextListEntry(Resource nextEntry) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(list, RDF.REST, nextEntry));
list = nextEntry;
subject = null;
}
Resource getVar(String name) throws RDFHandlerException {
Resource res = (wellKnownVars != null) ? wellKnownVars.apply(name) : null;
if (res == null) {
res = varBNodes.get(name);
if (res == null) {
BNode bnode = valueFactory.createBNode(name);
varBNodes.put(name, bnode);
res = bnode;
}
}
return res;
}
ListContext getNamedGraph(Var context) throws RDFHandlerException {
ListContext currentCtx;
namedGraphContext = namedGraphLists.get(context.getName());
if (namedGraphContext == null) {
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.NAMED_GRAPH_CLASS));
predicate = SP.GRAPH_NAME_NODE_PROPERTY;
context.visit(this);
Resource elementsList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.ELEMENTS_PROPERTY, elementsList));
currentCtx = newList(elementsList);
namedGraphContext = save();
namedGraphLists.put(context.getName(), namedGraphContext);
} else {
currentCtx = save();
restore(namedGraphContext);
}
return currentCtx;
}
void restoreNamedGraph(ListContext ctx) {
update(namedGraphContext);
restore(ctx);
}
public void end() throws RDFHandlerException {
if (list != null) {
endList(null);
}
// end any named graph lists
for (ListContext elementsCtx : namedGraphLists.values()) {
restore(elementsCtx);
endList(null);
}
// output all the var BNodes together to give a more friendlier RDF
// structure
for (Map.Entry entry : varBNodes.entrySet()) {
handler.handleStatement(valueFactory.createStatement(entry.getValue(), SP.VAR_NAME_PROPERTY,
valueFactory.createLiteral(entry.getKey())));
}
}
@Override
public void meet(MultiProjection node) throws RDFHandlerException {
ExtensionContext oldInlineBindings = meetExtension(node.getArg());
ListContext ctx = startTemplateList();
isMultiProjection = true;
for (ProjectionElemList proj : node.getProjections()) {
proj.visit(this);
}
endTemplateList(ctx);
isMultiProjection = false;
visitWhere(node.getArg());
inlineBindings = oldInlineBindings;
}
ListContext startTemplateList() throws RDFHandlerException {
Resource elemListBNode = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.TEMPLATES_PROPERTY, elemListBNode));
return newList(elemListBNode);
}
void endTemplateList(ListContext ctx) throws RDFHandlerException {
endList(ctx);
}
@Override
public void meet(Projection node) throws RDFHandlerException {
ExtensionContext oldInlineBindings = meetExtension(node.getArg());
if (isSubQuery) {
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.SUB_QUERY_CLASS));
Resource queryBNode = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.QUERY_PROPERTY, queryBNode));
subject = queryBNode;
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.SELECT_CLASS));
}
node.getProjectionElemList().visit(this);
visitWhere(node.getArg());
node.getArg().visit(new GroupVisitor());
node.getArg().visit(new OrderVisitor());
inlineBindings = oldInlineBindings;
hasGroup = false;
}
private void visitWhere(TupleExpr where) throws RDFHandlerException {
Resource whereBNode = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.WHERE_PROPERTY, whereBNode));
isSubQuery = true; // further projection elements are for
// sub-queries
ListContext ctx = newList(whereBNode);
where.visit(this);
endList(ctx);
}
@Override
public void meet(ProjectionElemList node) throws RDFHandlerException {
Resource elemListBNode = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.RESULT_VARIABLES_PROPERTY, elemListBNode));
ListContext ctx = newList(elemListBNode);
super.meet(node);
endList(ctx);
}
@Override
public void meet(ProjectionElem node) throws RDFHandlerException {
ValueExpr valueExpr = null;
if (inlineBindings != null) {
String varName = node.getName();
valueExpr = inlineBindings.getValueExpr(varName);
}
Resource targetVar = getVar(node.getProjectionAlias().orElse(node.getName()));
listEntry(targetVar);
if (valueExpr != null && !(valueExpr instanceof Var)) {
Resource currentSubj = subject;
subject = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(targetVar, SP.EXPRESSION_PROPERTY, subject));
valueExpr.visit(new ExtensionVisitor());
subject = currentSubj;
}
}
@Override
public void meet(Extension node) throws RDFHandlerException {
if (inlineBindings != null && inlineBindings.extension == node) {
// this is the first Extension node and has already been handled
// by meetExtension()
// to produce inline bindings in SELECT so we can skip it here
node.getArg().visit(this);
} else {
// any further Extension nodes produce BIND() clauses
node.getArg().visit(this);
for (ExtensionElem elem : node.getElements()) {
elem.visit(this);
}
}
}
@Override
public void meet(ExtensionElem node) throws RDFHandlerException {
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.BIND_CLASS));
Resource var = getVar(node.getName());
handler.handleStatement(valueFactory.createStatement(subject, SP.VARIABLE_PROPERTY, var));
meet(node.getExpr());
}
private void meet(ValueExpr node) throws RDFHandlerException {
predicate = SP.EXPRESSION_PROPERTY;
ListContext ctx = save();
list = null;
node.visit(this);
restore(ctx);
}
private void flushPendingStatement() throws RDFHandlerException {
if (predicate != null) {
Resource res = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, predicate, res));
subject = res;
}
}
@Override
public void meet(StatementPattern node) throws RDFHandlerException {
ListContext ctxList = null;
if (StatementPattern.Scope.NAMED_CONTEXTS == node.getScope()) {
ctxList = getNamedGraph(node.getContextVar());
}
listEntry();
predicate = SP.SUBJECT_PROPERTY;
node.getSubjectVar().visit(this);
predicate = SP.PREDICATE_PROPERTY;
node.getPredicateVar().visit(this);
predicate = SP.OBJECT_PROPERTY;
node.getObjectVar().visit(this);
predicate = null;
if (ctxList != null) {
restoreNamedGraph(ctxList);
}
}
@Override
public void meet(Var node) throws RDFHandlerException {
Value value;
if (node.isConstant()) {
value = node.getValue();
} else {
value = getVar(node.getName());
}
handler.handleStatement(valueFactory.createStatement(subject, predicate, value));
}
@Override
public void meet(ValueConstant node) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(subject, predicate, node.getValue()));
}
@Override
public void meet(Filter node) throws RDFHandlerException {
hasGroup = false;
node.getArg().visit(this);
if (!hasGroup) {
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.FILTER_CLASS));
meet(node.getCondition());
}
}
@Override
public void meet(Compare node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, toValue(node.getOperator())));
predicate = SP.ARG1_PROPERTY;
node.getLeftArg().visit(this);
predicate = SP.ARG2_PROPERTY;
node.getRightArg().visit(this);
subject = currentSubj;
predicate = null;
}
private Value toValue(CompareOp op) {
switch (op) {
case EQ:
return SP.EQ;
case NE:
return SP.NE;
case LT:
return SP.LT;
case LE:
return SP.LE;
case GE:
return SP.GE;
case GT:
return SP.GT;
}
throw new AssertionError("Unrecognised CompareOp: " + op);
}
@Override
public void meet(MathExpr node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, toValue(node.getOperator())));
predicate = SP.ARG1_PROPERTY;
node.getLeftArg().visit(this);
predicate = SP.ARG2_PROPERTY;
node.getRightArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(And node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.AND));
predicate = SP.ARG1_PROPERTY;
node.getLeftArg().visit(this);
predicate = SP.ARG2_PROPERTY;
node.getRightArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(Or node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.OR));
predicate = SP.ARG1_PROPERTY;
node.getLeftArg().visit(this);
predicate = SP.ARG2_PROPERTY;
node.getRightArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(Bound node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.BOUND));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(If node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.IF));
predicate = SP.ARG1_PROPERTY;
node.getCondition().visit(this);
predicate = SP.ARG2_PROPERTY;
node.getResult().visit(this);
predicate = SP.ARG3_PROPERTY;
node.getAlternative().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(Coalesce node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.COALESCE));
List args = node.getArguments();
for (int i = 0; i < args.size(); i++) {
predicate = toArgProperty(i + 1);
args.get(i).visit(this);
}
subject = currentSubj;
predicate = null;
}
@Override
public void meet(IsURI node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.IS_IRI));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(IsBNode node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.IS_BLANK));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(IsLiteral node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.IS_LITERAL));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(IsNumeric node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.IS_NUMERIC));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(Str node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.STR));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(Lang node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.LANG));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(Datatype node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.DATATYPE));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(IRIFunction node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.IRI));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(BNodeGenerator node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.BNODE));
if (node.getNodeIdExpr() != null) {
predicate = SP.ARG1_PROPERTY;
node.getNodeIdExpr().visit(this);
}
subject = currentSubj;
predicate = null;
}
@Override
public void meet(Regex node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.REGEX));
predicate = SP.ARG1_PROPERTY;
node.getLeftArg().visit(this);
predicate = SP.ARG2_PROPERTY;
node.getRightArg().visit(this);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(LocalName node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, AFN.LOCALNAME));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
private Value toValue(MathOp op) {
switch (op) {
case PLUS:
return SP.ADD;
case MINUS:
return SP.SUB;
case MULTIPLY:
return SP.MUL;
case DIVIDE:
return SP.DIVIDE;
}
throw new AssertionError("Unrecognised MathOp: " + op);
}
@Override
public void meet(FunctionCall node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, toValue(node)));
List args = node.getArgs();
for (int i = 0; i < args.size(); i++) {
predicate = toArgProperty(i + 1);
args.get(i).visit(this);
}
subject = currentSubj;
predicate = null;
}
private Value toValue(FunctionCall node) {
String funcName = node.getURI();
IRI funcUri = wellKnownFunctions.apply(funcName);
if (funcUri == null) {
funcUri = valueFactory.createIRI(funcName);
}
return funcUri;
}
private IRI toArgProperty(int i) {
switch (i) {
case 1:
return SP.ARG1_PROPERTY;
case 2:
return SP.ARG2_PROPERTY;
case 3:
return SP.ARG3_PROPERTY;
case 4:
return SP.ARG4_PROPERTY;
case 5:
return SP.ARG5_PROPERTY;
default:
return valueFactory.createIRI(SP.NAMESPACE, "arg" + i);
}
}
@Override
public void meet(Not node) throws RDFHandlerException {
if (node.getArg() instanceof Exists) {
super.meet(node);
} else {
Resource currentSubj = subject;
flushPendingStatement();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.NOT));
predicate = SP.ARG1_PROPERTY;
node.getArg().visit(this);
subject = currentSubj;
predicate = null;
}
}
@Override
public void meet(Exists node) throws RDFHandlerException {
Resource currentSubj = subject;
flushPendingStatement();
Resource op = (node.getParentNode() instanceof Not) ? SP.NOT_EXISTS : SP.EXISTS;
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, op));
Resource elementsList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.ELEMENTS_PROPERTY, elementsList));
ListContext elementsCtx = newList(elementsList);
node.getSubQuery().visit(this);
endList(elementsCtx);
subject = currentSubj;
predicate = null;
}
@Override
public void meet(Group node) throws RDFHandlerException {
// skip over GroupElem - leave this to the GroupVisitor later
node.getArg().visit(this);
hasGroup = true;
}
@Override
public void meet(Order node) throws RDFHandlerException {
// skip over OrderElem - leave this to the OrderVisitor later
node.getArg().visit(this);
}
@Override
public void meet(Slice node) throws RDFHandlerException {
node.getArg().visit(this);
if (node.hasLimit()) {
handler.handleStatement(valueFactory.createStatement(subject, SP.LIMIT_PROPERTY,
valueFactory.createLiteral(Long.toString(node.getLimit()), XSD.INTEGER)));
}
if (node.hasOffset()) {
handler.handleStatement(valueFactory.createStatement(subject, SP.OFFSET_PROPERTY,
valueFactory.createLiteral(Long.toString(node.getOffset()), XSD.INTEGER)));
}
}
@Override
public void meet(Distinct node) throws RDFHandlerException {
node.getArg().visit(this);
handler.handleStatement(valueFactory.createStatement(subject, SP.DISTINCT_PROPERTY, BooleanLiteral.TRUE));
}
@Override
public void meet(Reduced node) throws RDFHandlerException {
node.getArg().visit(this);
handler.handleStatement(valueFactory.createStatement(subject, SP.REDUCED_PROPERTY, BooleanLiteral.TRUE));
}
@Override
public void meet(Join node) throws RDFHandlerException {
boolean isGroupGraphPattern = (node.getRightArg() instanceof Join);
if (!isGroupGraphPattern) {
super.meet(node);
} else {
listEntry();
ListContext leftGroupCtx = newList(subject);
node.getLeftArg().visit(this);
endList(leftGroupCtx);
listEntry();
ListContext rightGroupCtx = newList(subject);
node.getRightArg().visit(this);
endList(rightGroupCtx);
}
}
@Override
public void meet(LeftJoin node) throws RDFHandlerException {
node.getLeftArg().visit(this);
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.OPTIONAL_CLASS));
Resource elementsList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.ELEMENTS_PROPERTY, elementsList));
ListContext elementsCtx = newList(elementsList);
node.getRightArg().visit(this);
endList(elementsCtx);
}
@Override
public void meet(Union node) throws RDFHandlerException {
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.UNION_CLASS));
Resource elementsList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.ELEMENTS_PROPERTY, elementsList));
ListContext elementsCtx = newList(elementsList);
listEntry();
ListContext leftCtx = newList(subject);
node.getLeftArg().visit(this);
endList(leftCtx);
listEntry();
ListContext rightCtx = newList(subject);
node.getRightArg().visit(this);
endList(rightCtx);
endList(elementsCtx);
}
@Override
public void meet(Difference node) throws RDFHandlerException {
node.getLeftArg().visit(this);
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.MINUS_CLASS));
Resource elementsList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.ELEMENTS_PROPERTY, elementsList));
ListContext elementsCtx = newList(elementsList);
node.getRightArg().visit(this);
endList(elementsCtx);
}
@Override
public void meet(Service node) throws RDFHandlerException {
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.SERVICE_CLASS));
predicate = SP.SERVICE_URI_PROPERTY;
node.getServiceRef().visit(this);
predicate = null;
Resource elementsList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.ELEMENTS_PROPERTY, elementsList));
ListContext elementsCtx = newList(elementsList);
node.getArg().visit(this);
endList(elementsCtx);
}
@Override
public void meet(BindingSetAssignment node) throws RDFHandlerException {
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.VALUES_CLASS));
Resource bindingList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.BINDINGS_PROPERTY, bindingList));
ListContext bindingCtx = newList(bindingList);
List bindingVars = new ArrayList<>(node.getBindingNames());
for (BindingSet bs : node.getBindingSets()) {
listEntry();
ListContext setCtx = newList(subject);
for (String varName : bindingVars) {
Value v = bs.getValue(varName);
if (v == null) {
v = SP.UNDEF;
}
listEntry(v);
}
endList(setCtx);
}
endList(bindingCtx);
Resource varNameList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.VAR_NAMES_PROPERTY, varNameList));
ListContext varnameCtx = newList(varNameList);
for (String varName : bindingVars) {
listEntry(valueFactory.createLiteral(varName));
}
endList(varnameCtx);
}
@Override
public void meet(Modify node) throws RDFHandlerException {
TupleExpr insertExpr = node.getInsertExpr();
TupleExpr deleteExpr = node.getDeleteExpr();
TupleExpr whereExpr = node.getWhereExpr();
if (insertExpr == null && whereExpr.equals(deleteExpr)) {
// DELETE WHERE
visitWhere(whereExpr);
} else {
if (insertExpr != null) {
Resource insertList = valueFactory.createBNode();
handler.handleStatement(
valueFactory.createStatement(subject, SP.INSERT_PATTERN_PROPERTY, insertList));
ListContext insertCtx = newList(insertList);
insertExpr.visit(this);
endList(insertCtx);
}
if (deleteExpr != null) {
Resource deleteList = valueFactory.createBNode();
handler.handleStatement(
valueFactory.createStatement(subject, SP.DELETE_PATTERN_PROPERTY, deleteList));
ListContext deleteCtx = newList(deleteList);
deleteExpr.visit(this);
endList(deleteCtx);
}
if (whereExpr != null) {
Resource whereList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.WHERE_PROPERTY, whereList));
ListContext whereCtx = newList(whereList);
whereExpr.visit(this);
endList(whereCtx);
}
}
}
@Override
public void meet(InsertData node) throws RDFHandlerException {
Resource dataList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.DATA_PROPERTY, dataList));
ListContext dataCtx = newList(dataList);
renderDataBlock(node.getDataBlock());
endList(dataCtx);
}
@Override
public void meet(DeleteData node) throws RDFHandlerException {
Resource dataList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.DATA_PROPERTY, dataList));
ListContext dataCtx = newList(dataList);
renderDataBlock(node.getDataBlock());
endList(dataCtx);
}
private void renderDataBlock(String data) throws RDFHandlerException {
SPARQLUpdateDataBlockParser parser = new SPARQLUpdateDataBlockParser(valueFactory);
parser.setAllowBlankNodes(false); // no blank nodes allowed
parser.setRDFHandler(new AbstractRDFHandler() {
final Map namedGraphLists = new HashMap<>();
ListContext namedGraphContext;
@Override
public void handleStatement(Statement st) throws RDFHandlerException {
ListContext ctxList = null;
if (st.getContext() != null) {
ctxList = getNamedGraph(st.getContext());
}
listEntry();
handler.handleStatement(
valueFactory.createStatement(subject, SP.SUBJECT_PROPERTY, st.getSubject()));
handler.handleStatement(
valueFactory.createStatement(subject, SP.PREDICATE_PROPERTY, st.getPredicate()));
handler.handleStatement(valueFactory.createStatement(subject, SP.OBJECT_PROPERTY, st.getObject()));
if (ctxList != null) {
restoreNamedGraph(ctxList);
}
}
@Override
public void endRDF() throws RDFHandlerException {
for (ListContext ctxList : namedGraphLists.values()) {
endList(ctxList);
}
}
private ListContext getNamedGraph(Resource context) throws RDFHandlerException {
ListContext currentCtx;
namedGraphContext = namedGraphLists.get(context);
if (namedGraphContext == null) {
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.NAMED_GRAPH_CLASS));
handler.handleStatement(
valueFactory.createStatement(subject, SP.GRAPH_NAME_NODE_PROPERTY, context));
Resource elementsList = valueFactory.createBNode();
handler.handleStatement(
valueFactory.createStatement(subject, SP.ELEMENTS_PROPERTY, elementsList));
currentCtx = newList(elementsList);
namedGraphContext = save();
namedGraphLists.put(context, namedGraphContext);
} else {
currentCtx = save();
restore(namedGraphContext);
}
return currentCtx;
}
private void restoreNamedGraph(ListContext ctx) {
update(namedGraphContext);
restore(ctx);
}
});
try {
parser.parse(new StringReader(data), "");
} catch (RDFParseException | IOException e) {
throw new RDFHandlerException(e);
}
}
@Override
public void meet(Load node) throws RDFHandlerException {
handler.handleStatement(
valueFactory.createStatement(subject, SP.DOCUMENT_PROPERTY, node.getSource().getValue()));
handler.handleStatement(
valueFactory.createStatement(subject, SP.INTO_PROPERTY, node.getGraph().getValue()));
}
@Override
public void meet(Clear node) throws RDFHandlerException {
if (node.isSilent()) {
handler.handleStatement(valueFactory.createStatement(subject, SP.SILENT_PROPERTY, BooleanLiteral.TRUE));
}
if (node.getGraph() != null) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.GRAPH_IRI_PROPERTY, node.getGraph().getValue()));
} else if (node.getScope() != null) {
switch (node.getScope()) {
case DEFAULT_CONTEXTS:
handler.handleStatement(
valueFactory.createStatement(subject, SP.DEFAULT_PROPERTY, BooleanLiteral.TRUE));
break;
case NAMED_CONTEXTS:
handler.handleStatement(
valueFactory.createStatement(subject, SP.NAMED_PROPERTY, BooleanLiteral.TRUE));
break;
}
} else {
handler.handleStatement(valueFactory.createStatement(subject, SP.ALL_PROPERTY, BooleanLiteral.TRUE));
}
}
@Override
public void meet(Create node) throws RDFHandlerException {
if (node.isSilent()) {
handler.handleStatement(valueFactory.createStatement(subject, SP.SILENT_PROPERTY, BooleanLiteral.TRUE));
}
if (node.getGraph() != null) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.GRAPH_IRI_PROPERTY, node.getGraph().getValue()));
}
}
private final class ExtensionVisitor extends AbstractQueryModelVisitor {
@Override
public void meet(Count node) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.COUNT_CLASS));
if (node.isDistinct()) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.DISTINCT_PROPERTY, BooleanLiteral.TRUE));
}
Resource oldSubject = subject;
super.meet(node);
handler.handleStatement(valueFactory.createStatement(oldSubject, SP.EXPRESSION_PROPERTY, subject));
subject = oldSubject;
}
@Override
public void meet(Max node) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.MAX_CLASS));
if (node.isDistinct()) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.DISTINCT_PROPERTY, BooleanLiteral.TRUE));
}
Resource oldSubject = subject;
super.meet(node);
handler.handleStatement(valueFactory.createStatement(oldSubject, SP.EXPRESSION_PROPERTY, subject));
subject = oldSubject;
}
@Override
public void meet(Min node) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.MIN_CLASS));
if (node.isDistinct()) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.DISTINCT_PROPERTY, BooleanLiteral.TRUE));
}
Resource oldSubject = subject;
super.meet(node);
handler.handleStatement(valueFactory.createStatement(oldSubject, SP.EXPRESSION_PROPERTY, subject));
subject = oldSubject;
}
@Override
public void meet(Sum node) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.SUM_CLASS));
if (node.isDistinct()) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.DISTINCT_PROPERTY, BooleanLiteral.TRUE));
}
Resource oldSubject = subject;
super.meet(node);
handler.handleStatement(valueFactory.createStatement(oldSubject, SP.EXPRESSION_PROPERTY, subject));
subject = oldSubject;
}
@Override
public void meet(Avg node) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.AVG_CLASS));
if (node.isDistinct()) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.DISTINCT_PROPERTY, BooleanLiteral.TRUE));
}
Resource oldSubject = subject;
super.meet(node);
handler.handleStatement(valueFactory.createStatement(oldSubject, SP.EXPRESSION_PROPERTY, subject));
subject = oldSubject;
}
@Override
public void meet(GroupConcat node) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.GROUP_CONCAT_CLASS));
if (node.isDistinct()) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.DISTINCT_PROPERTY, BooleanLiteral.TRUE));
}
Resource oldSubject = subject;
super.meet(node);
handler.handleStatement(valueFactory.createStatement(oldSubject, SP.EXPRESSION_PROPERTY, subject));
subject = oldSubject;
}
@Override
public void meet(Sample node) throws RDFHandlerException {
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, SP.SAMPLE_CLASS));
if (node.isDistinct()) {
handler.handleStatement(
valueFactory.createStatement(subject, SP.DISTINCT_PROPERTY, BooleanLiteral.TRUE));
}
Resource oldSubject = subject;
super.meet(node);
handler.handleStatement(valueFactory.createStatement(oldSubject, SP.EXPRESSION_PROPERTY, subject));
subject = oldSubject;
}
@Override
public void meet(Var node) throws RDFHandlerException {
subject = getVar(node.getName());
}
}
private final class GroupVisitor extends AbstractQueryModelVisitor {
Group group;
@Override
public void meet(Order node) throws RDFHandlerException {
node.getArg().visit(this);
}
@Override
public void meet(Extension node) throws RDFHandlerException {
node.getArg().visit(this);
}
@Override
public void meet(Group node) throws RDFHandlerException {
group = node;
Set groupNames = node.getGroupBindingNames();
if (!groupNames.isEmpty()) {
Resource groupByList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.GROUP_BY_PROPERTY, groupByList));
ListContext groupByCtx = newList(groupByList);
for (String groupName : groupNames) {
Resource var = getVar(groupName);
listEntry(var);
}
endList(groupByCtx);
}
}
@Override
public void meet(Filter node) throws RDFHandlerException {
node.getArg().visit(this);
if (group != null) {
Resource havingList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.HAVING_PROPERTY, havingList));
ListContext havingCtx = newList(havingList);
listEntry();
node.getCondition().visit(SpinVisitor.this);
endList(havingCtx);
}
}
@Override
protected void meetNode(QueryModelNode node) {
// stop
}
}
private final class OrderVisitor extends AbstractQueryModelVisitor {
@Override
public void meet(Order node) throws RDFHandlerException {
Resource orderByList = valueFactory.createBNode();
handler.handleStatement(valueFactory.createStatement(subject, SP.ORDER_BY_PROPERTY, orderByList));
ListContext orderByCtx = newList(orderByList);
for (OrderElem elem : node.getElements()) {
elem.visit(this);
}
endList(orderByCtx);
}
@Override
public void meet(OrderElem node) throws RDFHandlerException {
IRI asc = node.isAscending() ? SP.ASC_CLASS : SP.DESC_CLASS;
listEntry();
handler.handleStatement(valueFactory.createStatement(subject, RDF.TYPE, asc));
SpinVisitor.this.meet(node.getExpr());
}
@Override
protected void meetNode(QueryModelNode node) {
// stop
}
}
}
private static final class ExtensionContext extends AbstractQueryModelVisitor {
Extension extension;
Map extensionExprs;
public ValueExpr getValueExpr(String name) {
return extensionExprs.get(name);
}
@Override
public void meet(Order node) {
node.getArg().visit(this);
}
@Override
public void meet(Extension node) {
extension = node;
List elements = node.getElements();
// NB: preserve ExtensionElem order
extensionExprs = new LinkedHashMap<>(elements.size());
for (ExtensionElem elem : elements) {
extensionExprs.put(elem.getName(), elem.getExpr());
}
}
@Override
protected void meetNode(QueryModelNode node) {
// stop
}
}
private static final class ListContext {
Resource list;
Resource subject;
ListContext(Resource list, Resource subject) {
this.list = list;
this.subject = subject;
}
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy