213 lines
7.6 KiB
JavaScript
213 lines
7.6 KiB
JavaScript
var t = db.geo_s2overlappingpolys
|
|
t.drop()
|
|
|
|
t.ensureIndex( { geo : "2dsphere" } );
|
|
|
|
var minError = 0.8e-13;
|
|
|
|
var canonPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-1.0, -1.0], [1.0, -1.0], [1.0, 1.0], [-1.0, 1.0], [-1.0, -1.0]]
|
|
]};
|
|
t.insert({geo: canonPoly});
|
|
|
|
// Test 1: If a poly completely encloses the canonPoly, we expect the canonPoly
|
|
// to be returned for both $within and $geoIntersect
|
|
|
|
var outerPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-2.0, -2.0], [2.0, -2.0], [2.0, 2.0], [-2.0, 2.0], [-2.0, -2.0]]
|
|
]};
|
|
var result = t.find({geo: {$within: {$geometry: outerPoly}}});
|
|
assert.eq(result.count(), 1);
|
|
result = t.find({geo: {$geoIntersects: {$geometry: outerPoly}}});
|
|
assert.eq(result.count(), 1);
|
|
|
|
|
|
// Test 2: If a poly that covers half of the canonPoly, we expect that it should
|
|
// geoIntersect, but should not be within.
|
|
|
|
var partialPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-2.0, -2.0], [2.0, -2.0], [2.0, 0.0], [-2.0, 0.0], [-2.0, -2.0]]
|
|
]};
|
|
|
|
//Should not be within
|
|
result = t.find({geo: {$within: {$geometry: partialPoly}}});
|
|
assert.eq(result.count(), 0);
|
|
|
|
//This should however count as a geoIntersect
|
|
result = t.find({geo: {$geoIntersects: {$geometry: partialPoly}}});
|
|
assert.eq(result.count(), 1);
|
|
|
|
|
|
// Test 3: Polygons that intersect at a point or an edge have undefined
|
|
// behaviour in s2 The s2 library we're using appears to have
|
|
// the following behaviour.
|
|
|
|
// Case (a): Polygons that intersect at one point (not a vertex).
|
|
// behaviour: geoIntersects.
|
|
|
|
var sharedPointPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[0.0, -2.0], [0.0, -1.0], [1.0, -2.0], [0.0, -2.0]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: sharedPointPoly}}});
|
|
assert.eq(result.count(), 1);
|
|
|
|
// Case (b): Polygons that intersect at one point (a vertex).
|
|
// behaviour: not geoIntersect
|
|
|
|
var sharedVertexPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[0.0, -2.0], [1.0, -1.0], [1.0, -2.0], [0.0, -2.0]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: sharedVertexPoly}}});
|
|
assert.eq(result.count(), 0);
|
|
|
|
// Case (c): Polygons that intesersect at one point that is very close to a
|
|
// vertex should have the same behaviour as Case (b).
|
|
|
|
var almostSharedVertexPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[0.0, -2.0], [1.0 - minError, -1.0], [1.0, -2.0], [0.0, -2.0]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: almostSharedVertexPoly}}});
|
|
assert.eq(result.count(), 0);
|
|
|
|
|
|
// Case (d): Polygons that intesersect at one point that is not quite as close
|
|
// to a vertex should behave as though it were not a vertex, and should
|
|
// geoIntersect
|
|
|
|
var notCloseEnoughSharedVertexPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[0.0, -2.0], [1.0 - (10 * minError), -1.0], [1.0, -2.0], [0.0, -2.0]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: notCloseEnoughSharedVertexPoly}}});
|
|
assert.eq(result.count(), 1);
|
|
|
|
// Case (e): Polygons that come very close to having a point intersection
|
|
// on a non-vertex coordinate should intersect.
|
|
|
|
var almostSharedPointPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[0.0, -2.0], [0.0, (-1.0 - minError)], [1.0, -2.0], [0.0, -2.0]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: almostSharedPointPoly}}});
|
|
assert.eq(result.count(), 1);
|
|
|
|
|
|
// Case (f): If we increase the error a little, it should no longer act
|
|
// as though it's intersecting.
|
|
// NOTE: I think this error bound seems odd. Going to 0.000152297 will break this test.
|
|
// I've confirmed there is an error bound, but it's a lot larger than we experienced above.
|
|
var errorBound = 0.000152298
|
|
var notCloseEnoughSharedPointPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[0.0, -2.0], [0.0, -1.0 - errorBound], [1.0, -2.0], [0.0, -2.0]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: notCloseEnoughSharedPointPoly}}});
|
|
assert.eq(result.count(), 0);
|
|
|
|
/* Test 3: Importantly, polygons with shared edges have undefined intersection
|
|
* under s2. Therefore these test serve more to make sure nothing changes than
|
|
* to confirm an expected behaviour.
|
|
*/
|
|
|
|
// Case 1: A polygon who shares an edge with another polygon, where the searching
|
|
// polygon's edge is fully covered by the canon polygon's edge.
|
|
// Result: No intersection.
|
|
var fullyCoveredEdgePoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-2.0, -0.5], [-1.0, -0.5], [-1.0, 0.5], [-2.0, 0.5], [-2.0, -0.5]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: fullyCoveredEdgePoly}}});
|
|
assert.eq(result.count(), 0);
|
|
|
|
// Case 2: A polygon who shares an edge with another polygon, where the searching
|
|
// polygon's edge fully covers the canon polygon's edge.
|
|
// Result: Intersection.
|
|
var coveringEdgePoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-2.0, -1.5], [-1.0, -1.5], [-1.0, 1.5], [-2.0, 1.5], [-2.0, -1.5]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: coveringEdgePoly}}});
|
|
assert.eq(result.count(), 1);
|
|
|
|
// Case 2a: same as Case 2, except pulled slightly away from the polygon.
|
|
// Result: Intersection.
|
|
// NOTE: Scales of errors?
|
|
var closebyCoveringEdgePoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-2.0, -1.5], [-1.0 - (minError / 1000), -1.5], [-1.0 - (minError / 1000), 1.5], [-2.0, 1.5], [-2.0, -1.5]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: closebyCoveringEdgePoly}}});
|
|
assert.eq(result.count(), 1);
|
|
|
|
// Case 2b: same as Case 4, except pulled slightly away from the polygon, so that it's not intersecting.
|
|
// Result: No Intersection.
|
|
// NOTE: Scales of errors?
|
|
var notCloseEnoughCoveringEdgePoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-2.0, -1.5], [-1.0 - (minError / 100), -1.5], [-1.0 - (minError / 100), 1.5], [-2.0, 1.5], [-2.0, -1.5]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: notCloseEnoughCoveringEdgePoly}}});
|
|
assert.eq(result.count(), 0);
|
|
|
|
// Case 3: A polygon who shares an edge with another polygon, where the searching
|
|
// polygon's edge partially covers by the canon polygon's edge.
|
|
// Result: No intersection.
|
|
var partiallyCoveringEdgePoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-2.0, -1.5], [-1.0, -1.5], [-1.0, 0.5], [-2.0, 0.5], [-2.0, -1.5]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: partiallyCoveringEdgePoly}}});
|
|
assert.eq(result.count(), 0);
|
|
|
|
|
|
//Polygons that intersect at three non-co-linear points should geoIntersect
|
|
var sharedPointsPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[0.0, -3.0], [0.0, -1.0], [2.0, -2.0], [1.0, 0.0], [2.0, 2.0], [0.0, 1.0], [0.0, 3.0], [3.0, 3.0], [3.0, -3.0], [0.0, -3.0]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: sharedPointsPoly}}});
|
|
assert.eq(result.count(), 1);
|
|
|
|
//If a polygon contains a hole, and another polygon is within that hole, it should not be within or intersect.
|
|
|
|
var bigHolePoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-3.0, -3.0], [3.0, -3.0], [3.0, 3.0], [-3.0, 3.0], [-3.0, -3.0]],
|
|
[[-2.0, -2.0], [2.0, -2.0], [2.0, 2.0], [-2.0, 2.0], [-2.0, -2.0]]
|
|
]};
|
|
result = t.find({geo: {$within: {$geometry: bigHolePoly}}});
|
|
assert.eq(result.count(), 0);
|
|
result = t.find({geo: {$geoIntersects: {$geometry: bigHolePoly}}});
|
|
assert.eq(result.count(), 0);
|
|
|
|
// If a polygon has a hole, and another polygon is contained partially by that hole, it should be an intersection
|
|
// but not a within.
|
|
|
|
var internalOverlapPoly = {type: "Polygon",
|
|
coordinates: [
|
|
[[-3.0, -3.0], [3.0, -3.0], [3.0, 3.0], [-3.0, 3.0], [-3.0, -3.0]],
|
|
[[-2.0, 0.0], [2.0, 0.0], [2.0, 2.0], [-2.0, 2.0], [-2.0, 0.0]]
|
|
]};
|
|
|
|
result = t.find({geo: {$geoIntersects: {$geometry: internalOverlapPoly}}});
|
|
assert.eq(result.count(), 1);
|
|
result = t.find({geo: {$within: {$geometry: internalOverlapPoly}}});
|
|
assert.eq(result.count(), 0);
|