Geo: Fixes indexing of linestrings that go around the globe (#47471)

LINESTRING (0 0, 720 20) is now decomposed into 3 strings:
multilinestring (
  (0.0 0.0, 180.0 5.0),
  (-180.0 5.0, 180 15),
  (-180.0 15.0, 0 20)
)

It also fixes issues with linestrings that intersect antimeridian
more than 5 times.

Fixes #43837
Fixes #43826
This commit is contained in:
Igor Motov 2019-10-09 14:35:10 +04:00
parent d18ff24dbe
commit f8b8afdc70
3 changed files with 168 additions and 58 deletions

View File

@ -337,7 +337,7 @@ public class GeoUtils {
} }
} }
private static double centeredModulus(double dividend, double divisor) { public static double centeredModulus(double dividend, double divisor) {
double rtn = dividend % divisor; double rtn = dividend % divisor;
if (rtn <= 0) { if (rtn <= 0) {
rtn += divisor; rtn += divisor;

View File

@ -23,6 +23,7 @@ package org.elasticsearch.index.mapper;
import org.apache.lucene.document.LatLonShape; import org.apache.lucene.document.LatLonShape;
import org.apache.lucene.index.IndexableField; import org.apache.lucene.index.IndexableField;
import org.elasticsearch.common.collect.Tuple; import org.elasticsearch.common.collect.Tuple;
import org.elasticsearch.common.geo.GeoUtils;
import org.elasticsearch.geometry.Circle; import org.elasticsearch.geometry.Circle;
import org.elasticsearch.geometry.Geometry; import org.elasticsearch.geometry.Geometry;
import org.elasticsearch.geometry.GeometryCollection; import org.elasticsearch.geometry.GeometryCollection;
@ -222,88 +223,113 @@ public final class GeoShapeIndexer implements AbstractGeometryFieldMapper.Indexe
* Splits the specified line by datelines and adds them to the supplied lines array * Splits the specified line by datelines and adds them to the supplied lines array
*/ */
private List<Line> decomposeGeometry(Line line, List<Line> lines) { private List<Line> decomposeGeometry(Line line, List<Line> lines) {
for (Line part : decompose(line)) {
for (Line partPlus : decompose(+DATELINE, line)) { double[] lats = new double[part.length()];
for (Line partMinus : decompose(-DATELINE, partPlus)) { double[] lons = new double[part.length()];
double[] lats = new double[partMinus.length()]; for (int i = 0; i < part.length(); i++) {
double[] lons = new double[partMinus.length()]; lats[i] = normalizeLat(part.getY(i));
for (int i = 0; i < partMinus.length(); i++) { lons[i] = normalizeLonMinus180Inclusive(part.getX(i));
lats[i] = normalizeLat(partMinus.getY(i));
lons[i] = normalizeLonMinus180Inclusive(partMinus.getX(i));
}
lines.add(new Line(lons, lats));
} }
lines.add(new Line(lons, lats));
} }
return lines; return lines;
} }
/** /**
* Decompose a linestring given as array of coordinates at a vertical line. * Calculates how many degres the given longitude needs to be moved east in order to be in -180 - +180. +180 is inclusive only
* * if include180 is true.
* @param dateline x-axis intercept of the vertical line
* @param line linestring that should be decomposed
* @return array of linestrings given as coordinate arrays
*/ */
private List<Line> decompose(double dateline, Line line) { double calculateShift(double lon, boolean include180) {
double[] lons = line.getX(); double normalized = GeoUtils.centeredModulus(lon, 360);
double[] lats = line.getY(); double shift = Math.round(normalized - lon);
return decompose(dateline, lons, lats); if (!include180 && normalized == 180.0) {
shift = shift - 360;
}
return shift;
} }
/** /**
* Decompose a linestring given as two arrays of coordinates at a vertical line. * Decompose a linestring given as array of coordinates by anti-meridian.
*
* @param line linestring that should be decomposed
* @return array of linestrings given as coordinate arrays
*/ */
private List<Line> decompose(double dateline, double[] lons, double[] lats) { private List<Line> decompose(Line line) {
double[] lons = line.getX();
double[] lats = line.getY();
int offset = 0; int offset = 0;
ArrayList<Line> parts = new ArrayList<>(); ArrayList<Line> parts = new ArrayList<>();
double lastLon = lons[0]; double shift = 0;
double shift = lastLon > DATELINE ? DATELINE : (lastLon < -DATELINE ? -DATELINE : 0); int i = 1;
while (i < lons.length) {
for (int i = 1; i < lons.length; i++) { // Check where the line is going east (+1), west (-1) or directly north/south (0)
double t = intersection(lastLon, lons[i], dateline); int direction = Double.compare(lons[i], lons[i - 1]);
lastLon = lons[i]; double newShift = calculateShift(lons[i - 1], direction < 0);
if (Double.isNaN(t) == false) { // first point lon + shift is always between -180.0 and +180.0
double[] partLons = Arrays.copyOfRange(lons, offset, i + 1); if (i - offset > 1 && newShift != shift) {
double[] partLats = Arrays.copyOfRange(lats, offset, i + 1); // Jumping over anti-meridian - we need to start a new segment
if (t < 1) { double[] partLons = Arrays.copyOfRange(lons, offset, i);
Point intersection = position(new Point(lons[i - 1], lats[i - 1]), new Point(lons[i], lats[i]), t); double[] partLats = Arrays.copyOfRange(lats, offset, i);
partLons[partLons.length - 1] = intersection.getX(); performShift(shift, partLons);
partLats[partLats.length - 1] = intersection.getY(); shift = newShift;
offset = i - 1;
lons[offset + i - 1] = intersection.getX();
lats[offset + i - 1] = intersection.getY();
shift(shift, partLons);
offset = i - 1;
shift = lons[i] > DATELINE ? DATELINE : (lons[i] < -DATELINE ? -DATELINE : 0);
} else {
shift(shift, partLons);
offset = i;
}
parts.add(new Line(partLons, partLats)); parts.add(new Line(partLons, partLats));
} else {
// Check if new point intersects with anti-meridian
shift = newShift;
double t = intersection(lons[i - 1] + shift, lons[i] + shift);
if (Double.isNaN(t) == false) {
// Found intersection, all previous segments are now part of the linestring
double[] partLons = Arrays.copyOfRange(lons, offset, i + 1);
double[] partLats = Arrays.copyOfRange(lats, offset, i + 1);
lons[i - 1] = partLons[partLons.length - 1] = (direction > 0 ? DATELINE : -DATELINE) - shift;
lats[i - 1] = partLats[partLats.length - 1] = lats[i - 1] + (lats[i] - lats[i - 1]) * t;
performShift(shift, partLons);
offset = i - 1;
parts.add(new Line(partLons, partLats));
} else {
// Didn't find intersection - just continue checking
i++;
}
} }
} }
if (offset == 0) { if (offset == 0) {
shift(shift, lons); performShift(shift, lons);
parts.add(new Line(lons, lats)); parts.add(new Line(lons, lats));
} else if (offset < lons.length - 1) { } else if (offset < lons.length - 1) {
double[] partLons = Arrays.copyOfRange(lons, offset, lons.length); double[] partLons = Arrays.copyOfRange(lons, offset, lons.length);
double[] partLats = Arrays.copyOfRange(lats, offset, lats.length); double[] partLats = Arrays.copyOfRange(lats, offset, lats.length);
shift(shift, partLons); performShift(shift, partLons);
parts.add(new Line(partLons, partLats)); parts.add(new Line(partLons, partLats));
} }
return parts; return parts;
} }
/** /**
* shifts all coordinates by (- shift * 2) * Checks it the segment from p1x to p2x intersects with anti-meridian
* p1x must be with in -180 +180 range
*/ */
private static void shift(double shift, double[] lons) { private static double intersection(double p1x, double p2x) {
if (p1x == p2x) {
return Double.NaN;
}
final double t = ((p1x < p2x ? DATELINE : -DATELINE) - p1x) / (p2x - p1x);
if (t >= 1 || t <= 0) {
return Double.NaN;
} else {
return t;
}
}
/**
* shifts all coordinates by shift
*/
private static void performShift(double shift, double[] lons) {
if (shift != 0) { if (shift != 0) {
for (int j = 0; j < lons.length; j++) { for (int j = 0; j < lons.length; j++) {
lons[j] = lons[j] - 2 * shift; lons[j] = lons[j] + shift;
} }
} }
} }

View File

@ -22,6 +22,7 @@ package org.elasticsearch.common.geo;
import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentBuilder;
import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.common.xcontent.XContentFactory;
import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.common.xcontent.XContentParser;
import org.elasticsearch.geo.GeometryTestUtils;
import org.elasticsearch.geometry.Circle; import org.elasticsearch.geometry.Circle;
import org.elasticsearch.geometry.Geometry; import org.elasticsearch.geometry.Geometry;
import org.elasticsearch.geometry.GeometryCollection; import org.elasticsearch.geometry.GeometryCollection;
@ -32,7 +33,6 @@ import org.elasticsearch.geometry.MultiPoint;
import org.elasticsearch.geometry.MultiPolygon; import org.elasticsearch.geometry.MultiPolygon;
import org.elasticsearch.geometry.Point; import org.elasticsearch.geometry.Point;
import org.elasticsearch.geometry.Polygon; import org.elasticsearch.geometry.Polygon;
import org.elasticsearch.geometry.utils.WellKnownText;
import org.elasticsearch.index.mapper.GeoShapeIndexer; import org.elasticsearch.index.mapper.GeoShapeIndexer;
import org.elasticsearch.test.ESTestCase; import org.elasticsearch.test.ESTestCase;
@ -41,12 +41,11 @@ import java.text.ParseException;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collections; import java.util.Collections;
import static org.hamcrest.Matchers.instanceOf;
public class GeometryIndexerTests extends ESTestCase { public class GeometryIndexerTests extends ESTestCase {
GeoShapeIndexer indexer = new GeoShapeIndexer(true, "test"); GeoShapeIndexer indexer = new GeoShapeIndexer(true, "test");
private static final WellKnownText WKT = new WellKnownText(true, geometry -> {
});
public void testCircle() { public void testCircle() {
UnsupportedOperationException ex = UnsupportedOperationException ex =
@ -105,10 +104,96 @@ public class GeometryIndexerTests extends ESTestCase {
new Line(new double[]{160, 180}, new double[]{0, 5}), new Line(new double[]{160, 180}, new double[]{0, 5}),
new Line(new double[]{-180, -160, -180}, new double[]{5, 10, 15}), new Line(new double[]{-180, -160, -180}, new double[]{5, 10, 15}),
new Line(new double[]{180, 160}, new double[]{15, 20}) new Line(new double[]{180, 160}, new double[]{15, 20})
) ));
);
assertEquals(indexed, indexer.prepareForIndexing(line)); assertEquals(indexed, indexer.prepareForIndexing(line));
line = new Line(new double[]{0, 720}, new double[]{0, 20});
indexed = new MultiLine(Arrays.asList(
new Line(new double[]{0, 180}, new double[]{0, 5}),
new Line(new double[]{-180, 180}, new double[]{5, 15}),
new Line(new double[]{-180, 0}, new double[]{15, 20})
));
assertEquals(indexed, indexer.prepareForIndexing(line));
line = new Line(new double[]{160, 180, 180, 200, 160, 140}, new double[]{0, 10, 20, 30, 30, 40});
indexed = new MultiLine(Arrays.asList(
new Line(new double[]{160, 180}, new double[]{0, 10}),
new Line(new double[]{-180, -180, -160, -180}, new double[]{10, 20, 30, 30}),
new Line(new double[]{180, 160, 140}, new double[]{30, 30, 40})
));
assertEquals(indexed, indexer.prepareForIndexing(line));
line = new Line(new double[]{-70, 180, 900}, new double[]{0, 0, 4});
indexed = new MultiLine(Arrays.asList(
new Line(new double[]{-70, 180}, new double[]{0, 0}),
new Line(new double[]{-180, 180}, new double[]{0, 2}),
new Line(new double[]{-180, 180}, new double[]{2, 4})
));
assertEquals(indexed, indexer.prepareForIndexing(line));
line = new Line(new double[]{160, 200, 160, 200, 160, 200}, new double[]{0, 10, 20, 30, 40, 50});
indexed = new MultiLine(Arrays.asList(
new Line(new double[]{160, 180}, new double[]{0, 5}),
new Line(new double[]{-180, -160, -180}, new double[]{5, 10, 15}),
new Line(new double[]{180, 160, 180}, new double[]{15, 20, 25}),
new Line(new double[]{-180, -160, -180}, new double[]{25, 30, 35}),
new Line(new double[]{180, 160, 180}, new double[]{35, 40, 45}),
new Line(new double[]{-180, -160}, new double[]{45, 50})
));
assertEquals(indexed, indexer.prepareForIndexing(line));
}
/**
* Returns a sum of Euclidean distances between points in the linestring.
*/
public double length(Line line) {
double distance = 0;
for (int i = 1; i < line.length(); i++) {
distance += Math.sqrt((line.getLat(i) - line.getLat(i - 1)) * (line.getLat(i) - line.getLat(i - 1)) +
(line.getLon(i) - line.getLon(i - 1)) * (line.getLon(i) - line.getLon(i - 1)));
}
return distance;
}
/**
* A simple tests that generates a random lines crossing anti-merdian and checks that the decomposed segments of this line
* have the same total length (measured using Euclidean distances between neighboring points) as the original line.
*/
public void testRandomLine() {
int size = randomIntBetween(2, 20);
int shift = randomIntBetween(-2, 2);
double[] originalLats = new double[size];
double[] originalLons = new double[size];
for (int i = 0; i < size; i++) {
originalLats[i] = GeometryTestUtils.randomLat();
originalLons[i] = GeometryTestUtils.randomLon() + shift * 360;
if (randomInt(3) == 0) {
shift += randomFrom(-2, -1, 1, 2);
}
}
Line original = new Line(originalLons, originalLats);
Geometry decomposed = indexer.prepareForIndexing(original);
double decomposedLength = 0;
if (decomposed instanceof Line) {
decomposedLength = length((Line) decomposed);
} else {
assertThat(decomposed, instanceOf(MultiLine.class));
MultiLine lines = (MultiLine) decomposed;
for (int i = 0; i < lines.size(); i++) {
decomposedLength += length(lines.get(i));
}
}
assertEquals("Different Lengths between " + original + " and " + decomposed, length(original), decomposedLength, 0.001);
} }
public void testMultiLine() { public void testMultiLine() {
@ -254,5 +339,4 @@ public class GeometryIndexerTests extends ESTestCase {
return geometryParser.parse(parser); return geometryParser.parse(parser);
} }
} }
} }