SparqlBuilderTest.java

/*******************************************************************************
 * Copyright (c) 2020 Eclipse RDF4J contributors.
 *
 * 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.sparqlbuilder.core;

import static org.assertj.core.api.AssertionsForClassTypes.assertThat;
import static org.eclipse.rdf4j.sparqlbuilder.rdf.Rdf.iri;

import org.eclipse.rdf4j.model.vocabulary.DC;
import org.eclipse.rdf4j.sparqlbuilder.constraint.Expressions;
import org.eclipse.rdf4j.sparqlbuilder.core.query.Queries;
import org.eclipse.rdf4j.sparqlbuilder.core.query.SelectQuery;
import org.eclipse.rdf4j.sparqlbuilder.graphpattern.GraphPatternNotTriples;
import org.eclipse.rdf4j.sparqlbuilder.graphpattern.GraphPatterns;
import org.eclipse.rdf4j.sparqlbuilder.rdf.Rdf;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;

/**
 * Tests for precedence order of all operators in SPARQL builder Queries.
 */
public class SparqlBuilderTest {

	protected SelectQuery query;
	protected static final String EXAMPLE_ORG_NS = "https://example.org/ns#";
	protected static final String DC_NS = DC.NAMESPACE;

	@BeforeEach
	public void before() {
		query = Queries.SELECT();
	}

	@Test
	public void testLogicalOperatorPrecedence1() {
		Prefix dc = SparqlBuilder.prefix("dc", iri(DC_NS)), ns = SparqlBuilder.prefix("ns", iri(EXAMPLE_ORG_NS));
		Variable title = SparqlBuilder.var("title"), price = SparqlBuilder.var("price"), x = SparqlBuilder.var("x");

		GraphPatternNotTriples pricePattern = GraphPatterns.and(x.has(ns.iri("price"), price))
				.filter(Expressions.or(Expressions.lt(price, Rdf.literalOf(20)),
						Expressions.and(Expressions.lt(price, Rdf.literalOf(50)),
								Expressions.gt(price, Rdf.literalOf(30)))))
				.optional();
		query.prefix(dc, ns).select(title, price).where(x.has(dc.iri("title"), title), pricePattern);
		assertThat(query.getQueryString())
				.contains("( ?price < 50 && ?price > 30 )");
	}

	@Test
	public void testLogicalOperatorPrecedence2() {
		Prefix dc = SparqlBuilder.prefix("dc", iri(DC_NS)), ns = SparqlBuilder.prefix("ns", iri(EXAMPLE_ORG_NS));
		Variable title = SparqlBuilder.var("title"), price = SparqlBuilder.var("price"), x = SparqlBuilder.var("x");

		GraphPatternNotTriples pricePattern = GraphPatterns.and(x.has(ns.iri("price"), price))
				.filter(Expressions.or(Expressions.lt(price, Rdf.literalOf(20)),
						Expressions.and(Expressions.gt(price, Rdf.literalOf(50)),
								Expressions.or(Expressions.gt(price, Rdf.literalOf(60)),
										Expressions.lt(price, Rdf.literalOf(70))))))
				.optional();

		query.prefix(dc, ns).select(title, price).where(x.has(dc.iri("title"), title), pricePattern);
		assertThat(query.getQueryString())
				.contains("( ?price < 20 || ( ?price > 50 && ( ?price > 60 || ?price < 70 ) ) )");
	}

	@Test
	public void testArithmeticOperatorPrecedence1() {
		Prefix dc = SparqlBuilder.prefix("dc", iri(DC_NS)), ns = SparqlBuilder.prefix("ns", iri(EXAMPLE_ORG_NS));
		Variable title = SparqlBuilder.var("title"), price = SparqlBuilder.var("price"), x = SparqlBuilder.var("x");

		GraphPatternNotTriples pricePattern = GraphPatterns.and(x.has(ns.iri("price"), price))
				.filter(Expressions.or(Expressions.lt(price, Expressions.subtract(Rdf.literalOf(20),
						Expressions.multiply(Rdf.literalOf(2), Rdf.literalOf(5)))),
						Expressions.lt(price, 50)))
				.optional();

		query.prefix(dc, ns).select(title, price).where(x.has(dc.iri("title"), title), pricePattern);
		assertThat(query.getQueryString())
				.contains("( 20 - ( 2 * 5 ) )");
	}

