mirror of https://github.com/apache/lucene.git
LUCENE-7241: Fix intersection bounding so we don't get spurious non-matching crossings.
This commit is contained in:
parent
645889f6b2
commit
d7752408db
|
@ -923,31 +923,35 @@ class GeoComplexPolygon extends GeoBasePolygon {
|
||||||
|
|
||||||
this.testPointCutoffPlane = new SidedPlane(intersectionPoint, testPointPlane, testPoint);
|
this.testPointCutoffPlane = new SidedPlane(intersectionPoint, testPointPlane, testPoint);
|
||||||
this.checkPointCutoffPlane = new SidedPlane(intersectionPoint, travelPlane, thePoint);
|
this.checkPointCutoffPlane = new SidedPlane(intersectionPoint, travelPlane, thePoint);
|
||||||
|
this.testPointOtherCutoffPlane = new SidedPlane(testPoint, testPointPlane, intersectionPoint);
|
||||||
|
this.checkPointOtherCutoffPlane = new SidedPlane(thePoint, travelPlane, intersectionPoint);
|
||||||
|
|
||||||
// Convert travel plane to a sided plane
|
// Convert travel plane to a sided plane
|
||||||
this.testPointOtherCutoffPlane = new SidedPlane(testPoint, travelPlane, travelPlane.D);
|
final Membership intersectionBound1 = new SidedPlane(testPoint, travelPlane, travelPlane.D);
|
||||||
// Convert testPoint plane to a sided plane
|
// Convert testPoint plane to a sided plane
|
||||||
this.checkPointOtherCutoffPlane = new SidedPlane(thePoint, testPointPlane, testPointPlane.D);
|
final Membership intersectionBound2 = new SidedPlane(thePoint, testPointPlane, testPointPlane.D);
|
||||||
|
|
||||||
// Sanity check
|
// Sanity check
|
||||||
assert testPointCutoffPlane.isWithin(intersectionPoint) : "intersection must be within testPointCutoffPlane";
|
assert testPointCutoffPlane.isWithin(intersectionPoint) : "intersection must be within testPointCutoffPlane";
|
||||||
assert testPointOtherCutoffPlane.isWithin(intersectionPoint) : "intersection must be within testPointOtherCutoffPlane";
|
assert testPointOtherCutoffPlane.isWithin(intersectionPoint) : "intersection must be within testPointOtherCutoffPlane";
|
||||||
assert checkPointCutoffPlane.isWithin(intersectionPoint) : "intersection must be within checkPointCutoffPlane";
|
assert checkPointCutoffPlane.isWithin(intersectionPoint) : "intersection must be within checkPointCutoffPlane";
|
||||||
assert checkPointOtherCutoffPlane.isWithin(intersectionPoint) : "intersection must be within checkPointOtherCutoffPlane";
|
assert checkPointOtherCutoffPlane.isWithin(intersectionPoint) : "intersection must be within checkPointOtherCutoffPlane";
|
||||||
|
assert intersectionBound1.isWithin(intersectionPoint) : "intersection must be within intersectionBound1";
|
||||||
|
assert intersectionBound2.isWithin(intersectionPoint) : "intersection must be within intersectionBound2";
|
||||||
|
|
||||||
// Figure out which of the above/below planes are inside vs. outside. To do this,
|
// Figure out which of the above/below planes are inside vs. outside. To do this,
|
||||||
// we look for the point that is within the bounds of the testPointPlane and travelPlane. The two sides that intersected there are the inside
|
// we look for the point that is within the bounds of the testPointPlane and travelPlane. The two sides that intersected there are the inside
|
||||||
// borders.
|
// borders.
|
||||||
final Plane travelAbovePlane = new Plane(travelPlane, true);
|
final Plane travelAbovePlane = new Plane(travelPlane, true);
|
||||||
final Plane travelBelowPlane = new Plane(travelPlane, false);
|
final Plane travelBelowPlane = new Plane(travelPlane, false);
|
||||||
|
|
||||||
final GeoPoint[] aboveAbove = travelAbovePlane.findIntersections(planetModel, testPointAbovePlane, testPointOtherCutoffPlane, checkPointOtherCutoffPlane);
|
final GeoPoint[] aboveAbove = travelAbovePlane.findIntersections(planetModel, testPointAbovePlane, intersectionBound1, intersectionBound2);
|
||||||
assert aboveAbove != null : "Above + above should not be coplanar";
|
assert aboveAbove != null : "Above + above should not be coplanar";
|
||||||
final GeoPoint[] aboveBelow = travelAbovePlane.findIntersections(planetModel, testPointBelowPlane, testPointOtherCutoffPlane, checkPointOtherCutoffPlane);
|
final GeoPoint[] aboveBelow = travelAbovePlane.findIntersections(planetModel, testPointBelowPlane, intersectionBound1, intersectionBound2);
|
||||||
assert aboveBelow != null : "Above + below should not be coplanar";
|
assert aboveBelow != null : "Above + below should not be coplanar";
|
||||||
final GeoPoint[] belowBelow = travelBelowPlane.findIntersections(planetModel, testPointBelowPlane, testPointOtherCutoffPlane, checkPointOtherCutoffPlane);
|
final GeoPoint[] belowBelow = travelBelowPlane.findIntersections(planetModel, testPointBelowPlane, intersectionBound1, intersectionBound2);
|
||||||
assert belowBelow != null : "Below + below should not be coplanar";
|
assert belowBelow != null : "Below + below should not be coplanar";
|
||||||
final GeoPoint[] belowAbove = travelBelowPlane.findIntersections(planetModel, testPointAbovePlane, testPointOtherCutoffPlane, checkPointOtherCutoffPlane);
|
final GeoPoint[] belowAbove = travelBelowPlane.findIntersections(planetModel, testPointAbovePlane, intersectionBound1, intersectionBound2);
|
||||||
assert belowAbove != null : "Below + above should not be coplanar";
|
assert belowAbove != null : "Below + above should not be coplanar";
|
||||||
|
|
||||||
assert ((aboveAbove.length > 0)?1:0) + ((aboveBelow.length > 0)?1:0) + ((belowBelow.length > 0)?1:0) + ((belowAbove.length > 0)?1:0) == 1 : "Can be exactly one inside point, instead was: aa="+aboveAbove.length+" ab=" + aboveBelow.length+" bb="+ belowBelow.length+" ba=" + belowAbove.length;
|
assert ((aboveAbove.length > 0)?1:0) + ((aboveBelow.length > 0)?1:0) + ((belowBelow.length > 0)?1:0) + ((belowAbove.length > 0)?1:0) == 1 : "Can be exactly one inside point, instead was: aa="+aboveAbove.length+" ab=" + aboveBelow.length+" bb="+ belowBelow.length+" ba=" + belowAbove.length;
|
||||||
|
|
|
@ -885,7 +885,7 @@ public class Plane extends Vector {
|
||||||
GeoPoint point2 = new GeoPoint(lineVector.x * t2 + x0, lineVector.y * t2 + y0, lineVector.z * t2 + z0);
|
GeoPoint point2 = new GeoPoint(lineVector.x * t2 + x0, lineVector.y * t2 + y0, lineVector.z * t2 + z0);
|
||||||
//verifyPoint(planetModel, point1, q);
|
//verifyPoint(planetModel, point1, q);
|
||||||
//verifyPoint(planetModel, point2, q);
|
//verifyPoint(planetModel, point2, q);
|
||||||
//System.err.println(" "+point1+" and "+point2);
|
//System.err.println(" Considering points "+point1+" and "+point2);
|
||||||
if (point1.isWithin(bounds, moreBounds)) {
|
if (point1.isWithin(bounds, moreBounds)) {
|
||||||
if (point2.isWithin(bounds, moreBounds))
|
if (point2.isWithin(bounds, moreBounds))
|
||||||
return new GeoPoint[]{point1, point2};
|
return new GeoPoint[]{point1, point2};
|
||||||
|
|
|
@ -239,6 +239,7 @@ public class GeoPolygonTest {
|
||||||
shapes.add(new GeoPolygonFactory.PolygonDescription(points));
|
shapes.add(new GeoPolygonFactory.PolygonDescription(points));
|
||||||
|
|
||||||
c = GeoPolygonFactory.makeLargeGeoPolygon(PlanetModel.SPHERE, shapes);
|
c = GeoPolygonFactory.makeLargeGeoPolygon(PlanetModel.SPHERE, shapes);
|
||||||
|
|
||||||
// Sample some points within
|
// Sample some points within
|
||||||
gp = new GeoPoint(PlanetModel.SPHERE, 0.0, -0.45);
|
gp = new GeoPoint(PlanetModel.SPHERE, 0.0, -0.45);
|
||||||
assertTrue(c.isWithin(gp));
|
assertTrue(c.isWithin(gp));
|
||||||
|
@ -666,4 +667,70 @@ shape:
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testLargePolygonFailureCase1() {
|
||||||
|
/*
|
||||||
|
[junit4] > shape=GeoComplexPolygon: {planetmodel=PlanetModel.WGS84, number of shapes=1, address=65f193fc,
|
||||||
|
testPoint=[lat=1.3005550159098878, lon=-2.4043250791032897([X=-0.1972404544647752, Y=-0.17911237095124333, Z=0.9617794725902562])],
|
||||||
|
testPointInSet=false,
|
||||||
|
shapes={
|
||||||
|
{[lat=0.972005250702484, lon=-1.9776473855435277([X=-0.22278290030997686, Y=-0.5170266140533727, Z=0.8250470449472769])],
|
||||||
|
[lat=0.5530477484903267, lon=2.5300578442038137([X=-0.6968439858923609, Y=0.4886310878468911, Z=0.5253825248638686])],
|
||||||
|
[lat=1.5185372097372358, lon=-0.33848566616392867([X=0.04916162127975167, Y=-0.01730656055596007, Z=0.9964092501726799])]}}
|
||||||
|
[junit4] > bounds=XYZBounds: [xmin=-1.0011188544924792 xmax=0.04916162177975167 ymin=-1.0011188544924792 ymax=1.0011188544924792 zmin=-5.0E-10 zmax=0.99766957331525]
|
||||||
|
[junit4] > world bounds=( minX=-1.0011188539924791 maxX=1.0011188539924791 minY=-1.0011188539924791 maxY=1.0011188539924791 minZ=-0.9977622920221051 maxZ=0.9977622920221051
|
||||||
|
[junit4] > quantized point=[X=0.32866145093230836, Y=0.21519085912590594, Z=0.9177348472123349] within shape? true within bounds? false
|
||||||
|
[junit4] > unquantized point=[lat=1.166339260547107, lon=0.5797066870374205([X=0.3286614507856878, Y=0.21519085911319938, Z=0.9177348470779726])] within shape? true within bounds? false
|
||||||
|
[junit4] > docID=10 deleted?=false
|
||||||
|
[junit4] > query=PointInGeo3DShapeQuery: field=point: Shape: GeoComplexPolygon: {planetmodel=PlanetModel.WGS84, number of shapes=1, address=65f193fc, testPoint=[lat=1.3005550159098878, lon=-2.4043250791032897([X=-0.1972404544647752, Y=-0.17911237095124333, Z=0.9617794725902562])], testPointInSet=false, shapes={ {[lat=0.972005250702484, lon=-1.9776473855435277([X=-0.22278290030997686, Y=-0.5170266140533727, Z=0.8250470449472769])], [lat=0.5530477484903267, lon=2.5300578442038137([X=-0.6968439858923609, Y=0.4886310878468911, Z=0.5253825248638686])], [lat=1.5185372097372358, lon=-0.33848566616392867([X=0.04916162127975167, Y=-0.01730656055596007, Z=0.9964092501726799])]}}
|
||||||
|
[junit4] > explanation:
|
||||||
|
[junit4] > target is in leaf _0(7.0.0):c13 of full reader StandardDirectoryReader(segments:3:nrt _0(7.0.0):c13)
|
||||||
|
[junit4] > full BKD path to target doc:
|
||||||
|
[junit4] > Cell(x=-0.9060562472023252 TO 1.0010658113048514 y=-0.5681445384324596 TO 0.7613281936331098 z=-0.43144274682272304 TO 0.9977622920582089); Shape relationship = OVERLAPS; Quantized point within cell = true; Unquantized point within cell = true
|
||||||
|
[junit4] > on cell Cell(x=-0.9060562472023252 TO 1.0010658113048514 y=-0.5681445384324596 TO 0.7613281936331098 z=-0.43144274682272304 TO 0.9977622920582089); Shape relationship = OVERLAPS; Quantized point within cell = true; Unquantized point within cell = true, wrapped visitor returned CELL_CROSSES_QUERY
|
||||||
|
[junit4] > leaf visit docID=10 x=0.32866145093230836 y=0.21519085912590594 z=0.9177348472123349
|
||||||
|
*/
|
||||||
|
final GeoPoint testPoint = new GeoPoint(PlanetModel.WGS84, 1.3005550159098878, -2.4043250791032897);
|
||||||
|
final boolean testPointInSet = false;
|
||||||
|
final List<GeoPoint> pointList = new ArrayList<>();
|
||||||
|
pointList.add(new GeoPoint(PlanetModel.WGS84, 0.972005250702484, -1.9776473855435277));
|
||||||
|
pointList.add(new GeoPoint(PlanetModel.WGS84, 0.5530477484903267, 2.5300578442038137));
|
||||||
|
pointList.add(new GeoPoint(PlanetModel.WGS84, 1.5185372097372358, -0.33848566616392867));
|
||||||
|
|
||||||
|
final GeoPolygon pSanity = GeoPolygonFactory.makeGeoPolygon(PlanetModel.WGS84, pointList);
|
||||||
|
|
||||||
|
assertTrue(pSanity.isWithin(testPoint) == testPointInSet);
|
||||||
|
|
||||||
|
final List<List<GeoPoint>> shapeList = new ArrayList<>();
|
||||||
|
shapeList.add(pointList);
|
||||||
|
final GeoPolygon p = new GeoComplexPolygon(PlanetModel.WGS84, shapeList, testPoint, testPointInSet);
|
||||||
|
|
||||||
|
final GeoPoint intersectionPoint = new GeoPoint(0.26643017529034996, 0.0, 0.9617794725902564);
|
||||||
|
assertTrue(pSanity.isWithin(intersectionPoint) == p.isWithin(intersectionPoint));
|
||||||
|
assertTrue(p.isWithin(intersectionPoint));
|
||||||
|
|
||||||
|
final GeoPoint maxXPoint = new GeoPoint(PlanetModel.WGS84, 0.0, 0.0);
|
||||||
|
|
||||||
|
assertTrue(pSanity.isWithin(maxXPoint) == p.isWithin(maxXPoint));
|
||||||
|
|
||||||
|
final GeoPoint checkPoint = new GeoPoint(PlanetModel.WGS84, 1.166339260547107, 0.5797066870374205);
|
||||||
|
|
||||||
|
// Given the choice of test point, does this all make sense?
|
||||||
|
assertTrue(pSanity.isWithin(checkPoint) == p.isWithin(checkPoint));
|
||||||
|
|
||||||
|
final XYZBounds referenceBounds = new XYZBounds();
|
||||||
|
pSanity.getBounds(referenceBounds);
|
||||||
|
|
||||||
|
final XYZBounds actualBounds = new XYZBounds();
|
||||||
|
p.getBounds(actualBounds);
|
||||||
|
|
||||||
|
assertEquals(referenceBounds.getMinimumX(), actualBounds.getMinimumX(), 0.0000001);
|
||||||
|
assertEquals(referenceBounds.getMaximumX(), actualBounds.getMaximumX(), 0.0000001);
|
||||||
|
assertEquals(referenceBounds.getMinimumY(), actualBounds.getMinimumY(), 0.0000001);
|
||||||
|
assertEquals(referenceBounds.getMaximumY(), actualBounds.getMaximumY(), 0.0000001);
|
||||||
|
assertEquals(referenceBounds.getMinimumZ(), actualBounds.getMinimumZ(), 0.0000001);
|
||||||
|
assertEquals(referenceBounds.getMaximumZ(), actualBounds.getMaximumZ(), 0.0000001);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue