|
@@ -0,0 +1,153 @@
|
|
|
+/*
|
|
|
+ * 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.index.fielddata.ScriptDocValues;
|
|
|
+import org.elasticsearch.index.fielddata.SortedNumericDoubleValues;
|
|
|
+
|
|
|
+import java.io.IOException;
|
|
|
+import java.util.ArrayList;
|
|
|
+import java.util.Collections;
|
|
|
+import java.util.Iterator;
|
|
|
+import java.util.List;
|
|
|
+import java.util.NoSuchElementException;
|
|
|
+
|
|
|
+public class HalfFloatDocValuesField implements DocValuesField<Float>, ScriptDocValues.Supplier<Double> {
|
|
|
+
|
|
|
+ protected final SortedNumericDoubleValues input;
|
|
|
+ protected final String name;
|
|
|
+
|
|
|
+ protected double[] values = new double[0];
|
|
|
+ protected int count;
|
|
|
+
|
|
|
+ private ScriptDocValues.Doubles doubles = null;
|
|
|
+
|
|
|
+ public HalfFloatDocValuesField(SortedNumericDoubleValues input, String name) {
|
|
|
+ this.input = input;
|
|
|
+ this.name = name;
|
|
|
+ }
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public void setNextDocId(int docId) throws IOException {
|
|
|
+ if (input.advanceExact(docId)) {
|
|
|
+ resize(input.docValueCount());
|
|
|
+ for (int i = 0; i < count; i++) {
|
|
|
+ values[i] = input.nextValue();
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ resize(0);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ protected void resize(int newSize) {
|
|
|
+ count = newSize;
|
|
|
+
|
|
|
+ assert count >= 0 : "size must be positive (got " + count + "): likely integer overflow?";
|
|
|
+ values = ArrayUtil.grow(values, count);
|
|
|
+ }
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public ScriptDocValues<Double> getScriptDocValues() {
|
|
|
+ if (doubles == null) {
|
|
|
+ doubles = new ScriptDocValues.Doubles(this);
|
|
|
+ }
|
|
|
+
|
|
|
+ return doubles;
|
|
|
+ }
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public Double getInternal(int index) {
|
|
|
+ return values[index];
|
|
|
+ }
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public String getName() {
|
|
|
+ return name;
|
|
|
+ }
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public boolean isEmpty() {
|
|
|
+ return count == 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public int size() {
|
|
|
+ return count;
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Does a downcast for defaultValue from a double to a float
|
|
|
+ * to allow users to avoid explicit casting.
|
|
|
+ */
|
|
|
+ public float get(double defaultValue) {
|
|
|
+ return get(0, defaultValue);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Does a downcast for defaultValue from a double to a float
|
|
|
+ * to allow users to avoid explicit casting.
|
|
|
+ */
|
|
|
+ public float get(int index, double defaultValue) {
|
|
|
+ if (isEmpty() || index < 0 || index >= count) {
|
|
|
+ return (float) defaultValue;
|
|
|
+ }
|
|
|
+
|
|
|
+ return (float) values[index];
|
|
|
+ }
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public Iterator<Float> iterator() {
|
|
|
+ return new Iterator<Float>() {
|
|
|
+ private int index = 0;
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public boolean hasNext() {
|
|
|
+ return index < count;
|
|
|
+ }
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public Float next() {
|
|
|
+ if (hasNext() == false) {
|
|
|
+ throw new NoSuchElementException();
|
|
|
+ }
|
|
|
+ return (float) values[index++];
|
|
|
+ }
|
|
|
+ };
|
|
|
+ }
|
|
|
+
|
|
|
+ /** Converts all the values to {@code Double} and returns them as a {@code List}. */
|
|
|
+ public List<Double> asDoubles() {
|
|
|
+ if (isEmpty()) {
|
|
|
+ return Collections.emptyList();
|
|
|
+ }
|
|
|
+
|
|
|
+ List<Double> doubleValues = new ArrayList<>(count);
|
|
|
+
|
|
|
+ for (int index = 0; index < count; ++index) {
|
|
|
+ doubleValues.add(values[index]);
|
|
|
+ }
|
|
|
+
|
|
|
+ return doubleValues;
|
|
|
+ }
|
|
|
+
|
|
|
+ /** Returns the 0th index value as a {@code double} if it exists, otherwise {@code defaultValue}. */
|
|
|
+ public double asDouble(double defaultValue) {
|
|
|
+ return asDouble(0, defaultValue);
|
|
|
+ }
|
|
|
+
|
|
|
+ /** Returns the value at {@code index} as a {@code double} if it exists, otherwise {@code defaultValue}. */
|
|
|
+ public double asDouble(int index, double defaultValue) {
|
|
|
+ if (isEmpty() || index < 0 || index >= count) {
|
|
|
+ return defaultValue;
|
|
|
+ }
|
|
|
+
|
|
|
+ return values[index];
|
|
|
+ }
|
|
|
+}
|