/* Test boundary conditions for $minDistance option for $near and $nearSphere * queries. SERVER-9395. */ var t = db.geo_mindistance_boundaries; t.drop(); t.insert({loc: [1, 0]}); // 1 degree of longitude from origin. /* $minDistance is supported for 2dsphere index only, not 2d or geoHaystack. */ t.ensureIndex({loc: "2dsphere"}); // // Useful constants. // var km = 1000, earthRadiusMeters = 6378.1 * km, geoJSONPoint = {type: 'Point', coordinates: [0, 0]}, // One degree of longitude at the equator, about 111 km. degreeInMeters = 2 * Math.PI * earthRadiusMeters / 360, metersEpsilon = Number.MIN_VALUE; /* Grow epsilon's exponent until epsilon exceeds the margin of error for the * representation of degreeInMeters. The server uses 64-bit math, too, so we'll * find the smallest epsilon the server can detect. */ while (degreeInMeters + metersEpsilon == degreeInMeters) { metersEpsilon *= 2; } // // Test boundary conditions for $near and GeoJSON, in meters. // // minDistance must be within the args to $near, not on the side. assert.throws(function() { t.find({loc:{$near:{$geometry: geoJSONPoint}, $minDistance:0.1}}).itcount();}); assert.eq( 1, t.find({loc: { $near: {$geometry: geoJSONPoint, $minDistance: degreeInMeters }}}).itcount(), "Expected to find (0, 1) within $minDistance 1 degree from origin" ); assert.eq( 1, t.find({loc: { $near: {$geometry: geoJSONPoint, $minDistance: degreeInMeters - metersEpsilon }}}).itcount(), "Expected to find (0, 1) within $minDistance (1 degree - epsilon) from origin" ); assert.eq( 0, t.find({loc: { $near: {$geometry: geoJSONPoint, $minDistance: degreeInMeters + metersEpsilon }}}).itcount(), "Expected *not* to find (0, 1) within $minDistance (1 degree + epsilon) from origin" ); // // Test boundary conditions for $nearSphere and GeoJSON, in meters. // assert.eq( 1, t.find({loc: { $nearSphere: {$geometry: geoJSONPoint, $minDistance: degreeInMeters }}}).itcount(), "Expected to find (0, 1) within $minDistance 1 degree from origin" ); assert.eq( 1, t.find({loc: { $nearSphere: geoJSONPoint, $minDistance: degreeInMeters - metersEpsilon }}).itcount(), "Expected to find (0, 1) within $minDistance (1 degree - epsilon) from origin" ); assert.eq( 0, t.find({loc: { $nearSphere: geoJSONPoint, $minDistance: degreeInMeters + metersEpsilon }}).itcount(), "Expected *not* to find (0, 1) within $minDistance (1 degree + epsilon) from origin" ); // // Test boundary conditions for $nearSphere and a legacy point, in radians. // // $minDistance with legacy point requires $nearSphere; $near not // supported. // var legacyPoint = [0, 0], degreeInRadians = 2 * Math.PI / 360, radiansEpsilon = Number.MIN_VALUE; while (1 + radiansEpsilon == 1) { radiansEpsilon *= 2; } assert.eq( 1, t.find({loc: { $nearSphere: legacyPoint, $minDistance: degreeInRadians }}).itcount(), "Expected to find (0, 1) within $minDistance 1 degree from origin" ); assert.eq( 1, t.find({loc: { $nearSphere: legacyPoint, $minDistance: degreeInRadians - radiansEpsilon }}).itcount(), "Expected to find (0, 1) within $minDistance (1 degree - epsilon) from origin" ); assert.eq( 0, t.find({loc: { $nearSphere: legacyPoint, $minDistance: degreeInRadians + radiansEpsilon }}).itcount(), "Expected *not* to find (0, 1) within $minDistance (1 degree + epsilon) from origin" );