/** * Copyright 2014 National University of Ireland, Galway. * * This file is part of the SIREn project. Project and contact information: * * https://github.com/rdelbru/SIREn * * Licensed 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.sindice.siren.qparser.keyword.processors; import java.util.List; import java.util.Map; import org.apache.lucene.analysis.Analyzer; import org.apache.lucene.queryparser.flexible.core.QueryNodeException; import org.apache.lucene.queryparser.flexible.core.QueryNodeParseException; import org.apache.lucene.queryparser.flexible.core.messages.QueryParserMessages; import org.apache.lucene.queryparser.flexible.core.nodes.FieldQueryNode; import org.apache.lucene.queryparser.flexible.core.nodes.QueryNode; import org.apache.lucene.queryparser.flexible.core.nodes.RangeQueryNode; import org.apache.lucene.queryparser.flexible.core.processors.QueryNodeProcessorImpl; import org.apache.lucene.queryparser.flexible.messages.MessageImpl; import org.apache.lucene.queryparser.flexible.standard.nodes.NumericQueryNode; import org.apache.lucene.queryparser.flexible.standard.processors.NumericQueryNodeProcessor; import org.sindice.siren.analysis.NumericAnalyzer; import org.sindice.siren.qparser.keyword.config.KeywordQueryConfigHandler.KeywordConfigurationKeys; import org.sindice.siren.qparser.keyword.nodes.DatatypeQueryNode; import org.sindice.siren.qparser.keyword.nodes.NodeNumericQueryNode; import org.sindice.siren.qparser.keyword.nodes.NodeNumericRangeQueryNode; import org.sindice.siren.util.ReusableCharArrayReader; /** * This processor is used to convert {@link FieldQueryNode}s to * {@link NodeNumericRangeQueryNode}s. * * <p> * * It gets the numeric {@link Analyzer} that was previously tagged with * {@link DatatypeQueryNode#DATATYPE_TAGID} * in {@link DatatypeQueryNodeProcessor}. If set and is a {@link NumericAnalyzer}, * it considers this {@link FieldQueryNode} to be a numeric query and converts * it to {@link NodeNumericRangeQueryNode} with upper and lower inclusive and * lower and upper equals to the value represented by the {@link FieldQueryNode} * converted to {@link Number}. It means that <b>1^^<int></b> is converted * to <b>[1 TO 1]^^<int></b>. * * <p> * * The datatype of the value does not depend on a field as in Lucene (see {@link DatatypeQueryNodeProcessor}). * * <p> * * Note that {@link FieldQueryNode}s children of a * {@link RangeQueryNode} are ignored. * * <p> * * Copied from {@link NumericQueryNodeProcessor} and modified for the * SIREn use case. * * @see KeywordConfigurationKeys#DATATYPES_ANALYZERS * @see FieldQueryNode * @see NumericQueryNode * @see NodeNumericRangeQueryNode */ public class NodeNumericQueryNodeProcessor extends QueryNodeProcessorImpl { /** * Constructs an empty {@link NodeNumericQueryNodeProcessor} object. */ public NodeNumericQueryNodeProcessor() { } @Override protected QueryNode postProcessNode(final QueryNode node) throws QueryNodeException { if (node instanceof FieldQueryNode && !(node.getParent() instanceof RangeQueryNode)) { final FieldQueryNode fieldNode = (FieldQueryNode) node; final Map<String, Analyzer> dts = this.getQueryConfigHandler().get(KeywordConfigurationKeys.DATATYPES_ANALYZERS); final Analyzer analyzer = dts.get(node.getTag(DatatypeQueryNode.DATATYPE_TAGID)); if (analyzer instanceof NumericAnalyzer) { final NumericAnalyzer na = (NumericAnalyzer) analyzer; final char[] text = fieldNode.getTextAsString().toCharArray(); final ReusableCharArrayReader textReader = new ReusableCharArrayReader(text); final Number number; try { number = na.getNumericParser().parse(textReader); } catch (final Exception e) { throw new QueryNodeParseException(new MessageImpl(QueryParserMessages.COULD_NOT_PARSE_NUMBER, text), e); } final CharSequence field = fieldNode.getField(); final NodeNumericQueryNode lowerNode = new NodeNumericQueryNode(field, number); final NodeNumericQueryNode upperNode = new NodeNumericQueryNode(field, number); return new NodeNumericRangeQueryNode(lowerNode, upperNode, true, true, na); } } return node; } @Override protected QueryNode preProcessNode(final QueryNode node) throws QueryNodeException { return node; } @Override protected List<QueryNode> setChildrenOrder(final List<QueryNode> children) throws QueryNodeException { return children; } }