
org.elasticsearch.script.field.PointDocValuesField Maven / Gradle / Ivy
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/
package org.elasticsearch.script.field;
import org.apache.lucene.util.ArrayUtil;
import org.elasticsearch.common.geo.BoundingBox;
import org.elasticsearch.common.geo.SpatialPoint;
import org.elasticsearch.index.fielddata.MultiPointValues;
import org.elasticsearch.index.fielddata.ScriptDocValues;
import java.io.IOException;
import java.util.Iterator;
import java.util.NoSuchElementException;
import java.util.function.Supplier;
public abstract class PointDocValuesField extends AbstractScriptFieldFactory
implements
Field,
DocValuesScriptFieldFactory,
ScriptDocValues.GeometrySupplier {
protected final MultiPointValues input;
protected final String name;
protected T[] values;
protected int count;
private final Supplier pointMaker;
protected final T centroid;
protected final BoundingBox boundingBox;
private int labelIndex = 0;
public PointDocValuesField(MultiPointValues input, String name, Supplier pointMaker, BoundingBox boundingBox, T[] values) {
this.input = input;
this.name = name;
this.pointMaker = pointMaker;
this.centroid = pointMaker.get();
this.boundingBox = boundingBox;
this.values = values;
}
@Override
public void setNextDocId(int docId) throws IOException {
if (input.advanceExact(docId)) {
resize(input.docValueCount());
if (count == 1) {
setSingleValue();
} else {
setMultiValue();
}
} else {
resize(0);
}
}
private void resize(int newSize) {
count = newSize;
if (newSize > values.length) {
int oldLength = values.length;
values = ArrayUtil.grow(values, count);
for (int i = oldLength; i < values.length; ++i) {
values[i] = pointMaker.get();
}
}
}
protected abstract void resetPointAt(int i, T point);
protected abstract void resetCentroidAndBounds(T centroid, T topLeft, T bottomRight);
protected abstract double getXFrom(T point);
protected abstract double getYFrom(T point);
protected abstract T pointOf(double x, double y);
protected abstract double planeDistance(double x1, double y1, T point);
private void setSingleValue() throws IOException {
T point = input.nextValue();
resetPointAt(0, point);
resetCentroidAndBounds(point, point, point);
labelIndex = 0;
}
private void setMultiValue() throws IOException {
double centroidY = 0;
double centroidX = 0;
labelIndex = 0;
double maxX = Double.NEGATIVE_INFINITY;
double minX = Double.POSITIVE_INFINITY;
double maxY = Double.NEGATIVE_INFINITY;
double minY = Double.POSITIVE_INFINITY;
for (int i = 0; i < count; i++) {
T point = input.nextValue();
resetPointAt(i, point);
centroidX += getXFrom(point);
centroidY += getYFrom(point);
maxX = Math.max(maxX, getXFrom(values[i]));
minX = Math.min(minX, getXFrom(values[i]));
maxY = Math.max(maxY, getYFrom(values[i]));
minY = Math.min(minY, getYFrom(values[i]));
labelIndex = closestPoint(labelIndex, i, (minX + maxX) / 2, (minY + maxY) / 2);
}
resetCentroidAndBounds(pointOf(centroidX, centroidY), pointOf(minX, maxY), pointOf(maxX, minY));
}
private int closestPoint(int a, int b, double x, double y) {
if (a == b) {
return a;
}
double distA = planeDistance(x, y, values[a]);
double distB = planeDistance(x, y, values[b]);
return distA < distB ? a : b;
}
@Override
public T getInternal(int index) {
return values[index];
}
// maintain bwc by making centroid available to ScriptDocValues.GeoPoints
@Override
public T getInternalCentroid() {
return centroid;
}
// maintain bwc by making bounding box available to ScriptDocValues.GeoPoints
@Override
public BoundingBox getInternalBoundingBox() {
return boundingBox;
}
@Override
public T getInternalLabelPosition() {
return values[labelIndex];
}
@Override
public String getName() {
return name;
}
@Override
public boolean isEmpty() {
return count == 0;
}
@Override
public int size() {
return count;
}
public T get(T defaultValue) {
return get(0, defaultValue);
}
public T get(int index, T defaultValue) {
if (isEmpty() || index < 0 || index >= count) {
return defaultValue;
}
return values[index];
}
@Override
public Iterator iterator() {
return new Iterator<>() {
private int index = 0;
@Override
public boolean hasNext() {
return index < count;
}
@Override
public T next() {
if (hasNext() == false) {
throw new NoSuchElementException();
}
return values[index++];
}
};
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy