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

org.janusgraph.diskstorage.lucene.NumericTranslationQueryParser Maven / Gradle / Ivy

// Copyright 2019 JanusGraph Authors
//
// 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.janusgraph.diskstorage.lucene;

import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.document.DoublePoint;
import org.apache.lucene.document.IntPoint;
import org.apache.lucene.document.LongPoint;
import org.apache.lucene.index.Term;
import org.apache.lucene.queryparser.classic.ParseException;
import org.apache.lucene.queryparser.classic.QueryParser;
import org.apache.lucene.search.Query;
import org.apache.tinkerpop.shaded.jackson.databind.util.StdDateFormat;
import org.janusgraph.diskstorage.indexing.KeyInformation;
import org.janusgraph.graphdb.database.serialize.AttributeUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.time.Instant;
import java.util.Date;

public class NumericTranslationQueryParser extends QueryParser {
    private static final Logger log = LoggerFactory.getLogger(NumericTranslationQueryParser.class);
    private final KeyInformation.StoreRetriever storeRetriever;

    public NumericTranslationQueryParser(KeyInformation.StoreRetriever storeRetriever, String field, Analyzer analyzer) {
        super(field, analyzer);
        this.storeRetriever = storeRetriever;
    }

    @Override
    protected Query newRangeQuery(final String field, final String start, final String end, final boolean startInclusive,
                                  final boolean endInclusive) {
        Class dataType = getKeyDataType(field);
        if (isPossibleRangeQuery(dataType)) {
            try {
                return buildNumericRangeQuery(field, dataType, start, end, startInclusive, endInclusive);
            } catch (NumberFormatException e) {
                printNumberFormatException(field, dataType, e);
            }
        }

        return super.newRangeQuery(field, start, end, startInclusive, endInclusive);
    }

    @Override
    protected Query newFieldQuery(final Analyzer analyzer, final String field, final String queryText, final boolean quoted) throws ParseException {
        Class dataType = getKeyDataType(field);
        if (isPossibleRangeQuery(dataType) || Boolean.class.equals(dataType)) {
            try {
                return buildNumericQuery(field, queryText, dataType);
            } catch (NumberFormatException e) {
                printNumberFormatException(field, dataType, e);
            }
        }
        return super.newFieldQuery(analyzer, field, queryText, quoted);
    }

    @Override
    protected Query newWildcardQuery(final Term t) {
        if (t.field() == null) {
            return super.newWildcardQuery(t);
        }
        Class dataType = getKeyDataType(t.field());
        if (isPossibleRangeQuery(dataType) || Boolean.class.equals(dataType)) {
            try {
                return buildNumericQuery(t.field(), t.text(), dataType);
            } catch (NumberFormatException e) {
                printNumberFormatException(t.field(), dataType, e);
            }
        }

        return super.newWildcardQuery(t);
    }

    private Query buildNumericRangeQuery(final String field, final Class type, String start, String end, final boolean includeLower,
                                         final boolean includeUpper) {
        if (AttributeUtils.isWholeNumber(type) || isTemporalType(type)) {
            long min;
            long max;
            if (isTemporalType(type)) {
                min = isMatchAll(start) ? Long.MIN_VALUE : parseDate(start).getTime();
                max = isMatchAll(end) ? Long.MAX_VALUE : parseDate(end).getTime();
            } else {
                min = isMatchAll(start) ? Long.MIN_VALUE : Long.parseLong(start);
                max = isMatchAll(end) ? Long.MAX_VALUE : Long.parseLong(end);
            }
            if (!includeLower) {
                min = Math.addExact(min, 1);
            }
            if (!includeUpper) {
                max = Math.addExact(max, -1);
            }
            return LongPoint.newRangeQuery(field, min, max);
        } else {
            double min = isMatchAll(start) ? Double.NEGATIVE_INFINITY : Double.parseDouble(start);
            double max = isMatchAll(end) ? Double.POSITIVE_INFINITY : Double.parseDouble(end);
            if (!includeLower) {
                min = DoublePoint.nextUp(min);
            }
            if (!includeUpper) {
                max = DoublePoint.nextDown(max);
            }
            return DoublePoint.newRangeQuery(field, min, max);
        }
    }

    private Query buildNumericQuery(final String field, final String value, Class type) {
        Query query;
        if (AttributeUtils.isWholeNumber(type) || isTemporalType(type)) {
            if (isMatchAll(value)) {
                query = LongPoint.newRangeQuery(field, Long.MIN_VALUE, Long.MAX_VALUE);
            } else {
                if (isTemporalType(type)) {
                    query = LongPoint.newExactQuery(field, parseDate(value).getTime());
                } else {
                    query = LongPoint.newExactQuery(field, Long.parseLong(value));
                }
            }
        } else if (Boolean.class.isAssignableFrom(type)) {
            if (isMatchAll(value)) {
                return IntPoint.newRangeQuery(field, Integer.MIN_VALUE, Integer.MAX_VALUE);
            } else {
                return IntPoint.newExactQuery(field, Boolean.parseBoolean(value) ? 1 : 0);
            }
        } else {
            if (isMatchAll(value)) {
                query = DoublePoint.newRangeQuery(field, Double.NEGATIVE_INFINITY, Double.POSITIVE_INFINITY);
            } else {
                query = DoublePoint.newExactQuery(field, Double.parseDouble(value));
            }
        }
        return query;
    }

    private Class getKeyDataType(final String field) {
        KeyInformation keyInformation = storeRetriever.get(field);
        if (keyInformation == null) {
            log.warn(String.format("Could not find key information for: %s", field));
            return null;
        }
        return keyInformation.getDataType();
    }

    private Date parseDate(String value) {
        try {
            return StdDateFormat.instance.parse(value);
        } catch (java.text.ParseException e) {
            throw new RuntimeException(e);
        }
    }

    private boolean isPossibleRangeQuery(final Class dataType) {
        if (dataType == null) {
            return false;
        }
        return Number.class.isAssignableFrom(dataType) || isTemporalType(dataType);
    }

    private boolean isTemporalType(final Class dataType) {
        return Date.class.equals(dataType) || Instant.class.equals(dataType);
    }

    private boolean isMatchAll(final String value) {
        return value == null || "*".equals(value);
    }

    private void printNumberFormatException(final String field, final Class dataType, final NumberFormatException e) {
        log.warn("Expected Number type for " + field + ":" + dataType, e);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy