[7.x-backport] Centralize BoundingBox logic to a dedicated class (#50469)
Both geo_bounding_box query and geo_bounds aggregation have a very similar definition of a "bounding box". A lot of this logic (serialization, xcontent-parsing, etc) can be centralized instead of having separated efforts to do the same things
This commit is contained in:
parent
694b119f0a
commit
bed121efaf
|
@ -0,0 +1,229 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elasticsearch under one or more contributor
|
||||||
|
* license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright
|
||||||
|
* ownership. Elasticsearch licenses this file to you 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.elasticsearch.common.geo;
|
||||||
|
|
||||||
|
import org.elasticsearch.ElasticsearchParseException;
|
||||||
|
import org.elasticsearch.common.ParseField;
|
||||||
|
import org.elasticsearch.common.io.stream.StreamInput;
|
||||||
|
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||||
|
import org.elasticsearch.common.io.stream.Writeable;
|
||||||
|
import org.elasticsearch.common.xcontent.ToXContentObject;
|
||||||
|
import org.elasticsearch.common.xcontent.XContentBuilder;
|
||||||
|
import org.elasticsearch.common.xcontent.XContentParser;
|
||||||
|
import org.elasticsearch.geometry.Geometry;
|
||||||
|
import org.elasticsearch.geometry.Rectangle;
|
||||||
|
import org.elasticsearch.geometry.ShapeType;
|
||||||
|
import org.elasticsearch.geometry.utils.StandardValidator;
|
||||||
|
import org.elasticsearch.geometry.utils.WellKnownText;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.text.ParseException;
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A class representing a Geo-Bounding-Box for use by Geo queries and aggregations
|
||||||
|
* that deal with extents/rectangles representing rectangular areas of interest.
|
||||||
|
*/
|
||||||
|
public class GeoBoundingBox implements ToXContentObject, Writeable {
|
||||||
|
private static final WellKnownText WKT_PARSER = new WellKnownText(true, new StandardValidator(true));
|
||||||
|
static final ParseField TOP_RIGHT_FIELD = new ParseField("top_right");
|
||||||
|
static final ParseField BOTTOM_LEFT_FIELD = new ParseField("bottom_left");
|
||||||
|
static final ParseField TOP_FIELD = new ParseField("top");
|
||||||
|
static final ParseField BOTTOM_FIELD = new ParseField("bottom");
|
||||||
|
static final ParseField LEFT_FIELD = new ParseField("left");
|
||||||
|
static final ParseField RIGHT_FIELD = new ParseField("right");
|
||||||
|
static final ParseField WKT_FIELD = new ParseField("wkt");
|
||||||
|
public static final ParseField BOUNDS_FIELD = new ParseField("bounds");
|
||||||
|
public static final ParseField LAT_FIELD = new ParseField("lat");
|
||||||
|
public static final ParseField LON_FIELD = new ParseField("lon");
|
||||||
|
public static final ParseField TOP_LEFT_FIELD = new ParseField("top_left");
|
||||||
|
public static final ParseField BOTTOM_RIGHT_FIELD = new ParseField("bottom_right");
|
||||||
|
|
||||||
|
private final GeoPoint topLeft;
|
||||||
|
private final GeoPoint bottomRight;
|
||||||
|
|
||||||
|
public GeoBoundingBox(GeoPoint topLeft, GeoPoint bottomRight) {
|
||||||
|
this.topLeft = topLeft;
|
||||||
|
this.bottomRight = bottomRight;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GeoBoundingBox(StreamInput input) throws IOException {
|
||||||
|
this.topLeft = input.readGeoPoint();
|
||||||
|
this.bottomRight = input.readGeoPoint();
|
||||||
|
}
|
||||||
|
|
||||||
|
public GeoPoint topLeft() {
|
||||||
|
return topLeft;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GeoPoint bottomRight() {
|
||||||
|
return bottomRight;
|
||||||
|
}
|
||||||
|
|
||||||
|
public double top() {
|
||||||
|
return topLeft.lat();
|
||||||
|
}
|
||||||
|
|
||||||
|
public double bottom() {
|
||||||
|
return bottomRight.lat();
|
||||||
|
}
|
||||||
|
|
||||||
|
public double left() {
|
||||||
|
return topLeft.lon();
|
||||||
|
}
|
||||||
|
|
||||||
|
public double right() {
|
||||||
|
return bottomRight.lon();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
|
||||||
|
builder.startObject(BOUNDS_FIELD.getPreferredName());
|
||||||
|
toXContentFragment(builder, true);
|
||||||
|
builder.endObject();
|
||||||
|
return builder;
|
||||||
|
}
|
||||||
|
|
||||||
|
public XContentBuilder toXContentFragment(XContentBuilder builder, boolean buildLatLonFields) throws IOException {
|
||||||
|
if (buildLatLonFields) {
|
||||||
|
builder.startObject(TOP_LEFT_FIELD.getPreferredName());
|
||||||
|
builder.field(LAT_FIELD.getPreferredName(), topLeft.lat());
|
||||||
|
builder.field(LON_FIELD.getPreferredName(), topLeft.lon());
|
||||||
|
builder.endObject();
|
||||||
|
} else {
|
||||||
|
builder.array(TOP_LEFT_FIELD.getPreferredName(), topLeft.lon(), topLeft.lat());
|
||||||
|
}
|
||||||
|
if (buildLatLonFields) {
|
||||||
|
builder.startObject(BOTTOM_RIGHT_FIELD.getPreferredName());
|
||||||
|
builder.field(LAT_FIELD.getPreferredName(), bottomRight.lat());
|
||||||
|
builder.field(LON_FIELD.getPreferredName(), bottomRight.lon());
|
||||||
|
builder.endObject();
|
||||||
|
} else {
|
||||||
|
builder.array(BOTTOM_RIGHT_FIELD.getPreferredName(), bottomRight.lon(), bottomRight.lat());
|
||||||
|
}
|
||||||
|
return builder;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeTo(StreamOutput out) throws IOException {
|
||||||
|
out.writeGeoPoint(topLeft);
|
||||||
|
out.writeGeoPoint(bottomRight);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
if (this == o) return true;
|
||||||
|
if (o == null || getClass() != o.getClass()) return false;
|
||||||
|
GeoBoundingBox that = (GeoBoundingBox) o;
|
||||||
|
return topLeft.equals(that.topLeft) &&
|
||||||
|
bottomRight.equals(that.bottomRight);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return Objects.hash(topLeft, bottomRight);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "BBOX (" + topLeft.lon() + ", " + bottomRight.lon() + ", " + topLeft.lat() + ", " + bottomRight.lat() + ")";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses the bounding box and returns bottom, top, left, right coordinates
|
||||||
|
*/
|
||||||
|
public static GeoBoundingBox parseBoundingBox(XContentParser parser) throws IOException, ElasticsearchParseException {
|
||||||
|
XContentParser.Token token = parser.currentToken();
|
||||||
|
if (token != XContentParser.Token.START_OBJECT) {
|
||||||
|
throw new ElasticsearchParseException("failed to parse bounding box. Expected start object but found [{}]", token);
|
||||||
|
}
|
||||||
|
|
||||||
|
double top = Double.NaN;
|
||||||
|
double bottom = Double.NaN;
|
||||||
|
double left = Double.NaN;
|
||||||
|
double right = Double.NaN;
|
||||||
|
|
||||||
|
String currentFieldName;
|
||||||
|
GeoPoint sparse = new GeoPoint();
|
||||||
|
Rectangle envelope = null;
|
||||||
|
|
||||||
|
while ((token = parser.nextToken()) != XContentParser.Token.END_OBJECT) {
|
||||||
|
if (token == XContentParser.Token.FIELD_NAME) {
|
||||||
|
currentFieldName = parser.currentName();
|
||||||
|
token = parser.nextToken();
|
||||||
|
if (WKT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
try {
|
||||||
|
Geometry geometry = WKT_PARSER.fromWKT(parser.text());
|
||||||
|
if (ShapeType.ENVELOPE.equals(geometry.type()) == false) {
|
||||||
|
throw new ElasticsearchParseException("failed to parse WKT bounding box. ["
|
||||||
|
+ geometry.type() + "] found. expected [" + ShapeType.ENVELOPE + "]");
|
||||||
|
}
|
||||||
|
envelope = (Rectangle) geometry;
|
||||||
|
} catch (ParseException|IllegalArgumentException e) {
|
||||||
|
throw new ElasticsearchParseException("failed to parse WKT bounding box", e);
|
||||||
|
}
|
||||||
|
} else if (TOP_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
top = parser.doubleValue();
|
||||||
|
} else if (BOTTOM_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
bottom = parser.doubleValue();
|
||||||
|
} else if (LEFT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
left = parser.doubleValue();
|
||||||
|
} else if (RIGHT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
right = parser.doubleValue();
|
||||||
|
} else {
|
||||||
|
if (TOP_LEFT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
GeoUtils.parseGeoPoint(parser, sparse, false, GeoUtils.EffectivePoint.TOP_LEFT);
|
||||||
|
top = sparse.getLat();
|
||||||
|
left = sparse.getLon();
|
||||||
|
} else if (BOTTOM_RIGHT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
GeoUtils.parseGeoPoint(parser, sparse, false, GeoUtils.EffectivePoint.BOTTOM_RIGHT);
|
||||||
|
bottom = sparse.getLat();
|
||||||
|
right = sparse.getLon();
|
||||||
|
} else if (TOP_RIGHT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
GeoUtils.parseGeoPoint(parser, sparse, false, GeoUtils.EffectivePoint.TOP_RIGHT);
|
||||||
|
top = sparse.getLat();
|
||||||
|
right = sparse.getLon();
|
||||||
|
} else if (BOTTOM_LEFT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
||||||
|
GeoUtils.parseGeoPoint(parser, sparse, false, GeoUtils.EffectivePoint.BOTTOM_LEFT);
|
||||||
|
bottom = sparse.getLat();
|
||||||
|
left = sparse.getLon();
|
||||||
|
} else {
|
||||||
|
throw new ElasticsearchParseException("failed to parse bounding box. unexpected field [{}]", currentFieldName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
throw new ElasticsearchParseException("failed to parse bounding box. field name expected but [{}] found", token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (envelope != null) {
|
||||||
|
if (Double.isNaN(top) == false || Double.isNaN(bottom) == false || Double.isNaN(left) == false ||
|
||||||
|
Double.isNaN(right) == false) {
|
||||||
|
throw new ElasticsearchParseException("failed to parse bounding box. Conflicting definition found "
|
||||||
|
+ "using well-known text and explicit corners.");
|
||||||
|
}
|
||||||
|
GeoPoint topLeft = new GeoPoint(envelope.getMaxLat(), envelope.getMinLon());
|
||||||
|
GeoPoint bottomRight = new GeoPoint(envelope.getMinLat(), envelope.getMaxLon());
|
||||||
|
return new GeoBoundingBox(topLeft, bottomRight);
|
||||||
|
}
|
||||||
|
GeoPoint topLeft = new GeoPoint(top, left);
|
||||||
|
GeoPoint bottomRight = new GeoPoint(bottom, right);
|
||||||
|
return new GeoBoundingBox(topLeft, bottomRight);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -21,7 +21,6 @@ package org.elasticsearch.index.query;
|
||||||
|
|
||||||
import org.apache.lucene.document.LatLonDocValuesField;
|
import org.apache.lucene.document.LatLonDocValuesField;
|
||||||
import org.apache.lucene.document.LatLonPoint;
|
import org.apache.lucene.document.LatLonPoint;
|
||||||
//import org.apache.lucene.geo.Rectangle;
|
|
||||||
import org.apache.lucene.search.IndexOrDocValuesQuery;
|
import org.apache.lucene.search.IndexOrDocValuesQuery;
|
||||||
import org.apache.lucene.search.MatchNoDocsQuery;
|
import org.apache.lucene.search.MatchNoDocsQuery;
|
||||||
import org.apache.lucene.search.Query;
|
import org.apache.lucene.search.Query;
|
||||||
|
@ -29,11 +28,9 @@ import org.elasticsearch.ElasticsearchParseException;
|
||||||
import org.elasticsearch.common.Numbers;
|
import org.elasticsearch.common.Numbers;
|
||||||
import org.elasticsearch.common.ParseField;
|
import org.elasticsearch.common.ParseField;
|
||||||
import org.elasticsearch.common.ParsingException;
|
import org.elasticsearch.common.ParsingException;
|
||||||
|
import org.elasticsearch.common.geo.GeoBoundingBox;
|
||||||
import org.elasticsearch.common.geo.GeoPoint;
|
import org.elasticsearch.common.geo.GeoPoint;
|
||||||
import org.elasticsearch.common.geo.GeoShapeType;
|
|
||||||
import org.elasticsearch.common.geo.GeoUtils;
|
import org.elasticsearch.common.geo.GeoUtils;
|
||||||
import org.elasticsearch.common.geo.builders.EnvelopeBuilder;
|
|
||||||
import org.elasticsearch.common.geo.parsers.GeoWKTParser;
|
|
||||||
import org.elasticsearch.common.io.stream.StreamInput;
|
import org.elasticsearch.common.io.stream.StreamInput;
|
||||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||||
import org.elasticsearch.common.xcontent.XContentBuilder;
|
import org.elasticsearch.common.xcontent.XContentBuilder;
|
||||||
|
@ -66,24 +63,12 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
|
|
||||||
private static final ParseField TYPE_FIELD = new ParseField("type");
|
private static final ParseField TYPE_FIELD = new ParseField("type");
|
||||||
private static final ParseField VALIDATION_METHOD_FIELD = new ParseField("validation_method");
|
private static final ParseField VALIDATION_METHOD_FIELD = new ParseField("validation_method");
|
||||||
private static final ParseField TOP_FIELD = new ParseField("top");
|
|
||||||
private static final ParseField BOTTOM_FIELD = new ParseField("bottom");
|
|
||||||
private static final ParseField LEFT_FIELD = new ParseField("left");
|
|
||||||
private static final ParseField RIGHT_FIELD = new ParseField("right");
|
|
||||||
private static final ParseField TOP_LEFT_FIELD = new ParseField("top_left");
|
|
||||||
private static final ParseField BOTTOM_RIGHT_FIELD = new ParseField("bottom_right");
|
|
||||||
private static final ParseField TOP_RIGHT_FIELD = new ParseField("top_right");
|
|
||||||
private static final ParseField BOTTOM_LEFT_FIELD = new ParseField("bottom_left");
|
|
||||||
private static final ParseField IGNORE_UNMAPPED_FIELD = new ParseField("ignore_unmapped");
|
private static final ParseField IGNORE_UNMAPPED_FIELD = new ParseField("ignore_unmapped");
|
||||||
private static final ParseField WKT_FIELD = new ParseField("wkt");
|
|
||||||
|
|
||||||
|
|
||||||
/** Name of field holding geo coordinates to compute the bounding box on.*/
|
/** Name of field holding geo coordinates to compute the bounding box on.*/
|
||||||
private final String fieldName;
|
private final String fieldName;
|
||||||
/** Top left corner coordinates of bounding box. */
|
private GeoBoundingBox geoBoundingBox = new GeoBoundingBox(new GeoPoint(Double.NaN, Double.NaN), new GeoPoint(Double.NaN, Double.NaN));
|
||||||
private GeoPoint topLeft = new GeoPoint(Double.NaN, Double.NaN);
|
|
||||||
/** Bottom right corner coordinates of bounding box.*/
|
|
||||||
private GeoPoint bottomRight = new GeoPoint(Double.NaN, Double.NaN);
|
|
||||||
/** How to deal with incorrect coordinates.*/
|
/** How to deal with incorrect coordinates.*/
|
||||||
private GeoValidationMethod validationMethod = GeoValidationMethod.DEFAULT;
|
private GeoValidationMethod validationMethod = GeoValidationMethod.DEFAULT;
|
||||||
/** How the query should be run. */
|
/** How the query should be run. */
|
||||||
|
@ -108,8 +93,7 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
public GeoBoundingBoxQueryBuilder(StreamInput in) throws IOException {
|
public GeoBoundingBoxQueryBuilder(StreamInput in) throws IOException {
|
||||||
super(in);
|
super(in);
|
||||||
fieldName = in.readString();
|
fieldName = in.readString();
|
||||||
topLeft = in.readGeoPoint();
|
geoBoundingBox = new GeoBoundingBox(in);
|
||||||
bottomRight = in.readGeoPoint();
|
|
||||||
type = GeoExecType.readFromStream(in);
|
type = GeoExecType.readFromStream(in);
|
||||||
validationMethod = GeoValidationMethod.readFromStream(in);
|
validationMethod = GeoValidationMethod.readFromStream(in);
|
||||||
ignoreUnmapped = in.readBoolean();
|
ignoreUnmapped = in.readBoolean();
|
||||||
|
@ -118,8 +102,7 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
@Override
|
@Override
|
||||||
protected void doWriteTo(StreamOutput out) throws IOException {
|
protected void doWriteTo(StreamOutput out) throws IOException {
|
||||||
out.writeString(fieldName);
|
out.writeString(fieldName);
|
||||||
out.writeGeoPoint(topLeft);
|
geoBoundingBox.writeTo(out);
|
||||||
out.writeGeoPoint(bottomRight);
|
|
||||||
type.writeTo(out);
|
type.writeTo(out);
|
||||||
validationMethod.writeTo(out);
|
validationMethod.writeTo(out);
|
||||||
out.writeBoolean(ignoreUnmapped);
|
out.writeBoolean(ignoreUnmapped);
|
||||||
|
@ -162,8 +145,8 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
// we do not check longitudes as the query generation code can deal with flipped left/right values
|
// we do not check longitudes as the query generation code can deal with flipped left/right values
|
||||||
}
|
}
|
||||||
|
|
||||||
topLeft.reset(top, left);
|
geoBoundingBox.topLeft().reset(top, left);
|
||||||
bottomRight.reset(bottom, right);
|
geoBoundingBox.bottomRight().reset(bottom, right);
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -197,12 +180,12 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
|
|
||||||
/** Returns the top left corner of the bounding box. */
|
/** Returns the top left corner of the bounding box. */
|
||||||
public GeoPoint topLeft() {
|
public GeoPoint topLeft() {
|
||||||
return topLeft;
|
return geoBoundingBox.topLeft();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns the bottom right corner of the bounding box. */
|
/** Returns the bottom right corner of the bounding box. */
|
||||||
public GeoPoint bottomRight() {
|
public GeoPoint bottomRight() {
|
||||||
return bottomRight;
|
return geoBoundingBox.bottomRight();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -295,6 +278,9 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GeoPoint topLeft = geoBoundingBox.topLeft();
|
||||||
|
GeoPoint bottomRight = geoBoundingBox.bottomRight();
|
||||||
|
|
||||||
QueryValidationException validationException = null;
|
QueryValidationException validationException = null;
|
||||||
// For everything post 2.0 validate latitude and longitude unless validation was explicitly turned off
|
// For everything post 2.0 validate latitude and longitude unless validation was explicitly turned off
|
||||||
if (GeoUtils.isValidLatitude(topLeft.getLat()) == false) {
|
if (GeoUtils.isValidLatitude(topLeft.getLat()) == false) {
|
||||||
|
@ -335,8 +321,8 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
throw new QueryShardException(context, "couldn't validate latitude/ longitude values", exception);
|
throw new QueryShardException(context, "couldn't validate latitude/ longitude values", exception);
|
||||||
}
|
}
|
||||||
|
|
||||||
GeoPoint luceneTopLeft = new GeoPoint(topLeft);
|
GeoPoint luceneTopLeft = new GeoPoint(geoBoundingBox.topLeft());
|
||||||
GeoPoint luceneBottomRight = new GeoPoint(bottomRight);
|
GeoPoint luceneBottomRight = new GeoPoint(geoBoundingBox.bottomRight());
|
||||||
if (GeoValidationMethod.isCoerce(validationMethod)) {
|
if (GeoValidationMethod.isCoerce(validationMethod)) {
|
||||||
// Special case: if the difference between the left and right is 360 and the right is greater than the left, we are asking for
|
// Special case: if the difference between the left and right is 360 and the right is greater than the left, we are asking for
|
||||||
// the complete longitude range so need to set longitude to the complete longitude range
|
// the complete longitude range so need to set longitude to the complete longitude range
|
||||||
|
@ -368,8 +354,7 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
builder.startObject(NAME);
|
builder.startObject(NAME);
|
||||||
|
|
||||||
builder.startObject(fieldName);
|
builder.startObject(fieldName);
|
||||||
builder.array(TOP_LEFT_FIELD.getPreferredName(), topLeft.getLon(), topLeft.getLat());
|
geoBoundingBox.toXContentFragment(builder, false);
|
||||||
builder.array(BOTTOM_RIGHT_FIELD.getPreferredName(), bottomRight.getLon(), bottomRight.getLat());
|
|
||||||
builder.endObject();
|
builder.endObject();
|
||||||
builder.field(VALIDATION_METHOD_FIELD.getPreferredName(), validationMethod);
|
builder.field(VALIDATION_METHOD_FIELD.getPreferredName(), validationMethod);
|
||||||
builder.field(TYPE_FIELD.getPreferredName(), type);
|
builder.field(TYPE_FIELD.getPreferredName(), type);
|
||||||
|
@ -391,7 +376,7 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
boolean ignoreUnmapped = DEFAULT_IGNORE_UNMAPPED;
|
boolean ignoreUnmapped = DEFAULT_IGNORE_UNMAPPED;
|
||||||
|
|
||||||
// bottom (minLat), top (maxLat), left (minLon), right (maxLon)
|
// bottom (minLat), top (maxLat), left (minLon), right (maxLon)
|
||||||
double[] bbox = null;
|
GeoBoundingBox bbox = null;
|
||||||
String type = "memory";
|
String type = "memory";
|
||||||
|
|
||||||
while ((token = parser.nextToken()) != XContentParser.Token.END_OBJECT) {
|
while ((token = parser.nextToken()) != XContentParser.Token.END_OBJECT) {
|
||||||
|
@ -399,7 +384,7 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
currentFieldName = parser.currentName();
|
currentFieldName = parser.currentName();
|
||||||
} else if (token == XContentParser.Token.START_OBJECT) {
|
} else if (token == XContentParser.Token.START_OBJECT) {
|
||||||
try {
|
try {
|
||||||
bbox = parseBoundingBox(parser);
|
bbox = GeoBoundingBox.parseBoundingBox(parser);
|
||||||
fieldName = currentFieldName;
|
fieldName = currentFieldName;
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new ElasticsearchParseException("failed to parse [{}] query. [{}]", NAME, e.getMessage());
|
throw new ElasticsearchParseException("failed to parse [{}] query. [{}]", NAME, e.getMessage());
|
||||||
|
@ -426,11 +411,8 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
throw new ElasticsearchParseException("failed to parse [{}] query. bounding box not provided", NAME);
|
throw new ElasticsearchParseException("failed to parse [{}] query. bounding box not provided", NAME);
|
||||||
}
|
}
|
||||||
|
|
||||||
final GeoPoint topLeft = new GeoPoint(bbox[1], bbox[2]);
|
|
||||||
final GeoPoint bottomRight = new GeoPoint(bbox[0], bbox[3]);
|
|
||||||
|
|
||||||
GeoBoundingBoxQueryBuilder builder = new GeoBoundingBoxQueryBuilder(fieldName);
|
GeoBoundingBoxQueryBuilder builder = new GeoBoundingBoxQueryBuilder(fieldName);
|
||||||
builder.setCorners(topLeft, bottomRight);
|
builder.setCorners(bbox.topLeft(), bbox.bottomRight());
|
||||||
builder.queryName(queryName);
|
builder.queryName(queryName);
|
||||||
builder.boost(boost);
|
builder.boost(boost);
|
||||||
builder.type(GeoExecType.fromString(type));
|
builder.type(GeoExecType.fromString(type));
|
||||||
|
@ -444,8 +426,7 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected boolean doEquals(GeoBoundingBoxQueryBuilder other) {
|
protected boolean doEquals(GeoBoundingBoxQueryBuilder other) {
|
||||||
return Objects.equals(topLeft, other.topLeft) &&
|
return Objects.equals(geoBoundingBox, other.geoBoundingBox) &&
|
||||||
Objects.equals(bottomRight, other.bottomRight) &&
|
|
||||||
Objects.equals(type, other.type) &&
|
Objects.equals(type, other.type) &&
|
||||||
Objects.equals(validationMethod, other.validationMethod) &&
|
Objects.equals(validationMethod, other.validationMethod) &&
|
||||||
Objects.equals(fieldName, other.fieldName) &&
|
Objects.equals(fieldName, other.fieldName) &&
|
||||||
|
@ -454,7 +435,7 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected int doHashCode() {
|
protected int doHashCode() {
|
||||||
return Objects.hash(topLeft, bottomRight, type, validationMethod, fieldName, ignoreUnmapped);
|
return Objects.hash(geoBoundingBox, type, validationMethod, fieldName, ignoreUnmapped);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -462,72 +443,4 @@ public class GeoBoundingBoxQueryBuilder extends AbstractQueryBuilder<GeoBounding
|
||||||
return NAME;
|
return NAME;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Parses the bounding box and returns bottom, top, left, right coordinates
|
|
||||||
*/
|
|
||||||
public static double[] parseBoundingBox(XContentParser parser) throws IOException, ElasticsearchParseException {
|
|
||||||
XContentParser.Token token = parser.currentToken();
|
|
||||||
if (token != XContentParser.Token.START_OBJECT) {
|
|
||||||
throw new ElasticsearchParseException("failed to parse bounding box. Expected start object but found [{}]", token);
|
|
||||||
}
|
|
||||||
|
|
||||||
double top = Double.NaN;
|
|
||||||
double bottom = Double.NaN;
|
|
||||||
double left = Double.NaN;
|
|
||||||
double right = Double.NaN;
|
|
||||||
|
|
||||||
String currentFieldName;
|
|
||||||
GeoPoint sparse = new GeoPoint();
|
|
||||||
EnvelopeBuilder envelope = null;
|
|
||||||
|
|
||||||
while ((token = parser.nextToken()) != XContentParser.Token.END_OBJECT) {
|
|
||||||
if (token == XContentParser.Token.FIELD_NAME) {
|
|
||||||
currentFieldName = parser.currentName();
|
|
||||||
token = parser.nextToken();
|
|
||||||
if (WKT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
envelope = (EnvelopeBuilder)(GeoWKTParser.parseExpectedType(parser, GeoShapeType.ENVELOPE));
|
|
||||||
} else if (TOP_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
top = parser.doubleValue();
|
|
||||||
} else if (BOTTOM_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
bottom = parser.doubleValue();
|
|
||||||
} else if (LEFT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
left = parser.doubleValue();
|
|
||||||
} else if (RIGHT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
right = parser.doubleValue();
|
|
||||||
} else {
|
|
||||||
if (TOP_LEFT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
GeoUtils.parseGeoPoint(parser, sparse, false, GeoUtils.EffectivePoint.TOP_LEFT);
|
|
||||||
top = sparse.getLat();
|
|
||||||
left = sparse.getLon();
|
|
||||||
} else if (BOTTOM_RIGHT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
GeoUtils.parseGeoPoint(parser, sparse, false, GeoUtils.EffectivePoint.BOTTOM_RIGHT);
|
|
||||||
bottom = sparse.getLat();
|
|
||||||
right = sparse.getLon();
|
|
||||||
} else if (TOP_RIGHT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
GeoUtils.parseGeoPoint(parser, sparse, false, GeoUtils.EffectivePoint.TOP_RIGHT);
|
|
||||||
top = sparse.getLat();
|
|
||||||
right = sparse.getLon();
|
|
||||||
} else if (BOTTOM_LEFT_FIELD.match(currentFieldName, parser.getDeprecationHandler())) {
|
|
||||||
GeoUtils.parseGeoPoint(parser, sparse, false, GeoUtils.EffectivePoint.BOTTOM_LEFT);
|
|
||||||
bottom = sparse.getLat();
|
|
||||||
left = sparse.getLon();
|
|
||||||
} else {
|
|
||||||
throw new ElasticsearchParseException("failed to parse bounding box. unexpected field [{}]", currentFieldName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
throw new ElasticsearchParseException("failed to parse bounding box. field name expected but [{}] found", token);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (envelope != null) {
|
|
||||||
if (Double.isNaN(top) == false || Double.isNaN(bottom) == false || Double.isNaN(left) == false ||
|
|
||||||
Double.isNaN(right) == false) {
|
|
||||||
throw new ElasticsearchParseException("failed to parse bounding box. Conflicting definition found "
|
|
||||||
+ "using well-known text and explicit corners.");
|
|
||||||
}
|
|
||||||
org.locationtech.spatial4j.shape.Rectangle r = envelope.buildS4J();
|
|
||||||
return new double[]{r.getMinY(), r.getMaxY(), r.getMinX(), r.getMaxX()};
|
|
||||||
}
|
|
||||||
return new double[]{bottom, top, left, right};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,7 +19,7 @@
|
||||||
|
|
||||||
package org.elasticsearch.search.aggregations.metrics;
|
package org.elasticsearch.search.aggregations.metrics;
|
||||||
|
|
||||||
import org.elasticsearch.common.ParseField;
|
import org.elasticsearch.common.geo.GeoBoundingBox;
|
||||||
import org.elasticsearch.common.geo.GeoPoint;
|
import org.elasticsearch.common.geo.GeoPoint;
|
||||||
import org.elasticsearch.common.io.stream.StreamInput;
|
import org.elasticsearch.common.io.stream.StreamInput;
|
||||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||||
|
@ -33,14 +33,6 @@ import java.util.Map;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
|
||||||
public class InternalGeoBounds extends InternalAggregation implements GeoBounds {
|
public class InternalGeoBounds extends InternalAggregation implements GeoBounds {
|
||||||
|
|
||||||
static final ParseField BOUNDS_FIELD = new ParseField("bounds");
|
|
||||||
static final ParseField TOP_LEFT_FIELD = new ParseField("top_left");
|
|
||||||
static final ParseField BOTTOM_RIGHT_FIELD = new ParseField("bottom_right");
|
|
||||||
static final ParseField LAT_FIELD = new ParseField("lat");
|
|
||||||
static final ParseField LON_FIELD = new ParseField("lon");
|
|
||||||
|
|
||||||
|
|
||||||
final double top;
|
final double top;
|
||||||
final double bottom;
|
final double bottom;
|
||||||
final double posLeft;
|
final double posLeft;
|
||||||
|
@ -132,30 +124,30 @@ public class InternalGeoBounds extends InternalAggregation implements GeoBounds
|
||||||
if (path.isEmpty()) {
|
if (path.isEmpty()) {
|
||||||
return this;
|
return this;
|
||||||
} else if (path.size() == 1) {
|
} else if (path.size() == 1) {
|
||||||
BoundingBox boundingBox = resolveBoundingBox();
|
GeoBoundingBox geoBoundingBox = resolveGeoBoundingBox();
|
||||||
String bBoxSide = path.get(0);
|
String bBoxSide = path.get(0);
|
||||||
switch (bBoxSide) {
|
switch (bBoxSide) {
|
||||||
case "top":
|
case "top":
|
||||||
return boundingBox.topLeft.lat();
|
return geoBoundingBox.top();
|
||||||
case "left":
|
case "left":
|
||||||
return boundingBox.topLeft.lon();
|
return geoBoundingBox.left();
|
||||||
case "bottom":
|
case "bottom":
|
||||||
return boundingBox.bottomRight.lat();
|
return geoBoundingBox.bottom();
|
||||||
case "right":
|
case "right":
|
||||||
return boundingBox.bottomRight.lon();
|
return geoBoundingBox.right();
|
||||||
default:
|
default:
|
||||||
throw new IllegalArgumentException("Found unknown path element [" + bBoxSide + "] in [" + getName() + "]");
|
throw new IllegalArgumentException("Found unknown path element [" + bBoxSide + "] in [" + getName() + "]");
|
||||||
}
|
}
|
||||||
} else if (path.size() == 2) {
|
} else if (path.size() == 2) {
|
||||||
BoundingBox boundingBox = resolveBoundingBox();
|
GeoBoundingBox geoBoundingBox = resolveGeoBoundingBox();
|
||||||
GeoPoint cornerPoint = null;
|
GeoPoint cornerPoint = null;
|
||||||
String cornerString = path.get(0);
|
String cornerString = path.get(0);
|
||||||
switch (cornerString) {
|
switch (cornerString) {
|
||||||
case "top_left":
|
case "top_left":
|
||||||
cornerPoint = boundingBox.topLeft;
|
cornerPoint = geoBoundingBox.topLeft();
|
||||||
break;
|
break;
|
||||||
case "bottom_right":
|
case "bottom_right":
|
||||||
cornerPoint = boundingBox.bottomRight;
|
cornerPoint = geoBoundingBox.bottomRight();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new IllegalArgumentException("Found unknown path element [" + cornerString + "] in [" + getName() + "]");
|
throw new IllegalArgumentException("Found unknown path element [" + cornerString + "] in [" + getName() + "]");
|
||||||
|
@ -176,78 +168,50 @@ public class InternalGeoBounds extends InternalAggregation implements GeoBounds
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public XContentBuilder doXContentBody(XContentBuilder builder, Params params) throws IOException {
|
public XContentBuilder doXContentBody(XContentBuilder builder, Params params) throws IOException {
|
||||||
GeoPoint topLeft = topLeft();
|
GeoBoundingBox bbox = resolveGeoBoundingBox();
|
||||||
GeoPoint bottomRight = bottomRight();
|
if (bbox != null) {
|
||||||
if (topLeft != null) {
|
bbox.toXContent(builder, params);
|
||||||
builder.startObject(BOUNDS_FIELD.getPreferredName());
|
|
||||||
builder.startObject(TOP_LEFT_FIELD.getPreferredName());
|
|
||||||
builder.field(LAT_FIELD.getPreferredName(), topLeft.lat());
|
|
||||||
builder.field(LON_FIELD.getPreferredName(), topLeft.lon());
|
|
||||||
builder.endObject();
|
|
||||||
builder.startObject(BOTTOM_RIGHT_FIELD.getPreferredName());
|
|
||||||
builder.field(LAT_FIELD.getPreferredName(), bottomRight.lat());
|
|
||||||
builder.field(LON_FIELD.getPreferredName(), bottomRight.lon());
|
|
||||||
builder.endObject();
|
|
||||||
builder.endObject();
|
|
||||||
}
|
}
|
||||||
return builder;
|
return builder;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class BoundingBox {
|
private GeoBoundingBox resolveGeoBoundingBox() {
|
||||||
private final GeoPoint topLeft;
|
|
||||||
private final GeoPoint bottomRight;
|
|
||||||
|
|
||||||
BoundingBox(GeoPoint topLeft, GeoPoint bottomRight) {
|
|
||||||
this.topLeft = topLeft;
|
|
||||||
this.bottomRight = bottomRight;
|
|
||||||
}
|
|
||||||
|
|
||||||
public GeoPoint topLeft() {
|
|
||||||
return topLeft;
|
|
||||||
}
|
|
||||||
|
|
||||||
public GeoPoint bottomRight() {
|
|
||||||
return bottomRight;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private BoundingBox resolveBoundingBox() {
|
|
||||||
if (Double.isInfinite(top)) {
|
if (Double.isInfinite(top)) {
|
||||||
return null;
|
return null;
|
||||||
} else if (Double.isInfinite(posLeft)) {
|
} else if (Double.isInfinite(posLeft)) {
|
||||||
return new BoundingBox(new GeoPoint(top, negLeft), new GeoPoint(bottom, negRight));
|
return new GeoBoundingBox(new GeoPoint(top, negLeft), new GeoPoint(bottom, negRight));
|
||||||
} else if (Double.isInfinite(negLeft)) {
|
} else if (Double.isInfinite(negLeft)) {
|
||||||
return new BoundingBox(new GeoPoint(top, posLeft), new GeoPoint(bottom, posRight));
|
return new GeoBoundingBox(new GeoPoint(top, posLeft), new GeoPoint(bottom, posRight));
|
||||||
} else if (wrapLongitude) {
|
} else if (wrapLongitude) {
|
||||||
double unwrappedWidth = posRight - negLeft;
|
double unwrappedWidth = posRight - negLeft;
|
||||||
double wrappedWidth = (180 - posLeft) - (-180 - negRight);
|
double wrappedWidth = (180 - posLeft) - (-180 - negRight);
|
||||||
if (unwrappedWidth <= wrappedWidth) {
|
if (unwrappedWidth <= wrappedWidth) {
|
||||||
return new BoundingBox(new GeoPoint(top, negLeft), new GeoPoint(bottom, posRight));
|
return new GeoBoundingBox(new GeoPoint(top, negLeft), new GeoPoint(bottom, posRight));
|
||||||
} else {
|
} else {
|
||||||
return new BoundingBox(new GeoPoint(top, posLeft), new GeoPoint(bottom, negRight));
|
return new GeoBoundingBox(new GeoPoint(top, posLeft), new GeoPoint(bottom, negRight));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return new BoundingBox(new GeoPoint(top, negLeft), new GeoPoint(bottom, posRight));
|
return new GeoBoundingBox(new GeoPoint(top, negLeft), new GeoPoint(bottom, posRight));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public GeoPoint topLeft() {
|
public GeoPoint topLeft() {
|
||||||
BoundingBox boundingBox = resolveBoundingBox();
|
GeoBoundingBox geoBoundingBox = resolveGeoBoundingBox();
|
||||||
if (boundingBox == null) {
|
if (geoBoundingBox == null) {
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
return boundingBox.topLeft();
|
return geoBoundingBox.topLeft();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public GeoPoint bottomRight() {
|
public GeoPoint bottomRight() {
|
||||||
BoundingBox boundingBox = resolveBoundingBox();
|
GeoBoundingBox geoBoundingBox = resolveGeoBoundingBox();
|
||||||
if (boundingBox == null) {
|
if (geoBoundingBox == null) {
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
return boundingBox.bottomRight();
|
return geoBoundingBox.bottomRight();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -20,6 +20,7 @@
|
||||||
package org.elasticsearch.search.aggregations.metrics;
|
package org.elasticsearch.search.aggregations.metrics;
|
||||||
|
|
||||||
import org.elasticsearch.common.collect.Tuple;
|
import org.elasticsearch.common.collect.Tuple;
|
||||||
|
import org.elasticsearch.common.geo.GeoBoundingBox;
|
||||||
import org.elasticsearch.common.geo.GeoPoint;
|
import org.elasticsearch.common.geo.GeoPoint;
|
||||||
import org.elasticsearch.common.xcontent.ConstructingObjectParser;
|
import org.elasticsearch.common.xcontent.ConstructingObjectParser;
|
||||||
import org.elasticsearch.common.xcontent.ObjectParser;
|
import org.elasticsearch.common.xcontent.ObjectParser;
|
||||||
|
@ -30,15 +31,14 @@ import org.elasticsearch.search.aggregations.ParsedAggregation;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
import static org.elasticsearch.common.xcontent.ConstructingObjectParser.constructorArg;
|
import static org.elasticsearch.common.xcontent.ConstructingObjectParser.constructorArg;
|
||||||
import static org.elasticsearch.search.aggregations.metrics.InternalGeoBounds.BOTTOM_RIGHT_FIELD;
|
import static org.elasticsearch.common.geo.GeoBoundingBox.BOTTOM_RIGHT_FIELD;
|
||||||
import static org.elasticsearch.search.aggregations.metrics.InternalGeoBounds.BOUNDS_FIELD;
|
import static org.elasticsearch.common.geo.GeoBoundingBox.BOUNDS_FIELD;
|
||||||
import static org.elasticsearch.search.aggregations.metrics.InternalGeoBounds.LAT_FIELD;
|
import static org.elasticsearch.common.geo.GeoBoundingBox.LAT_FIELD;
|
||||||
import static org.elasticsearch.search.aggregations.metrics.InternalGeoBounds.LON_FIELD;
|
import static org.elasticsearch.common.geo.GeoBoundingBox.LON_FIELD;
|
||||||
import static org.elasticsearch.search.aggregations.metrics.InternalGeoBounds.TOP_LEFT_FIELD;
|
import static org.elasticsearch.common.geo.GeoBoundingBox.TOP_LEFT_FIELD;
|
||||||
|
|
||||||
public class ParsedGeoBounds extends ParsedAggregation implements GeoBounds {
|
public class ParsedGeoBounds extends ParsedAggregation implements GeoBounds {
|
||||||
private GeoPoint topLeft;
|
private GeoBoundingBox geoBoundingBox;
|
||||||
private GeoPoint bottomRight;
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getType() {
|
public String getType() {
|
||||||
|
@ -47,29 +47,20 @@ public class ParsedGeoBounds extends ParsedAggregation implements GeoBounds {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public XContentBuilder doXContentBody(XContentBuilder builder, Params params) throws IOException {
|
public XContentBuilder doXContentBody(XContentBuilder builder, Params params) throws IOException {
|
||||||
if (topLeft != null) {
|
if (geoBoundingBox != null) {
|
||||||
builder.startObject(BOUNDS_FIELD.getPreferredName());
|
geoBoundingBox.toXContent(builder, params);
|
||||||
builder.startObject(TOP_LEFT_FIELD.getPreferredName());
|
|
||||||
builder.field(LAT_FIELD.getPreferredName(), topLeft.getLat());
|
|
||||||
builder.field(LON_FIELD.getPreferredName(), topLeft.getLon());
|
|
||||||
builder.endObject();
|
|
||||||
builder.startObject(BOTTOM_RIGHT_FIELD.getPreferredName());
|
|
||||||
builder.field(LAT_FIELD.getPreferredName(), bottomRight.getLat());
|
|
||||||
builder.field(LON_FIELD.getPreferredName(), bottomRight.getLon());
|
|
||||||
builder.endObject();
|
|
||||||
builder.endObject();
|
|
||||||
}
|
}
|
||||||
return builder;
|
return builder;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public GeoPoint topLeft() {
|
public GeoPoint topLeft() {
|
||||||
return topLeft;
|
return geoBoundingBox.topLeft();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public GeoPoint bottomRight() {
|
public GeoPoint bottomRight() {
|
||||||
return bottomRight;
|
return geoBoundingBox.bottomRight();
|
||||||
}
|
}
|
||||||
|
|
||||||
private static final ObjectParser<ParsedGeoBounds, Void> PARSER = new ObjectParser<>(ParsedGeoBounds.class.getSimpleName(), true,
|
private static final ObjectParser<ParsedGeoBounds, Void> PARSER = new ObjectParser<>(ParsedGeoBounds.class.getSimpleName(), true,
|
||||||
|
@ -85,8 +76,7 @@ public class ParsedGeoBounds extends ParsedAggregation implements GeoBounds {
|
||||||
static {
|
static {
|
||||||
declareAggregationFields(PARSER);
|
declareAggregationFields(PARSER);
|
||||||
PARSER.declareObject((agg, bbox) -> {
|
PARSER.declareObject((agg, bbox) -> {
|
||||||
agg.topLeft = bbox.v1();
|
agg.geoBoundingBox = new GeoBoundingBox(bbox.v1(), bbox.v2());
|
||||||
agg.bottomRight = bbox.v2();
|
|
||||||
}, BOUNDS_PARSER, BOUNDS_FIELD);
|
}, BOUNDS_PARSER, BOUNDS_FIELD);
|
||||||
|
|
||||||
BOUNDS_PARSER.declareObject(constructorArg(), GEO_POINT_PARSER, TOP_LEFT_FIELD);
|
BOUNDS_PARSER.declareObject(constructorArg(), GEO_POINT_PARSER, TOP_LEFT_FIELD);
|
||||||
|
|
|
@ -0,0 +1,149 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elasticsearch under one or more contributor
|
||||||
|
* license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright
|
||||||
|
* ownership. Elasticsearch licenses this file to you 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.elasticsearch.common.geo;
|
||||||
|
|
||||||
|
import org.elasticsearch.ElasticsearchParseException;
|
||||||
|
import org.elasticsearch.common.xcontent.XContentBuilder;
|
||||||
|
import org.elasticsearch.common.xcontent.XContentFactory;
|
||||||
|
import org.elasticsearch.common.xcontent.XContentParser;
|
||||||
|
import org.elasticsearch.geo.GeometryTestUtils;
|
||||||
|
import org.elasticsearch.test.ESTestCase;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
import static org.hamcrest.Matchers.equalTo;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link GeoBoundingBox}
|
||||||
|
*/
|
||||||
|
public class GeoBoundingBoxTests extends ESTestCase {
|
||||||
|
|
||||||
|
public void testInvalidParseInvalidWKT() throws IOException {
|
||||||
|
XContentBuilder bboxBuilder = XContentFactory.jsonBuilder()
|
||||||
|
.startObject()
|
||||||
|
.field("wkt", "invalid")
|
||||||
|
.endObject();
|
||||||
|
XContentParser parser = createParser(bboxBuilder);
|
||||||
|
parser.nextToken();
|
||||||
|
ElasticsearchParseException e = expectThrows(ElasticsearchParseException.class, () -> GeoBoundingBox.parseBoundingBox(parser));
|
||||||
|
assertThat(e.getMessage(), equalTo("failed to parse WKT bounding box"));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testInvalidParsePoint() throws IOException {
|
||||||
|
XContentBuilder bboxBuilder = XContentFactory.jsonBuilder()
|
||||||
|
.startObject()
|
||||||
|
.field("wkt", "POINT (100.0 100.0)")
|
||||||
|
.endObject();
|
||||||
|
XContentParser parser = createParser(bboxBuilder);
|
||||||
|
parser.nextToken();
|
||||||
|
ElasticsearchParseException e = expectThrows(ElasticsearchParseException.class, () -> GeoBoundingBox.parseBoundingBox(parser));
|
||||||
|
assertThat(e.getMessage(), equalTo("failed to parse WKT bounding box. [POINT] found. expected [ENVELOPE]"));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testWKT() throws IOException {
|
||||||
|
GeoBoundingBox geoBoundingBox = randomBBox();
|
||||||
|
assertBBox(geoBoundingBox,
|
||||||
|
XContentFactory.jsonBuilder()
|
||||||
|
.startObject()
|
||||||
|
.field("wkt", geoBoundingBox.toString())
|
||||||
|
.endObject()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testTopBottomLeftRight() throws Exception {
|
||||||
|
GeoBoundingBox geoBoundingBox = randomBBox();
|
||||||
|
assertBBox(geoBoundingBox,
|
||||||
|
XContentFactory.jsonBuilder()
|
||||||
|
.startObject()
|
||||||
|
.field("top", geoBoundingBox.top())
|
||||||
|
.field("bottom", geoBoundingBox.bottom())
|
||||||
|
.field("left", geoBoundingBox.left())
|
||||||
|
.field("right", geoBoundingBox.right())
|
||||||
|
.endObject()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testTopLeftBottomRight() throws Exception {
|
||||||
|
GeoBoundingBox geoBoundingBox = randomBBox();
|
||||||
|
assertBBox(geoBoundingBox,
|
||||||
|
XContentFactory.jsonBuilder()
|
||||||
|
.startObject()
|
||||||
|
.field("top_left", geoBoundingBox.topLeft())
|
||||||
|
.field("bottom_right", geoBoundingBox.bottomRight())
|
||||||
|
.endObject()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testTopRightBottomLeft() throws Exception {
|
||||||
|
GeoBoundingBox geoBoundingBox = randomBBox();
|
||||||
|
assertBBox(geoBoundingBox,
|
||||||
|
XContentFactory.jsonBuilder()
|
||||||
|
.startObject()
|
||||||
|
.field("top_right", new GeoPoint(geoBoundingBox.top(), geoBoundingBox.right()))
|
||||||
|
.field("bottom_left", new GeoPoint(geoBoundingBox.bottom(), geoBoundingBox.left()))
|
||||||
|
.endObject()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// test that no exception is thrown. BBOX parsing is not validated
|
||||||
|
public void testNullTopBottomLeftRight() throws Exception {
|
||||||
|
GeoBoundingBox geoBoundingBox = randomBBox();
|
||||||
|
XContentBuilder builder = XContentFactory.jsonBuilder().startObject();
|
||||||
|
for (String field : randomSubsetOf(Arrays.asList("top", "bottom", "left", "right"))) {
|
||||||
|
switch (field) {
|
||||||
|
case "top":
|
||||||
|
builder.field("top", geoBoundingBox.top());
|
||||||
|
break;
|
||||||
|
case "bottom":
|
||||||
|
builder.field("bottom", geoBoundingBox.bottom());
|
||||||
|
break;
|
||||||
|
case "left":
|
||||||
|
builder.field("left", geoBoundingBox.left());
|
||||||
|
break;
|
||||||
|
case "right":
|
||||||
|
builder.field("right", geoBoundingBox.right());
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new IllegalStateException("unexpected branching");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
builder.endObject();
|
||||||
|
try (XContentParser parser = createParser(builder)) {
|
||||||
|
parser.nextToken();
|
||||||
|
GeoBoundingBox.parseBoundingBox(parser);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertBBox(GeoBoundingBox expected, XContentBuilder builder) throws IOException {
|
||||||
|
try (XContentParser parser = createParser(builder)) {
|
||||||
|
parser.nextToken();
|
||||||
|
assertThat(GeoBoundingBox.parseBoundingBox(parser), equalTo(expected));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private GeoBoundingBox randomBBox() {
|
||||||
|
double topLat = GeometryTestUtils.randomLat();
|
||||||
|
double bottomLat = randomDoubleBetween(GeoUtils.MIN_LAT, topLat, true);
|
||||||
|
return new GeoBoundingBox(new GeoPoint(topLat, GeometryTestUtils.randomLon()),
|
||||||
|
new GeoPoint(bottomLat, GeometryTestUtils.randomLon()));
|
||||||
|
}
|
||||||
|
}
|
|
@ -39,7 +39,7 @@ import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertAcke
|
||||||
import static org.hamcrest.Matchers.anyOf;
|
import static org.hamcrest.Matchers.anyOf;
|
||||||
import static org.hamcrest.Matchers.equalTo;
|
import static org.hamcrest.Matchers.equalTo;
|
||||||
|
|
||||||
public class GeoBoundingBoxIT extends ESIntegTestCase {
|
public class GeoBoundingBoxQueryIT extends ESIntegTestCase {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected boolean forbidPrivateIndexSettings() {
|
protected boolean forbidPrivateIndexSettings() {
|
Loading…
Reference in New Issue