LUCENE-7241: Another round of tree debugging, and hook large polygons up to the random tester.

This commit is contained in:
Karl Wright 2016-05-03 02:07:59 -04:00
parent dac044c94a
commit 0a0037517b
3 changed files with 143 additions and 15 deletions

View File

@ -445,6 +445,7 @@ class GeoComplexPolygon extends GeoBasePolygon {
protected static final int OVERLAPS_MAXIMUM = 3; protected static final int OVERLAPS_MAXIMUM = 3;
protected static final int LESS = 4; protected static final int LESS = 4;
protected static final int GREATER = 5; protected static final int GREATER = 5;
protected static final int EXACT = 6;
/** Add a new edge to the tree. /** Add a new edge to the tree.
* @param edge is the edge to add. * @param edge is the edge to add.
@ -484,11 +485,23 @@ class GeoComplexPolygon extends GeoBasePolygon {
int result = compareForAdd(node.minimumValue, node.maximumValue, minimumValue, maximumValue); int result = compareForAdd(node.minimumValue, node.maximumValue, minimumValue, maximumValue);
switch (result) { switch (result) {
case CONTAINED: case CONTAINED:
// The node is contained in the range provided. We need to create a new node and insert {
final double lesserMaximum = Math.nextDown(node.minimumValue);
final double greaterMinimum = Math.nextUp(node.maximumValue);
node.lesser = addEdge(node.lesser, newEdge, minimumValue, lesserMaximum);
node.greater = addEdge(node.greater, newEdge, greaterMinimum, maximumValue);
return addEdge(node, newEdge, node.minimumValue, node.maximumValue);
}
case EXACT:
// The node is exactly equal to the range provided. We need to create a new node and insert
// it into the "within" chain. // it into the "within" chain.
final Node rval = new Node(newEdge, minimumValue, maximumValue); final Node rval = new Node(newEdge, minimumValue, maximumValue);
//System.err.println(" Inserting new node "+rval+" at head of current 'within' chain in tree "+this); //System.err.println(" Inserting new node "+rval+" at head of current 'within' chain in tree "+this);
rval.within = node; rval.within = node;
rval.lesser = node.lesser;
rval.greater = node.greater;
node.lesser = null;
node.greater = null;
return rval; return rval;
case WITHIN: case WITHIN:
// The new edge is within the node provided // The new edge is within the node provided
@ -496,19 +509,23 @@ class GeoComplexPolygon extends GeoBasePolygon {
node.within = addEdge(node.within, newEdge, minimumValue, maximumValue); node.within = addEdge(node.within, newEdge, minimumValue, maximumValue);
return node; return node;
case OVERLAPS_MINIMUM: case OVERLAPS_MINIMUM:
// The new edge overlaps the minimum value, but not the maximum value. {
// Here we need to create TWO entries: one for the lesser side, and one for the within chain. // The new edge overlaps the minimum value, but not the maximum value.
//System.err.println(" Inserting edge into BOTH lesser chain and within chain in tree "+this); // Here we need to create TWO entries: one for the lesser side, and one for the within chain.
final double lesserMaximum = Math.nextDown(node.minimumValue); //System.err.println(" Inserting edge into BOTH lesser chain and within chain in tree "+this);
node.lesser = addEdge(node.lesser, newEdge, minimumValue, lesserMaximum); final double lesserMaximum = Math.nextDown(node.minimumValue);
return addEdge(node, newEdge, node.minimumValue, maximumValue); node.lesser = addEdge(node.lesser, newEdge, minimumValue, lesserMaximum);
return addEdge(node, newEdge, node.minimumValue, maximumValue);
}
case OVERLAPS_MAXIMUM: case OVERLAPS_MAXIMUM:
// The new edge overlaps the maximum value, but not the minimum value. {
// Need to create two entries, one on the greater side, and one back into the current node. // The new edge overlaps the maximum value, but not the minimum value.
//System.err.println(" Inserting edge into BOTH greater chain and within chain in tree "+this); // Need to create two entries, one on the greater side, and one back into the current node.
final double greaterMinimum = Math.nextUp(node.maximumValue); //System.err.println(" Inserting edge into BOTH greater chain and within chain in tree "+this);
node.greater = addEdge(node.greater, newEdge, greaterMinimum, maximumValue); final double greaterMinimum = Math.nextUp(node.maximumValue);
return addEdge(node, newEdge, minimumValue, node.maximumValue); node.greater = addEdge(node.greater, newEdge, greaterMinimum, maximumValue);
return addEdge(node, newEdge, minimumValue, node.maximumValue);
}
case LESS: case LESS:
// The new edge is clearly less than the current node. // The new edge is clearly less than the current node.
//System.err.println(" Edge goes into the lesser chain in tree "+this); //System.err.println(" Edge goes into the lesser chain in tree "+this);
@ -606,7 +623,9 @@ class GeoComplexPolygon extends GeoBasePolygon {
* @return the comparison result. * @return the comparison result.
*/ */
protected int compareForAdd(final double nodeMinimumValue, final double nodeMaximumValue, final double minimumValue, final double maximumValue) { protected int compareForAdd(final double nodeMinimumValue, final double nodeMaximumValue, final double minimumValue, final double maximumValue) {
if (minimumValue <= nodeMinimumValue && maximumValue >= nodeMaximumValue) { if (minimumValue == nodeMinimumValue && maximumValue == nodeMaximumValue) {
return EXACT;
} else if (minimumValue <= nodeMinimumValue && maximumValue >= nodeMaximumValue) {
return CONTAINED; return CONTAINED;
} else if (nodeMinimumValue <= minimumValue && nodeMaximumValue >= maximumValue) { } else if (nodeMinimumValue <= minimumValue && nodeMaximumValue >= maximumValue) {
return WITHIN; return WITHIN;

View File

@ -538,8 +538,24 @@ public class TestGeo3DPoint extends LuceneTestCase {
private static Query random3DQuery(final String field) { private static Query random3DQuery(final String field) {
while (true) { while (true) {
final int shapeType = random().nextInt(4); final int shapeType = random().nextInt(5);
switch (shapeType) { switch (shapeType) {
case 4: {
// Large polygons
final boolean isClockwise = random().nextDouble() < 0.5;
try {
final Query q = Geo3DPoint.newLargePolygonQuery(field, makePoly(PlanetModel.WGS84,
new GeoPoint(PlanetModel.WGS84, toRadians(GeoTestUtil.nextLatitude()), toRadians(GeoTestUtil.nextLongitude())),
isClockwise,
true));
//System.err.println("Generated: "+q);
//assertTrue(false);
return q;
} catch (IllegalArgumentException e) {
continue;
}
}
case 0: { case 0: {
// Polygons // Polygons
final boolean isClockwise = random().nextDouble() < 0.5; final boolean isClockwise = random().nextDouble() < 0.5;

View File

@ -733,4 +733,97 @@ shape:
} }
@Test
public void testLargePolygonFailureCase2() {
/*
[junit4] > Throwable #1: java.lang.AssertionError: FAIL: id=2 should have matched but did not
[junit4] > shape=GeoComplexPolygon: {planetmodel=PlanetModel.WGS84, number of shapes=1, address=6eccd33b,
testPoint=[lat=0.03170690566178683, lon=1.0862414976732029([X=0.46609969117964495, Y=0.8854242006628827, Z=0.0317369552646047])],
testPointInSet=false,
shapes={ {
[lat=1.0774842300167298, lon=-0.11534121538553185([X=0.46969930266058374, Y=-0.054417217622152375, Z=0.8794587218580684])],
[lat=0.05101544777239065, lon=1.031558236908661([X=0.5133835679471972, Y=0.8579350866926241, Z=0.051049928818862174])],
[lat=-0.011222928649880962, lon=1.5851249038356199([X=-0.01434320835886277, Y=1.0009526216234983, Z=-0.011235244842183226])],
[lat=-0.02571365137215876, lon=0.5627875521419741([X=0.8464356149277266, Y=0.5339650936800929, Z=-0.025739527171261035])],
[lat=0.03833766792865358, lon=1.0082901344798614([X=0.5335096521470836, Y=0.8462411929752105, Z=0.03837097111317845])],
[lat=0.1719054969347345, lon=0.9024290407832926([X=0.6111941952395734, Y=0.7740553755547761, Z=0.17123457719021212])],
[lat=0.08180947807010808, lon=1.0107147265848113([X=0.5300590148023426, Y=0.8453039531721928, Z=0.08180784289673602])]}}
[junit4] > bounds=XYZBounds: [xmin=-1.0011188544924792 xmax=1.0011188544924792
ymin=-1.0011188544924792 ymax=1.0011188544924792
zmin=-0.025739527671261034 zmax=0.9977622925221051]
[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.477874179571219, Y=0.5908091335156603, Z=-0.6495967142221521] within shape? true within bounds? false
[junit4] > unquantized point=[lat=-0.7073124559987376, lon=2.2509085326629887([X=-0.47787417938801546, Y=0.5908091336704123, Z=-0.6495967140640758])] within shape? true within bounds? false
[junit4] > docID=2 deleted?=false
[junit4] > query=PointInGeo3DShapeQuery: field=point: Shape: GeoComplexPolygon: {planetmodel=PlanetModel.WGS84, number of shapes=1, address=6eccd33b, testPoint=[lat=0.03170690566178683, lon=1.0862414976732029([X=0.46609969117964495, Y=0.8854242006628827, Z=0.0317369552646047])], testPointInSet=false, shapes={ {[lat=1.0774842300167298, lon=-0.11534121538553185([X=0.46969930266058374, Y=-0.054417217622152375, Z=0.8794587218580684])], [lat=0.05101544777239065, lon=1.031558236908661([X=0.5133835679471972, Y=0.8579350866926241, Z=0.051049928818862174])], [lat=-0.011222928649880962, lon=1.5851249038356199([X=-0.01434320835886277, Y=1.0009526216234983, Z=-0.011235244842183226])], [lat=-0.02571365137215876, lon=0.5627875521419741([X=0.8464356149277266, Y=0.5339650936800929, Z=-0.025739527171261035])], [lat=0.03833766792865358, lon=1.0082901344798614([X=0.5335096521470836, Y=0.8462411929752105, Z=0.03837097111317845])], [lat=0.1719054969347345, lon=0.9024290407832926([X=0.6111941952395734, Y=0.7740553755547761, Z=0.17123457719021212])], [lat=0.08180947807010808, lon=1.0107147265848113([X=0.5300590148023426, Y=0.8453039531721928, Z=0.08180784289673602])]}}
[junit4] > explanation:
[junit4] > target is in leaf _0(7.0.0):C11 of full reader StandardDirectoryReader(segments:3:nrt _0(7.0.0):C11)
[junit4] > full BKD path to target doc:
[junit4] > Cell(x=-0.8906255176936849 TO 1.0005089994430834 y=-0.6808995306272861 TO 0.9675171153117977 z=-0.997762292058209 TO 0.9939318087373729); Shape relationship = OVERLAPS; Quantized point within cell = true; Unquantized point within cell = true
[junit4] > on cell Cell(x=-0.8906255176936849 TO 1.0005089994430834 y=-0.6808995306272861 TO 0.9675171153117977 z=-0.997762292058209 TO 0.9939318087373729); Shape relationship = OVERLAPS; Quantized point within cell = true; Unquantized point within cell = true, wrapped visitor returned CELL_CROSSES_QUERY
[junit4] > leaf visit docID=2 x=-0.477874179571219 y=0.5908091335156603 z=-0.6495967142221521
*/
final GeoPoint testPoint = new GeoPoint(PlanetModel.WGS84, 0.03170690566178683, 1.0862414976732029);
final boolean testPointInSet = false;
final List<GeoPoint> pointList = new ArrayList<>();
// If the 1.07748... line is at the top, the bounds are correct and the test succeeds.
// If this line is at the bottom, though, the bounds are wrong and the test fails.
//pointList.add(new GeoPoint(PlanetModel.WGS84, 1.0774842300167298, -0.11534121538553185));
pointList.add(new GeoPoint(PlanetModel.WGS84, 0.05101544777239065, 1.031558236908661));
pointList.add(new GeoPoint(PlanetModel.WGS84, -0.011222928649880962, 1.5851249038356199));
pointList.add(new GeoPoint(PlanetModel.WGS84, -0.02571365137215876, 0.5627875521419741));
pointList.add(new GeoPoint(PlanetModel.WGS84, 0.03833766792865358, 1.0082901344798614));
pointList.add(new GeoPoint(PlanetModel.WGS84, 0.1719054969347345, 0.9024290407832926));
pointList.add(new GeoPoint(PlanetModel.WGS84, 0.08180947807010808, 1.0107147265848113));
pointList.add(new GeoPoint(PlanetModel.WGS84, 1.0774842300167298, -0.11534121538553185));
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);
//System.err.println(p);
/*
[junit4] 2> GeoComplexPolygon: {planetmodel=PlanetModel.WGS84, number of shapes=1, address=dcf3e99,
testPoint=[lat=0.03170690566178683, lon=1.0862414976732029([X=0.46609969117964506, Y=0.8854242006628825, Z=0.0317369552646047])],
testPointInSet=false,
shapes={ {
[lat=1.0774842300167298, lon=-0.11534121538553185([X=0.46969930266058374, Y=-0.054417217622152375, Z=0.8794587218580684])],
[lat=0.05101544777239065, lon=1.031558236908661([X=0.5133835679471972, Y=0.8579350866926241, Z=0.051049928818862174])],
[lat=-0.011222928649880962, lon=1.5851249038356199([X=-0.01434320835886277, Y=1.0009526216234983, Z=-0.011235244842183226])],
[lat=-0.02571365137215876, lon=0.5627875521419741([X=0.8464356149277266, Y=0.5339650936800929, Z=-0.025739527171261035])],
[lat=0.03833766792865358, lon=1.0082901344798614([X=0.5335096521470836, Y=0.8462411929752105, Z=0.03837097111317845])],
[lat=0.1719054969347345, lon=0.9024290407832926([X=0.6111941952395734, Y=0.7740553755547761, Z=0.17123457719021212])]}}
[lat=0.08180947807010808, lon=1.0107147265848113([X=0.5300590148023426, Y=0.8453039531721928, Z=0.08180784289673602])],
*/
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);
final XYZSolid solid = XYZSolidFactory.makeXYZSolid(PlanetModel.WGS84,
actualBounds.getMinimumX(), actualBounds.getMaximumX(),
actualBounds.getMinimumY(), actualBounds.getMaximumY(),
actualBounds.getMinimumZ(), actualBounds.getMaximumZ());
final GeoPoint checkPoint = new GeoPoint(PlanetModel.WGS84, -0.7073124559987376, 2.2509085326629887);
// Given the choice of test point, does this all make sense?
assertTrue(pSanity.isWithin(checkPoint) == p.isWithin(checkPoint));
assertTrue(p.isWithin(checkPoint));
assertTrue(solid.isWithin(checkPoint));
}
} }