Package org.geotools.filter.text.ecql

Source Code of org.geotools.filter.text.ecql.FilterToECQL

/*
*    GeoTools - The Open Source Java GIS Toolkit
*    http://geotools.org
*
*    (C) 2006-2008, Open Source Geospatial Foundation (OSGeo)
*
*    This library is free software; you can redistribute it and/or
*    modify it under the terms of the GNU Lesser General Public
*    License as published by the Free Software Foundation;
*    version 2.1 of the License.
*
*    This library is distributed in the hope that it will be useful,
*    but WITHOUT ANY WARRANTY; without even the implied warranty of
*    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
*    Lesser General Public License for more details.
*/
package org.geotools.filter.text.ecql;

import java.util.Iterator;
import java.util.List;
import java.util.regex.Pattern;

import org.geotools.filter.text.commons.ExpressionToText;
import org.geotools.filter.text.commons.FilterToTextUtil;
import org.opengis.filter.And;
import org.opengis.filter.ExcludeFilter;
import org.opengis.filter.Filter;
import org.opengis.filter.FilterVisitor;
import org.opengis.filter.Id;
import org.opengis.filter.IncludeFilter;
import org.opengis.filter.Not;
import org.opengis.filter.Or;
import org.opengis.filter.PropertyIsBetween;
import org.opengis.filter.PropertyIsEqualTo;
import org.opengis.filter.PropertyIsGreaterThan;
import org.opengis.filter.PropertyIsGreaterThanOrEqualTo;
import org.opengis.filter.PropertyIsLessThan;
import org.opengis.filter.PropertyIsLessThanOrEqualTo;
import org.opengis.filter.PropertyIsLike;
import org.opengis.filter.PropertyIsNil;
import org.opengis.filter.PropertyIsNotEqualTo;
import org.opengis.filter.PropertyIsNull;
import org.opengis.filter.expression.Expression;
import org.opengis.filter.expression.Function;
import org.opengis.filter.expression.Literal;
import org.opengis.filter.identity.Identifier;
import org.opengis.filter.spatial.BBOX;
import org.opengis.filter.spatial.Beyond;
import org.opengis.filter.spatial.Contains;
import org.opengis.filter.spatial.Crosses;
import org.opengis.filter.spatial.DWithin;
import org.opengis.filter.spatial.Disjoint;
import org.opengis.filter.spatial.Equals;
import org.opengis.filter.spatial.Intersects;
import org.opengis.filter.spatial.Overlaps;
import org.opengis.filter.spatial.Touches;
import org.opengis.filter.spatial.Within;
import org.opengis.filter.temporal.After;
import org.opengis.filter.temporal.AnyInteracts;
import org.opengis.filter.temporal.Before;
import org.opengis.filter.temporal.Begins;
import org.opengis.filter.temporal.BegunBy;
import org.opengis.filter.temporal.During;
import org.opengis.filter.temporal.EndedBy;
import org.opengis.filter.temporal.Ends;
import org.opengis.filter.temporal.Meets;
import org.opengis.filter.temporal.MetBy;
import org.opengis.filter.temporal.OverlappedBy;
import org.opengis.filter.temporal.TContains;
import org.opengis.filter.temporal.TEquals;
import org.opengis.filter.temporal.TOverlaps;


/**
* This class is responsible to transform a filter to an ECQL predicate.
*
* @author Mauricio Pazos
*
*/
final class FilterToECQL implements FilterVisitor {
   
    /**
     * Pattern used to match Id filters for output as plain number.
     */
    private static Pattern NUMBER = Pattern.compile("[0-9]+");
   
    ExpressionToText expressionVisitor = new ExpressionToText();
   
  @Override
  public Object visitNullFilter(Object extraData) {
    throw new NullPointerException("Cannot encode null as a Filter");
  }

  @Override
  public Object visit(ExcludeFilter filter, Object extraData) {
       
        return FilterToTextUtil.buildExclude(extraData);
  }

  @Override
  public Object visit(IncludeFilter filter, Object extraData) {
        return FilterToTextUtil.buildInclude(extraData);
  }

