All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.rdf4j.spin.SpinRenderer Maven / Gradle / Ivy

There is a newer version: 5.0.2
Show newest version
/*******************************************************************************
 * 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