LUCENE-10470: [Tessellator] Fix some failing polygons due to collinear edges (#756)

Check if polygon has been successfully tessellated before we fail (we are failing some valid
  tessellations) and allow filtering edges that fold on top of the previous one
This commit is contained in:
Ignacio Vera 2022-04-27 10:24:22 +02:00 committed by GitHub
parent 2b20b3f2ca
commit 5d3ab09676
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 95 additions and 11 deletions

View File

@ -152,6 +152,9 @@ Bug Fixes
* LUCENE-10529: Properly handle when TestTaxonomyFacetAssociations test case randomly indexes * LUCENE-10529: Properly handle when TestTaxonomyFacetAssociations test case randomly indexes
no documents instead of throwing an NPE. (Greg Miller) no documents instead of throwing an NPE. (Greg Miller)
* LUCENE-10470: Check if polygon has been successfully tessellated before we fail (we are failing some valid
tessellations) and allow filtering edges that fold on top of the previous one. (Ignacio Vera)
Build Build
--------------------- ---------------------

View File

@ -827,7 +827,8 @@ public final class Tessellator {
} }
searchNode = searchNode.next; searchNode = searchNode.next;
} while (searchNode != start); } while (searchNode != start);
return false; // if there is some area left, we failed
return signedArea(start, start) == 0;
} }
/** Computes if edge defined by a and b overlaps with a polygon edge * */ /** Computes if edge defined by a and b overlaps with a polygon edge * */
@ -1102,6 +1103,12 @@ public final class Tessellator {
/** Determine whether the polygon defined between node start and node end is CW */ /** Determine whether the polygon defined between node start and node end is CW */
private static boolean isCWPolygon(final Node start, final Node end) { private static boolean isCWPolygon(final Node start, final Node end) {
// The polygon must be CW
return (signedArea(start, end) < 0) ? true : false;
}
/** Determine the signed area between node start and node end */
private static double signedArea(final Node start, final Node end) {
Node next = start; Node next = start;
double windingSum = 0; double windingSum = 0;
do { do {
@ -1111,8 +1118,7 @@ public final class Tessellator {
next.getX(), next.getY(), next.next.getX(), next.next.getY(), end.getX(), end.getY()); next.getX(), next.getY(), next.next.getX(), next.next.getY(), end.getX(), end.getY());
next = next.next; next = next.next;
} while (next.next != end); } while (next.next != end);
// The polygon must be CW return windingSum;
return (windingSum < 0) ? true : false;
} }
private static final boolean isLocallyInside(final Node a, final Node b) { private static final boolean isLocallyInside(final Node a, final Node b) {
@ -1291,10 +1297,12 @@ public final class Tessellator {
// we can filter points when: // we can filter points when:
// 1. they are the same // 1. they are the same
// 2.- each one starts and ends in each other // 2.- each one starts and ends in each other
// 3.- they are co-linear and both edges have the same value in .isNextEdgeFromPolygon // 3.- they are collinear and both edges have the same value in .isNextEdgeFromPolygon
// 4.- they are collinear and second edge returns over the first edge
if (isVertexEquals(node, nextNode) if (isVertexEquals(node, nextNode)
|| isVertexEquals(prevNode, nextNode) || isVertexEquals(prevNode, nextNode)
|| (prevNode.isNextEdgeFromPolygon == node.isNextEdgeFromPolygon || ((prevNode.isNextEdgeFromPolygon == node.isNextEdgeFromPolygon
|| isPointInLine(prevNode, node, nextNode.getX(), nextNode.getY()))
&& area( && area(
prevNode.getX(), prevNode.getX(),
prevNode.getY(), prevNode.getY(),

View File

@ -128,10 +128,19 @@ public class TestTessellator extends LuceneTestCase {
public void testInvalidPolygonIntersects() throws Exception { public void testInvalidPolygonIntersects() throws Exception {
String wkt = "POLYGON((0 0, 1 1, 0 1, 1 0, 0 0))"; String wkt = "POLYGON((0 0, 1 1, 0 1, 1 0, 0 0))";
Polygon polygon = (Polygon) SimpleWKTShapeParser.parse(wkt); Polygon polygon = (Polygon) SimpleWKTShapeParser.parse(wkt);
{
IllegalArgumentException ex = IllegalArgumentException ex =
expectThrows(IllegalArgumentException.class, () -> Tessellator.tessellate(polygon, true)); expectThrows(IllegalArgumentException.class, () -> Tessellator.tessellate(polygon, true));
assertEquals("Polygon self-intersection at lat=0.5 lon=0.5", ex.getMessage()); assertEquals("Polygon self-intersection at lat=0.5 lon=0.5", ex.getMessage());
} }
{
IllegalArgumentException ex =
expectThrows(
IllegalArgumentException.class, () -> Tessellator.tessellate(polygon, false));
assertEquals(
"Unable to Tessellate shape. Possible malformed shape detected.", ex.getMessage());
}
}
public void testInvalidPolygonOverlap() throws Exception { public void testInvalidPolygonOverlap() throws Exception {
// holes are equal // holes are equal
@ -143,9 +152,19 @@ public class TestTessellator extends LuceneTestCase {
+ " (6.0391557 52.0929189, 6.0388667 52.0928373, 6.0387045 52.0928107, 6.038578 52.0927855, 6.0384897 52.0927195, 6.0384626 52.0927036, 6.0384412 52.0926911, 6.0382642 52.0926086, 6.0380309 52.092529, 6.0377877 52.0924683, 6.0377571 52.0924499, 6.0377263 52.0924189, 6.037857 52.0923747, 6.0383203 52.0923097, 6.0385012 52.0922528, 6.0385416 52.0922588, 6.0385632 52.0923458, 6.0386452 52.0924386, 6.0387875 52.0925001, 6.0391495 52.0926998, 6.0393437 52.0928496, 6.0393774 52.0928918, 6.0393715 52.092914, 6.0393239 52.0929308, 6.039246 52.0929349, 6.0391557 52.0929189)," + " (6.0391557 52.0929189, 6.0388667 52.0928373, 6.0387045 52.0928107, 6.038578 52.0927855, 6.0384897 52.0927195, 6.0384626 52.0927036, 6.0384412 52.0926911, 6.0382642 52.0926086, 6.0380309 52.092529, 6.0377877 52.0924683, 6.0377571 52.0924499, 6.0377263 52.0924189, 6.037857 52.0923747, 6.0383203 52.0923097, 6.0385012 52.0922528, 6.0385416 52.0922588, 6.0385632 52.0923458, 6.0386452 52.0924386, 6.0387875 52.0925001, 6.0391495 52.0926998, 6.0393437 52.0928496, 6.0393774 52.0928918, 6.0393715 52.092914, 6.0393239 52.0929308, 6.039246 52.0929349, 6.0391557 52.0929189),"
+ " (6.0377263 52.0924189, 6.0377571 52.0924499, 6.0377877 52.0924683, 6.0380309 52.092529, 6.0382642 52.0926086, 6.0384412 52.0926911, 6.0384626 52.0927036, 6.0384897 52.0927195, 6.038578 52.0927855, 6.0387045 52.0928107, 6.0388667 52.0928373, 6.0391557 52.0929189, 6.039246 52.0929349, 6.0393239 52.0929308, 6.0393715 52.092914, 6.0393774 52.0928918, 6.0393437 52.0928496, 6.0391495 52.0926998, 6.0387875 52.0925001, 6.0386452 52.0924386, 6.0385632 52.0923458, 6.0385416 52.0922588, 6.0385012 52.0922528, 6.0383203 52.0923097, 6.037857 52.0923747, 6.0377263 52.0924189))"; + " (6.0377263 52.0924189, 6.0377571 52.0924499, 6.0377877 52.0924683, 6.0380309 52.092529, 6.0382642 52.0926086, 6.0384412 52.0926911, 6.0384626 52.0927036, 6.0384897 52.0927195, 6.038578 52.0927855, 6.0387045 52.0928107, 6.0388667 52.0928373, 6.0391557 52.0929189, 6.039246 52.0929349, 6.0393239 52.0929308, 6.0393715 52.092914, 6.0393774 52.0928918, 6.0393437 52.0928496, 6.0391495 52.0926998, 6.0387875 52.0925001, 6.0386452 52.0924386, 6.0385632 52.0923458, 6.0385416 52.0922588, 6.0385012 52.0922528, 6.0383203 52.0923097, 6.037857 52.0923747, 6.0377263 52.0924189))";
Polygon polygon = (Polygon) SimpleWKTShapeParser.parse(wkt); Polygon polygon = (Polygon) SimpleWKTShapeParser.parse(wkt);
{
IllegalArgumentException ex = IllegalArgumentException ex =
expectThrows(IllegalArgumentException.class, () -> Tessellator.tessellate(polygon, true)); expectThrows(IllegalArgumentException.class, () -> Tessellator.tessellate(polygon, true));
assertEquals("Polygon ring self-intersection at lat=52.0924189 lon=6.0377263", ex.getMessage()); assertEquals(
"Polygon ring self-intersection at lat=52.0924189 lon=6.0377263", ex.getMessage());
}
{
IllegalArgumentException ex =
expectThrows(
IllegalArgumentException.class, () -> Tessellator.tessellate(polygon, false));
assertEquals(
"Unable to Tessellate shape. Possible malformed shape detected.", ex.getMessage());
}
} }
public void testLUCENE8559() throws Exception { public void testLUCENE8559() throws Exception {
@ -737,6 +756,60 @@ public class TestTessellator extends LuceneTestCase {
} }
} }
public void testComplexPolygon45() throws Exception {
String geoJson = GeoTestUtil.readShape("lucene-10470.geojson.gz");
Polygon[] polygons = Polygon.fromGeoJSON(geoJson);
for (int i = 0; i < polygons.length; i++) {
List<Tessellator.Triangle> tessellation =
Tessellator.tessellate(polygons[i], random().nextBoolean());
// calculate the area of big polygons have numerical error
assertEquals(area(polygons[i]), area(tessellation), 1e-11);
for (Tessellator.Triangle t : tessellation) {
checkTriangleEdgesFromPolygon(polygons[i], t);
}
}
}
public void testComplexPolygon46() throws Exception {
String wkt = GeoTestUtil.readShape("lucene-10470.wkt.gz");
Polygon polygon = (Polygon) SimpleWKTShapeParser.parse(wkt);
List<Tessellator.Triangle> tessellation =
Tessellator.tessellate(polygon, random().nextBoolean());
// calculate the area of big polygons have numerical error
assertEquals(area(polygon), area(tessellation), 1e-11);
for (Tessellator.Triangle t : tessellation) {
checkTriangleEdgesFromPolygon(polygon, t);
}
}
public void testComplexPolygon47() throws Exception {
String geoJson = GeoTestUtil.readShape("lucene-10470-2.geojson.gz");
Polygon[] polygons = Polygon.fromGeoJSON(geoJson);
for (int i = 0; i < polygons.length; i++) {
List<Tessellator.Triangle> tessellation =
Tessellator.tessellate(polygons[i], random().nextBoolean());
// calculate the area of big polygons have numerical error
assertEquals(area(polygons[i]), area(tessellation), 1e-11);
for (Tessellator.Triangle t : tessellation) {
checkTriangleEdgesFromPolygon(polygons[i], t);
}
}
}
@Nightly
public void testComplexPolygon48() throws Exception {
String geoJson = GeoTestUtil.readShape("lucene-10470-3.geojson.gz");
Polygon[] polygons = Polygon.fromGeoJSON(geoJson);
for (int i = 0; i < polygons.length; i++) {
List<Tessellator.Triangle> tessellation = Tessellator.tessellate(polygons[i], true);
// calculate the area of big polygons have numerical error
assertEquals(area(polygons[i]), area(tessellation), 1e-11);
for (Tessellator.Triangle t : tessellation) {
checkTriangleEdgesFromPolygon(polygons[i], t);
}
}
}
private void checkPolygon(String wkt) throws Exception { private void checkPolygon(String wkt) throws Exception {
Polygon polygon = (Polygon) SimpleWKTShapeParser.parse(wkt); Polygon polygon = (Polygon) SimpleWKTShapeParser.parse(wkt);
List<Tessellator.Triangle> tessellation = List<Tessellator.Triangle> tessellation =