  @Override
  public Object visit(And filter, Object extraData) {
      return FilterToTextUtil.buildBinaryLogicalOperator("AND", this, filter, extraData);
  }
 
  /**
   * builds a ecql id expression: in (id1, id2, ...)
   */
  @Override
  public Object visit(Id filter, Object extraData) {

    StringBuilder ecql = FilterToTextUtil.asStringBuilder(extraData);
    ecql.append("IN (");

    Iterator<Identifier> iter= filter.getIdentifiers().iterator();
    while(iter.hasNext()) {
      Identifier identifier = iter.next();
      String id = identifier.toString();
     
      boolean needsQuotes = !NUMBER.matcher( id ).matches();
     
      if( needsQuotes ) {
          ecql.append('\'');
      }
      ecql.append(identifier);
      if( needsQuotes ) {
                            ecql.append('\'');
                        }
     
      if(iter.hasNext()){
        ecql.append(",");
      }
    }
    ecql.append(")");
    return ecql;
  }

  /**
   * builds the Not logical operator
   */
  @Override
  public Object visit(Not filter, Object extraData) {
    return FilterToTextUtil.buildNot(this, filter, extraData);
  }


    /**
     * Builds the OR logical operator.
     *
     * This visitor checks for {@link #isInFilter(Or)} and is willing to output ECQL of the form <code>left IN (right, right, right)</code>.
     */
    @Override
    public Object visit(Or filter, Object extraData) {
        if (isInFilter(filter)) {
            return buildIN(filter, extraData);
        }
        // default to normal OR output
        return FilterToTextUtil.buildBinaryLogicalOperator("OR", this, filter, extraData);
    }

    /** Check if this is an encoding of ECQL IN */
    private boolean isInFilter(Or filter) {
        if (filter.getChildren() == null) {
            return false;
        }
        Expression left = null;
        for (Filter child : filter.getChildren()) {
            if (child instanceof PropertyIsEqualTo) {
                PropertyIsEqualTo equal = (PropertyIsEqualTo) child;
                if (left == null) {
                    left = equal.getExpression1();
                } else if (!left.equals(equal.getExpression1())) {
                    return false; // not IN
                }
            } else {
                return false; // not IN
            }
        }
        return true;
    }

    private Object buildIN(Or filter, Object extraData) {
        StringBuilder output = FilterToTextUtil.asStringBuilder(extraData);
        List<Filter> children = filter.getChildren();
        PropertyIsEqualTo first = (PropertyIsEqualTo) filter.getChildren().get(0);
        Expression left = first.getExpression1();
        left.accept(expressionVisitor,  output );
        output.append(" IN (");
        for (Iterator<Filter> i = children.iterator(); i.hasNext();) {
            PropertyIsEqualTo child = (PropertyIsEqualTo) i.next();
            Expression right = child.getExpression2();
            right.accept(expressionVisitor,  output );
            if (i.hasNext()) {
                output.append(",");
            }
        }
        output.append(")");
        return output;
    }

    /**
   * builds the BETWEEN predicate
   */
  @Override
  public Object visit(PropertyIsBetween filter, Object extraData) {
      return FilterToTextUtil.buildBetween(filter, extraData);
  }

    /**
     * Output EQUAL filter (will checks for ECQL geospatial operations).
     */
    @Override
    public Object visit(PropertyIsEqualTo filter, Object extraData) {
        StringBuilder output = FilterToTextUtil.asStringBuilder(extraData);
        if (isRelateOperation(filter)) {
            return buildRelate(filter, output);
        }
        else if (isFunctionTrue(filter,"PropertyExists",1)){
            return buildExists(filter, output);
        }
        return FilterToTextUtil.buildComparison(filter, output, "=");
    }

