Skip to content

Commit

Permalink
Geometry formatters should be applied to the full array (#75177) (#75341
Browse files Browse the repository at this point in the history
)

Changes the way we fetch values from source in geo fields so we can format all data in one go.
  • Loading branch information
iverase authored Jul 14, 2021
1 parent 73a7e62 commit b2b80f7
Show file tree
Hide file tree
Showing 12 changed files with 110 additions and 123 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -35,9 +35,9 @@
import org.elasticsearch.cluster.routing.ShardsIterator;
import org.elasticsearch.cluster.service.ClusterService;
import org.elasticsearch.common.CheckedBiFunction;
import org.elasticsearch.common.geo.GeometryFormatterFactory;
import org.elasticsearch.common.xcontent.ParseField;
import org.elasticsearch.common.bytes.BytesReference;
import org.elasticsearch.common.geo.GeoFormatterFactory;
import org.elasticsearch.common.geo.GeoPoint;
import org.elasticsearch.common.inject.Inject;
import org.elasticsearch.common.io.stream.StreamInput;
Expand All @@ -52,7 +52,6 @@
import org.elasticsearch.common.xcontent.XContentHelper;
import org.elasticsearch.common.xcontent.XContentParser;
import org.elasticsearch.common.xcontent.XContentType;
import org.elasticsearch.geometry.Geometry;
import org.elasticsearch.geometry.Point;
import org.elasticsearch.index.Index;
import org.elasticsearch.index.IndexService;
Expand Down Expand Up @@ -602,12 +601,9 @@ static Response innerShardOperation(Request request, ScriptService scriptService
List<GeoPoint> points = new ArrayList<>();
geoPointFieldScript.runGeoPointForDoc(0, gp -> points.add(new GeoPoint(gp)));
// convert geo points to the standard format of the fields api
Function<Geometry, Object> format = GeoFormatterFactory.getFormatter(GeoFormatterFactory.GEOJSON);
List<Object> objects = new ArrayList<>();
for (GeoPoint gp : points) {
objects.add(format.apply(new Point(gp.getLon(), gp.getLat())));
}
return new Response(objects);
Function<List<GeoPoint>, List<Object>> format =
GeometryFormatterFactory.getFormatter(GeometryFormatterFactory.GEOJSON, p -> new Point(p.lon(), p.lat()));
return new Response(format.apply(points));
}, indexService);
} else if (scriptContext == IpFieldScript.CONTEXT) {
return prepareRamIndex(request, (context, leafReaderContext) -> {
Expand Down

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
/*
* 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.common.geo;

import org.elasticsearch.geometry.Geometry;
import org.elasticsearch.geometry.utils.WellKnownText;

import java.util.ArrayList;
import java.util.List;
import java.util.function.Function;

/**
* Output formatters supported by geometry fields.
*/
public class GeometryFormatterFactory {

public static final String GEOJSON = "geojson";
public static final String WKT = "wkt";

/**
* Returns a formatter by name
*/
public static <T> Function<List<T>, List<Object>> getFormatter(String name, Function<T, Geometry> toGeometry) {
switch (name) {
case GEOJSON:
return geometries -> {
final List<Object> objects = new ArrayList<>(geometries.size());
geometries.forEach((shape) -> objects.add(GeoJson.toMap(toGeometry.apply(shape))));
return objects;
};
case WKT:
return geometries -> {
final List<Object> objects = new ArrayList<>(geometries.size());
geometries.forEach((shape) -> objects.add(WellKnownText.toWKT(toGeometry.apply(shape))));
return objects;
};
default: throw new IllegalArgumentException("Unrecognized geometry format [" + name + "].");
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@

import org.apache.lucene.search.Query;
import org.elasticsearch.common.Explicit;
import org.elasticsearch.common.geo.GeoFormatterFactory;
import org.elasticsearch.common.geo.GeometryFormatterFactory;
import org.elasticsearch.common.xcontent.XContentParser;
import org.elasticsearch.common.xcontent.support.MapXContentParser;
import org.elasticsearch.core.CheckedConsumer;
Expand Down Expand Up @@ -52,13 +52,14 @@ public abstract void parse(
CheckedConsumer<T, IOException> consumer,
Consumer<Exception> onMalformed) throws IOException;

private void fetchFromSource(Object sourceMap, Consumer<Object> consumer, Function<T, Object> formatter) {
private void fetchFromSource(Object sourceMap, Consumer<T> consumer) {
try (XContentParser parser = MapXContentParser.wrapObject(sourceMap)) {
parse(parser, v -> consumer.accept(formatter.apply(v)), e -> {}); /* ignore malformed */
parse(parser, v -> consumer.accept(v), e -> {}); /* ignore malformed */
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}

}

public abstract static class AbstractGeometryFieldType<T> extends MappedFieldType {
Expand All @@ -80,17 +81,17 @@ public final Query termQuery(Object value, SearchExecutionContext context) {
/**
* Gets the formatter by name.
*/
protected abstract Function<T, Object> getFormatter(String format);
protected abstract Function<List<T>, List<Object>> getFormatter(String format);

@Override
public ValueFetcher valueFetcher(SearchExecutionContext context, String format) {
Function<T, Object> formatter = getFormatter(format != null ? format : GeoFormatterFactory.GEOJSON);
Function<List<T>, List<Object>> formatter = getFormatter(format != null ? format : GeometryFormatterFactory.GEOJSON);
return new ArraySourceValueFetcher(name(), context) {
@Override
protected Object parseSourceValue(Object value) {
List<Object> values = new ArrayList<>();
geometryParser.fetchFromSource(value, values::add, formatter);
return values;
final List<T> values = new ArrayList<>();
geometryParser.fetchFromSource(value, values::add);
return formatter.apply(values);
}
};
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,10 +19,10 @@
import org.elasticsearch.ElasticsearchParseException;
import org.elasticsearch.common.CheckedBiFunction;
import org.elasticsearch.common.Explicit;
import org.elasticsearch.common.geo.GeoFormatterFactory;
import org.elasticsearch.common.geo.GeoPoint;
import org.elasticsearch.common.geo.GeoShapeUtils;
import org.elasticsearch.common.geo.GeoUtils;
import org.elasticsearch.common.geo.GeometryFormatterFactory;
import org.elasticsearch.common.geo.ShapeRelation;
import org.elasticsearch.common.unit.DistanceUnit;
import org.elasticsearch.common.xcontent.XContentParser;
Expand Down Expand Up @@ -242,18 +242,17 @@ public String typeName() {
}

@Override
protected Function<GeoPoint, Object> getFormatter(String format) {
Function<Geometry, Object> formatter = GeoFormatterFactory.getFormatter(format);
return (point) -> formatter.apply(new Point(point.lon(), point.lat()));
protected Function<List<GeoPoint>, List<Object>> getFormatter(String format) {
return GeometryFormatterFactory.getFormatter(format, p -> new Point(p.lon(), p.lat()));
}

@Override
public ValueFetcher valueFetcher(SearchExecutionContext context, String format) {
if (scriptValues == null) {
return super.valueFetcher(context, format);
}
Function<GeoPoint, Object> formatter = getFormatter(format != null ? format : GeoFormatterFactory.GEOJSON);
return FieldValues.valueFetcher(scriptValues, v -> formatter.apply((GeoPoint) v), context);
Function<List<GeoPoint>, List<Object>> formatter = getFormatter(format != null ? format : GeometryFormatterFactory.GEOJSON);
return FieldValues.valueListFetcher(scriptValues, formatter, context);
}

@Override
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,8 @@
import org.apache.lucene.search.Query;
import org.elasticsearch.Version;
import org.elasticsearch.common.Explicit;
import org.elasticsearch.common.geo.GeoFormatterFactory;
import org.elasticsearch.common.geo.GeoShapeUtils;
import org.elasticsearch.common.geo.GeometryFormatterFactory;
import org.elasticsearch.common.geo.GeometryParser;
import org.elasticsearch.common.geo.ShapeRelation;
import org.elasticsearch.common.geo.Orientation;
Expand Down Expand Up @@ -140,8 +140,8 @@ public Query geoShapeQuery(Geometry shape, String fieldName, ShapeRelation relat
}

@Override
protected Function<Geometry, Object> getFormatter(String format) {
return GeoFormatterFactory.getFormatter(format);
protected Function<List<Geometry>, List<Object>> getFormatter(String format) {
return GeometryFormatterFactory.getFormatter(format, Function.identity());
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,8 @@
import org.elasticsearch.ElasticsearchParseException;
import org.elasticsearch.Version;
import org.elasticsearch.common.Explicit;
import org.elasticsearch.common.geo.GeoFormatterFactory;
import org.elasticsearch.common.geo.GeoUtils;
import org.elasticsearch.common.geo.GeometryFormatterFactory;
import org.elasticsearch.common.geo.ShapeRelation;
import org.elasticsearch.common.geo.ShapesAvailability;
import org.elasticsearch.common.geo.SpatialStrategy;
Expand Down Expand Up @@ -441,9 +441,8 @@ public PrefixTreeStrategy resolvePrefixTreeStrategy(String strategyName) {
}

@Override
protected Function<ShapeBuilder<?, ?, ?>, Object> getFormatter(String format) {
Function<Geometry, Object> formatter = GeoFormatterFactory.getFormatter(format);
return (g) -> formatter.apply(g.buildGeometry());
protected Function<List<ShapeBuilder<?, ?, ?>>, List<Object>> getFormatter(String format) {
return GeometryFormatterFactory.getFormatter(format, ShapeBuilder::buildGeometry);
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -70,4 +70,35 @@ public List<Object> fetchValues(SourceLookup lookup) {
}
};
}

/**
* Creates a {@link ValueFetcher} that fetches values from a {@link FieldValues} instance
* @param fieldValues the source of the values
* @param formatter a function to format the list values
* @param context the search execution context
* @return the value fetcher
*/
static <T> ValueFetcher valueListFetcher(FieldValues<T> fieldValues, Function<List<T>, List<Object>> formatter,
SearchExecutionContext context) {
return new ValueFetcher() {
LeafReaderContext ctx;

@Override
public void setNextReader(LeafReaderContext context) {
this.ctx = context;
}

@Override
public List<Object> fetchValues(SourceLookup lookup) {
List<T> values = new ArrayList<>();
try {
fieldValues.valuesForDoc(context.lookup(), ctx, lookup.docId(), v -> values.add(v));
} catch (Exception e) {
// ignore errors - if they exist here then they existed at index time
// and so on_script_error must have been set to `ignore`
}
return formatter.apply(values);
}
};
}
}

This file was deleted.

Original file line number Diff line number Diff line change
Expand Up @@ -15,8 +15,8 @@
import org.apache.lucene.search.Query;
import org.elasticsearch.Version;
import org.elasticsearch.common.Explicit;
import org.elasticsearch.common.geo.GeoFormatterFactory;
import org.elasticsearch.common.geo.GeoShapeUtils;
import org.elasticsearch.common.geo.GeometryFormatterFactory;
import org.elasticsearch.common.geo.GeometryParser;
import org.elasticsearch.common.geo.ShapeRelation;
import org.elasticsearch.common.geo.Orientation;
Expand Down Expand Up @@ -172,8 +172,8 @@ public Query geoShapeQuery(Geometry shape, String fieldName, ShapeRelation relat
}

@Override
protected Function<Geometry, Object> getFormatter(String format) {
return GeoFormatterFactory.getFormatter(format);
protected Function<List<Geometry>, List<Object>> getFormatter(String format) {
return GeometryFormatterFactory.getFormatter(format, Function.identity());
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
import org.apache.lucene.search.Query;
import org.elasticsearch.common.CheckedBiFunction;
import org.elasticsearch.common.Explicit;
import org.elasticsearch.common.geo.GeometryFormatterFactory;
import org.elasticsearch.common.geo.ShapeRelation;
import org.elasticsearch.common.logging.DeprecationCategory;
import org.elasticsearch.common.logging.DeprecationLogger;
Expand All @@ -25,7 +26,6 @@
import org.elasticsearch.index.mapper.MappedFieldType;
import org.elasticsearch.index.mapper.DocumentParserContext;
import org.elasticsearch.index.query.SearchExecutionContext;
import org.elasticsearch.xpack.spatial.common.CartesianFormatterFactory;
import org.elasticsearch.xpack.spatial.common.CartesianPoint;
import org.elasticsearch.xpack.spatial.index.query.ShapeQueryPointProcessor;

Expand Down Expand Up @@ -179,9 +179,8 @@ public Query shapeQuery(Geometry shape, String fieldName, ShapeRelation relation
}

@Override
protected Function<CartesianPoint, Object> getFormatter(String format) {
Function<Geometry, Object> formatter = CartesianFormatterFactory.getFormatter(format);
return (point) -> formatter.apply(new Point(point.getX(), point.getY()));
protected Function<List<CartesianPoint>, List<Object>> getFormatter(String format) {
return GeometryFormatterFactory.getFormatter(format, p -> new Point(p.getX(), p.getY()));
}
}

Expand Down
Loading

0 comments on commit b2b80f7

Please sign in to comment.