|  | @@ -20,18 +20,17 @@
 | 
	
		
			
				|  |  |  package org.apache.lucene.util;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  import org.elasticsearch.common.geo.GeoDistance;
 | 
	
		
			
				|  |  | -import org.elasticsearch.common.geo.GeoUtils;
 | 
	
		
			
				|  |  |  import org.elasticsearch.common.unit.DistanceUnit;
 | 
	
		
			
				|  |  |  import org.elasticsearch.test.ElasticsearchTestCase;
 | 
	
		
			
				|  |  |  import org.junit.Test;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  import static org.hamcrest.number.IsCloseTo.closeTo;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -public class SloppyMathTests extends ElasticsearchTestCase {
 | 
	
		
			
				|  |  | +public class SloppyMathTests extends ElasticsearchTestCase {    
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      @Test
 | 
	
		
			
				|  |  |      public void testAccuracy() {
 | 
	
		
			
				|  |  | -        for (double lat1 = -90; lat1 <= 90; lat1+=1) {
 | 
	
		
			
				|  |  | +        for (double lat1 = -89; lat1 <= 89; lat1+=1) {
 | 
	
		
			
				|  |  |              final double lon1 = randomLongitude();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |              for (double i = -180; i <= 180; i+=1) {
 | 
	
	
		
			
				|  | @@ -45,12 +44,12 @@ public class SloppyMathTests extends ElasticsearchTestCase {
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      @Test
 | 
	
		
			
				|  |  |      public void testSloppyMath() {
 | 
	
		
			
				|  |  | -        assertThat(GeoDistance.ARC.calculate(-46.645, -171.057, -46.644, -171.058, DistanceUnit.METERS), closeTo(134.87709, maxError(134.87709)));
 | 
	
		
			
				|  |  | -        assertThat(GeoDistance.ARC.calculate(-77.912, -81.173, -77.912, -81.171, DistanceUnit.METERS), closeTo(46.57161, maxError(46.57161)));
 | 
	
		
			
				|  |  | -        assertThat(GeoDistance.ARC.calculate(65.75, -20.708, 65.75, -20.709, DistanceUnit.METERS), closeTo(45.66996, maxError(45.66996)));
 | 
	
		
			
				|  |  | -        assertThat(GeoDistance.ARC.calculate(-86.9, 53.738, -86.9, 53.741, DistanceUnit.METERS), closeTo(18.03998, maxError(18.03998)));
 | 
	
		
			
				|  |  | -        assertThat(GeoDistance.ARC.calculate(89.041, 115.93, 89.04, 115.946, DistanceUnit.METERS), closeTo(115.11711, maxError(115.11711)));
 | 
	
		
			
				|  |  | -        
 | 
	
		
			
				|  |  | +        assertThat(GeoDistance.SLOPPY_ARC.calculate(-46.645, -171.057, -46.644, -171.058, DistanceUnit.METERS), closeTo(134.87709, maxError(134.87709)));
 | 
	
		
			
				|  |  | +        assertThat(GeoDistance.SLOPPY_ARC.calculate(-77.912, -81.173, -77.912, -81.171, DistanceUnit.METERS), closeTo(46.57161, maxError(46.57161)));
 | 
	
		
			
				|  |  | +        assertThat(GeoDistance.SLOPPY_ARC.calculate(65.75, -20.708, 65.75, -20.709, DistanceUnit.METERS), closeTo(45.66996, maxError(45.66996)));
 | 
	
		
			
				|  |  | +        assertThat(GeoDistance.SLOPPY_ARC.calculate(-86.9, 53.738, -86.9, 53.741, DistanceUnit.METERS), closeTo(18.03998, maxError(18.03998)));
 | 
	
		
			
				|  |  | +        assertThat(GeoDistance.SLOPPY_ARC.calculate(89.041, 115.93, 89.04, 115.946, DistanceUnit.METERS), closeTo(115.11711, maxError(115.11711)));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |          testSloppyMath(DistanceUnit.METERS, 0.01, 5, 45, 90);
 | 
	
		
			
				|  |  |          testSloppyMath(DistanceUnit.KILOMETERS, 0.01, 5, 45, 90);
 | 
	
		
			
				|  |  |          testSloppyMath(DistanceUnit.INCH, 0.01, 5, 45, 90);
 | 
	
	
		
			
				|  | @@ -66,50 +65,34 @@ public class SloppyMathTests extends ElasticsearchTestCase {
 | 
	
		
			
				|  |  |          final double lon1 = randomLongitude();
 | 
	
		
			
				|  |  |          logger.info("testing SloppyMath with {} at \"{}, {}\"", unit, lat1, lon1);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -        GeoDistance.ArcFixedSourceDistance src = new GeoDistance.ArcFixedSourceDistance(lat1, lon1, unit); 
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |          for (int test = 0; test < deltaDeg.length; test++) {
 | 
	
		
			
				|  |  |              for (int i = 0; i < 100; i++) {
 | 
	
		
			
				|  |  | +                // crop pole areas, sine we now there the function
 | 
	
		
			
				|  |  | +                // is not accurate around lat(89°, 90°) and lat(-90°, -89°)
 | 
	
		
			
				|  |  | +                final double lat2 = Math.max(-89.0, Math.min(+89.0, lat1 + (randomDouble() - 0.5) * 2 * deltaDeg[test]));
 | 
	
		
			
				|  |  |                  final double lon2 = lon1 + (randomDouble() - 0.5) * 2 * deltaDeg[test];
 | 
	
		
			
				|  |  | -                final double lat2 = lat1 + (randomDouble() - 0.5) * 2 * deltaDeg[test];
 | 
	
		
			
				|  |  | -    
 | 
	
		
			
				|  |  | -                final double accurate = unit.fromMeters(accurateHaversin(lat1, lon1, lat2, lon2));
 | 
	
		
			
				|  |  | -                final double dist1 = GeoDistance.ARC.calculate(lat1, lon1, lat2, lon2, unit);
 | 
	
		
			
				|  |  | -                final double dist2 = src.calculate(lat2, lon2);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                final double accurate = GeoDistance.ARC.calculate(lat1, lon1, lat2, lon2, unit);
 | 
	
		
			
				|  |  | +                final double dist = GeoDistance.SLOPPY_ARC.calculate(lat1, lon1, lat2, lon2, unit);
 | 
	
		
			
				|  |  |      
 | 
	
		
			
				|  |  | -                assertThat("distance between("+lat1+", "+lon1+") and ("+lat2+", "+lon2+"))", dist1, closeTo(accurate, maxError(accurate)));
 | 
	
		
			
				|  |  | -                assertThat("distance between("+lat1+", "+lon1+") and ("+lat2+", "+lon2+"))", dist2, closeTo(accurate, maxError(accurate)));
 | 
	
		
			
				|  |  | +                assertThat("distance between("+lat1+", "+lon1+") and ("+lat2+", "+lon2+"))", dist, closeTo(accurate, maxError(accurate)));
 | 
	
		
			
				|  |  |              }
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  | -    
 | 
	
		
			
				|  |  | -    // Slow but accurate implementation of the haversin function
 | 
	
		
			
				|  |  | -    private static double accurateHaversin(double lat1, double lon1, double lat2, double lon2) {
 | 
	
		
			
				|  |  | -        double longitudeDifference = lon2 - lon1;
 | 
	
		
			
				|  |  | -        double a = Math.toRadians(90D - lat1);
 | 
	
		
			
				|  |  | -        double c = Math.toRadians(90D - lat2);
 | 
	
		
			
				|  |  | -        double factor = (Math.cos(a) * Math.cos(c)) + (Math.sin(a) * Math.sin(c) * Math.cos(Math.toRadians(longitudeDifference)));
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -        if (factor < -1D) {
 | 
	
		
			
				|  |  | -            return Math.PI * GeoUtils.EARTH_MEAN_RADIUS;
 | 
	
		
			
				|  |  | -        } else if (factor >= 1D) {
 | 
	
		
			
				|  |  | -            return 0;
 | 
	
		
			
				|  |  | -        } else {
 | 
	
		
			
				|  |  | -            return Math.acos(factor) * GeoUtils.EARTH_MEAN_RADIUS;
 | 
	
		
			
				|  |  | -        }
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | -    
 | 
	
		
			
				|  |  | +        
 | 
	
		
			
				|  |  |      private static void assertAccurate(double lat1, double lon1, double lat2, double lon2) {
 | 
	
		
			
				|  |  | -        double accurate = accurateHaversin(lat1, lon1, lat2, lon2);
 | 
	
		
			
				|  |  | -        double sloppy = GeoDistance.ARC.calculate(lat1, lon1, lat2, lon2, DistanceUnit.METERS);
 | 
	
		
			
				|  |  | +        double accurate = GeoDistance.ARC.calculate(lat1, lon1, lat2, lon2, DistanceUnit.METERS);
 | 
	
		
			
				|  |  | +        double sloppy = GeoDistance.SLOPPY_ARC.calculate(lat1, lon1, lat2, lon2, DistanceUnit.METERS);
 | 
	
		
			
				|  |  |          assertThat("distance between("+lat1+", "+lon1+") and ("+lat2+", "+lon2+"))", sloppy, closeTo(accurate, maxError(accurate)));
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      private static final double randomLatitude() {
 | 
	
		
			
				|  |  | -        return (getRandom().nextDouble() - 0.5) * 180d;
 | 
	
		
			
				|  |  | +        // crop pole areas, sine we now there the function
 | 
	
		
			
				|  |  | +        // is not accurate around lat(89°, 90°) and lat(-90°, -89°)
 | 
	
		
			
				|  |  | +        return (getRandom().nextDouble() - 0.5) * 178.0;
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      private static final double randomLongitude() {
 | 
	
		
			
				|  |  | -        return (getRandom().nextDouble() - 0.5) * 360d;
 | 
	
		
			
				|  |  | +        return (getRandom().nextDouble() - 0.5) * 360.0;
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  }
 |