    /** Check if this is an encoding of ECQL geospatial operation */
    private boolean isFunctionTrue(PropertyIsEqualTo filter,String operation, int numberOfArguments) {
        if (filter.getExpression1() instanceof Function) {
            Function function = (Function) filter.getExpression1();
            List<Expression> parameters = function.getParameters();
            if (parameters == null) {
                return false;
            }
            String name = function.getName();
            if (!operation.equalsIgnoreCase(name) || parameters.size() != numberOfArguments) {
                return false;
            }
        }
        else {
            return false;
        }
        if (filter.getExpression2() instanceof Literal) {
            Literal literal = (Literal) filter.getExpression2();
            Boolean value = literal.evaluate(null, Boolean.class);
            if (value == null || value == false) {
                return false;
            }
        } else {
            return false;
        }
        return true;
    }

    /** Check if this is EXSISTS encoding of PropertyExistsFunction */
    private boolean isPropertyExistsTrue(PropertyIsEqualTo filter) {
        if( isFunctionTrue(filter,"PropertyExists",1) ){
            Function function = (Function) filter.getExpression1();
            List<Expression> parameters = function.getParameters();
            Expression arg = parameters.get(0);
            if (arg instanceof Literal) {
                Literal literal = (Literal) arg;
                Object value = literal.getValue();
                if (value instanceof String) {
                    return true;
                }
            }
        }
        return false;
    }

    private Object buildExists(PropertyIsEqualTo filter, StringBuilder output) {
        Function function = (Function) filter.getExpression1();
        List<Expression> parameters = function.getParameters();
        Literal arg = (Literal) parameters.get(0);
       
        output.append( arg.getValue() );
        output.append( " EXISTS");
        return output;
    }

    /** Check if this is an encoding of ECQL geospatial operation */
    private boolean isRelateOperation(PropertyIsEqualTo filter) {
        if( isFunctionTrue(filter,"relatePattern",3)){
            Function function = (Function) filter.getExpression1();
            List<Expression> parameters = function.getParameters();
            Expression param3 = parameters.get(2);
            if (param3 instanceof Literal) {
                Literal literal = (Literal) param3;
                Object value = literal.getValue();
                if (!(value instanceof String)) {
                    return false; // not a relate
                }
            }
        }
        else {
            return false;
        }
        return true;
    }

    private Object buildRelate(PropertyIsEqualTo filter, StringBuilder output) {
        Function operation = (Function) filter.getExpression1();
        String name = operation.getName();
        output.append( "RELATE(" );
        List<Expression> parameters = operation.getParameters();
        Expression arg1 = parameters.get(0);
        Expression arg2 = parameters.get(1);
        Literal arg3 = (Literal) parameters.get(2);
       
        arg1.accept(expressionVisitor, output );
        output.append(",");
        arg2.accept(expressionVisitor, output );
        output.append(",");
        output.append( arg3.getValue() );
        output.append( ")" );
        return output;
    }
   
    @Override
  public Object visit(PropertyIsNotEqualTo filter, Object extraData) {
      return FilterToTextUtil.buildComparison(filter, extraData, "!=");
  }

  @Override
  public Object visit(PropertyIsGreaterThan filter, Object extraData) {
        return FilterToTextUtil.buildComparison(filter, extraData, ">");
  }

  @Override
  public Object visit(PropertyIsGreaterThanOrEqualTo filter, Object extraData) {
        return FilterToTextUtil.buildComparison(filter, extraData, ">=");
  }

  @Override
  public Object visit(PropertyIsLessThan filter, Object extraData) {
        return FilterToTextUtil.buildComparison(filter, extraData, "<");
  }

  @Override
  public Object visit(PropertyIsLessThanOrEqualTo filter, Object extraData) {
        return FilterToTextUtil.buildComparison(filter, extraData, "<=");
  }

  @Override
  public Object visit(PropertyIsLike filter, Object extraData) {
      return FilterToTextUtil.buildIsLike(filter, extraData);
  }

  @Override
  public Object visit(PropertyIsNull filter, Object extraData) {
      return FilterToTextUtil.buildIsNull(filter, extraData);
  }

  @Override
  public Object visit(PropertyIsNil filter, Object extraData) {
  throw new UnsupportedOperationException("PropertyIsNil not supported");
  }

  @Override
  public Object visit(BBOX filter, Object extraData) {
      return FilterToTextUtil.buildBBOX(filter, extraData);
  }