	@Test
	public void testArithmeticOperatorPrecedence2() {
		Prefix dc = SparqlBuilder.prefix("dc", iri(DC_NS)), ns = SparqlBuilder.prefix("ns", iri(EXAMPLE_ORG_NS));
		Variable title = SparqlBuilder.var("title"), price = SparqlBuilder.var("price"), x = SparqlBuilder.var("x");

		GraphPatternNotTriples pricePattern = GraphPatterns.and(x.has(ns.iri("price"), price))
				.filter(Expressions.or(Expressions.lt(price, Expressions.add(Rdf.literalOf(20),
						Expressions.divide(Rdf.literalOf(10), Rdf.literalOf(5)))),
						Expressions.lt(price, 50)))
				.optional();

		query.prefix(dc, ns).select(title, price).where(x.has(dc.iri("title"), title), pricePattern);
		assertThat(query.getQueryString())
				.contains("( 20 + ( 10 / 5 ) )");
	}

	@Test
	public void testArithmeticOperatorPrecedence3() {
		Prefix dc = SparqlBuilder.prefix("dc", iri(DC_NS)), ns = SparqlBuilder.prefix("ns", iri(EXAMPLE_ORG_NS));
		Variable title = SparqlBuilder.var("title"), price = SparqlBuilder.var("price"), x = SparqlBuilder.var("x");

		GraphPatternNotTriples pricePattern = GraphPatterns.and(x.has(ns.iri("price"), price))
				.filter(Expressions.lt(price, Expressions.multiply(Expressions.subtract(Rdf.literalOf(20),
						Rdf.literalOf(2)), Rdf.literalOf(5))))
				.optional();

		query.prefix(dc, ns).select(title, price).where(x.has(dc.iri("title"), title), pricePattern);
		assertThat(query.getQueryString())
				.contains("( ( 20 - 2 ) * 5 ) )");
	}

	@Test
	public void testArithmeticLogicalOperatorPrecedence1() {
		Prefix dc = SparqlBuilder.prefix("dc", iri(DC_NS)), ns = SparqlBuilder.prefix("ns", iri(EXAMPLE_ORG_NS));
		Variable title = SparqlBuilder.var("title"), price = SparqlBuilder.var("price"), x = SparqlBuilder.var("x");

		GraphPatternNotTriples pricePattern = GraphPatterns.and(x.has(ns.iri("price"), price))
				.filter(Expressions.or(Expressions.lt(price, Expressions.add(Rdf.literalOf(20),
						Expressions.divide(Rdf.literalOf(10), Rdf.literalOf(5)))),
						Expressions.lt(Rdf.literalOf(30), Rdf.literalOf(50))))
				.optional();

		query.prefix(dc, ns).select(title, price).where(x.has(dc.iri("title"), title), pricePattern);
		assertThat(query.getQueryString())
				.contains("( 20 + ( 10 / 5 ) ) || 30 < 50 )");
	}

	@Test
	public void testMultipleFilters() {
		Prefix dc = SparqlBuilder.prefix("dc", iri(DC_NS)), ns = SparqlBuilder.prefix("ns", iri(EXAMPLE_ORG_NS));
		Variable title = SparqlBuilder.var("title"), price = SparqlBuilder.var("price"), x = SparqlBuilder.var("x");

		GraphPatternNotTriples pricePattern = GraphPatterns.and(x.has(ns.iri("price"), price))
				.filter(Expressions.lt(price, Rdf.literalOf(50)))
				.filter(Expressions.gt(price, Rdf.literalOf(30)))
				.optional();
		query.prefix(dc, ns)
				.select(title, price)
				.where(x.has(dc.iri("title"), title),
						pricePattern);
		assertThat(query.getQueryString())
				.contains("FILTER ( ?price < 50 )");
		assertThat(query.getQueryString())
				.contains("FILTER ( ?price > 30 )");
	}

}