/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.apache.jena.sparql.algebra.op; import org.apache.jena.sparql.algebra.Op ; import org.apache.jena.sparql.algebra.OpVisitor ; import org.apache.jena.sparql.algebra.Transform ; import org.apache.jena.sparql.expr.Expr ; import org.apache.jena.sparql.expr.ExprList ; import org.apache.jena.sparql.sse.Tags ; import org.apache.jena.sparql.util.NodeIsomorphismMap ; public class OpFilter extends Op1 { protected ExprList expressions ; /** Add expression - mutates an existing filter */ public static Op filter(Expr expr, Op op) { OpFilter f = ensureFilter(op) ; f.getExprs().add(expr) ; return f ; } /** * Ensure that the algebra op is a filter. If the input is a filter, just return that, * else create a filter with no expressions and "this" as the subOp. * @apiNote * This operation assumes the caller is going to add expressions. * Filters without any expressions are discouraged. * Consider collecting the expressions together first and using {@link #filterBy}. */ public static OpFilter ensureFilter(Op op) { if ( op instanceof OpFilter ) return (OpFilter)op ; else return new OpFilter(op) ; } /** @deprecated Renamed as {#link filterBy} */ @Deprecated public static Op filter(ExprList exprs, Op op) { return filterBy(exprs, op) ; } /** @deprecated Use {@link #ensureFilter} */ @Deprecated public static OpFilter filter(Op op) { return ensureFilter(op) ; } /** Combine an ExprList with an Op so that the expressions filter the Op. * If the exprs are empty, return the Op. * If the op is already a OpFilter, merge the expressions into the filters existintg expressions. * Else create a new OpFilter with the expressions and subOp. */ public static Op filterBy(ExprList exprs, Op op) { if ( exprs == null || exprs.isEmpty() ) return op ; OpFilter f = ensureFilter(op) ; f.getExprs().addAll(exprs) ; return f ; } /** Create a OpFilter with the expressions and subOp. * If subOp is a filter, combine expressions (de-layer). */ public static OpFilter filterAlways(ExprList exprs, Op subOp) { OpFilter f = ensureFilter(subOp) ; f.getExprs().addAll(exprs) ; return f ; } /** Make a OpFilter - guaranteed to return an fresh OpFilter */ public static OpFilter filterDirect(ExprList exprs, Op op) { return new OpFilter(exprs, op) ; } /** Make a OpFilter - guaranteed to return an fresh OpFilter */ public static OpFilter filterDirect(Expr expr, Op op) { OpFilter f = new OpFilter(op) ; f.getExprs().add(expr); return f ; } private OpFilter(Op sub) { super(sub) ; expressions = new ExprList() ; } private OpFilter(ExprList exprs, Op sub) { super(sub) ; expressions = exprs ; } /** Compress multiple filters: (filter (filter (filter op)))) into one (filter op) */ public static OpFilter tidy(OpFilter base) { ExprList exprs = new ExprList() ; Op op = base ; while (op instanceof OpFilter) { OpFilter f = (OpFilter)op ; exprs.addAll(f.getExprs()) ; op = f.getSubOp() ; } return new OpFilter(exprs, op) ; } public ExprList getExprs() { return expressions ; } @Override public String getName() { return Tags.tagFilter ; } @Override public Op apply(Transform transform, Op subOp) { return transform.transform(this, subOp) ; } @Override public void visit(OpVisitor opVisitor) { opVisitor.visit(this) ; } @Override public Op1 copy(Op subOp) { return new OpFilter(expressions, subOp) ; } @Override public int hashCode() { return expressions.hashCode() ; } @Override public boolean equalTo(Op other, NodeIsomorphismMap labelMap) { if ( ! (other instanceof OpFilter) ) return false ; OpFilter opFilter = (OpFilter)other ; if ( ! expressions.equals(opFilter.expressions) ) return false ; return getSubOp().equalTo(opFilter.getSubOp(), labelMap) ; } }