  @Override
  public Object visit(Beyond filter, Object extraData) {
      return FilterToTextUtil.buildDistanceBufferOperation("BEYOND", filter, extraData);
  }

  @Override
  public Object visit(Contains filter, Object extraData) {
      return FilterToTextUtil.buildBinarySpatialOperator("CONTAINS", filter, extraData);
  }

  @Override
  public Object visit(Crosses filter, Object extraData) {
      return FilterToTextUtil.buildBinarySpatialOperator("CROSSES", filter, extraData);
  }

  @Override
  public Object visit(Disjoint filter, Object extraData) {
      return FilterToTextUtil.buildBinarySpatialOperator("DISJOINT", filter, extraData);
  }

  @Override
  public Object visit(DWithin filter, Object extraData) {
      return FilterToTextUtil.buildDWithin(filter, extraData);
  }

  @Override
  public Object visit(Equals filter, Object extraData) {
      return FilterToTextUtil.buildBinarySpatialOperator("EQUALS", filter, extraData);
  }

  @Override
  public Object visit(Intersects filter, Object extraData) {
      return FilterToTextUtil.buildBinarySpatialOperator("INTERSECTS", filter, extraData);
  }

  @Override
  public Object visit(Overlaps filter, Object extraData) {
      return FilterToTextUtil.buildBinarySpatialOperator("OVERLAPS", filter, extraData);
  }

  @Override
  public Object visit(Touches filter, Object extraData) {
      return FilterToTextUtil.buildBinarySpatialOperator("TOUCHES", filter, extraData);
  }

  @Override
  public Object visit(Within filter, Object extraData) {
      return FilterToTextUtil.buildBinarySpatialOperator("WITHIN", filter, extraData);
  }

  @Override
  public Object visit(After after, Object extraData) {
      return FilterToTextUtil.buildBinaryTemporalOperator("AFTER", after, extraData);
  }
  @Override
  public Object visit(Before before, Object extraData) {
      return FilterToTextUtil.buildBinaryTemporalOperator("BEFORE", before, extraData);
  }

  @Override
  public Object visit(AnyInteracts anyInteracts, Object extraData) {
    throw ecqlUnsupported("AnyInteracts");
  }


  @Override
  public Object visit(Begins begins, Object extraData) {
    throw ecqlUnsupported("Begins");
  }

  @Override
  public Object visit(BegunBy begunBy, Object extraData) {
        throw ecqlUnsupported("BegunBy");  
  }

  /**
   * New instance of unsupported exception with the name of filter
   * @param filterName filter unsupported
   * @return UnsupportedOperationException
   */
  static private UnsupportedOperationException ecqlUnsupported(final String filterName){
    return new UnsupportedOperationException("The"+ filterName + " has not an ECQL expression");
  }
 
  @Override
  public Object visit(During during, Object extraData) {
      return FilterToTextUtil.buildDuring(during, extraData);
  }

  @Override
  public Object visit(EndedBy endedBy, Object extraData) {
    throw ecqlUnsupported("EndedBy");
  }

  @Override
  public Object visit(Ends ends, Object extraData) {
    throw ecqlUnsupported("EndedBy");
  }

  @Override
  public Object visit(Meets meets, Object extraData) {
    throw ecqlUnsupported("Meets");
  }

  @Override
  public Object visit(MetBy metBy, Object extraData) {
    throw ecqlUnsupported("MetBy");
  }

  @Override
  public Object visit(OverlappedBy overlappedBy, Object extraData) {
    throw ecqlUnsupported("OverlappedBy");
  }

  @Override
  public Object visit(TContains contains, Object extraData) {
    throw ecqlUnsupported("TContains");
  }

  @Override
  public Object visit(TEquals equals, Object extraData) {
    throw ecqlUnsupported("TContains");
  }

  @Override
  public Object visit(TOverlaps contains, Object extraData) {
    throw ecqlUnsupported("TContains");
  }

}
TOP

Related Classes of org.geotools.filter.text.ecql.FilterToECQL

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.