diff --git a/rugged-core/src/main/java/org/orekit/rugged/core/BasicScanAlgorithm.java b/rugged-core/src/main/java/org/orekit/rugged/core/BasicScanAlgorithm.java index 8836cbe0997a951c0c77827bcdf2419f9176b8c1..5b9af1b17fbf0c096bc6e31f6322cc8a08bcb2f1 100644 --- a/rugged-core/src/main/java/org/orekit/rugged/core/BasicScanAlgorithm.java +++ b/rugged-core/src/main/java/org/orekit/rugged/core/BasicScanAlgorithm.java @@ -120,12 +120,21 @@ public class BasicScanAlgorithm implements IntersectionAlgorithm { for (int j = longitudeIndex(tile, minLongitude); j <= longitudeIndex(tile, maxLongitude); ++j) { GeodeticPoint gp = tile.pixelIntersection(entryPoint, ellipsoid.convertLos(entryPoint, los), i, j); if (gp != null) { + + // improve the point, by projecting it back on the 3D line, fixing the small body curvature at pixel level + final Vector3D delta = ellipsoid.transform(gp).subtract(position); + final double s = Vector3D.dotProduct(delta, los) / los.getNormSq(); + final GeodeticPoint projected = ellipsoid.transform(new Vector3D(1, position, s, los), + ellipsoid.getBodyFrame(), null); + gp = tile.pixelIntersection(projected, ellipsoid.convertLos(projected, los), i, j); + final Vector3D point = ellipsoid.transform(gp); final double dot = Vector3D.dotProduct(point.subtract(position), los); if (dot < intersectionDot) { intersectionGP = gp; intersectionDot = dot; } + } } } diff --git a/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java b/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java index 2f8aea9a7c7f044ab12665fab29fc8781a34aa1d..12e3d106a366f26d0e3f2b545811240fbf4f0fec 100644 --- a/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java +++ b/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java @@ -159,7 +159,16 @@ public class DuvenhageAlgorithm implements IntersectionAlgorithm { if (entryLat == exitLat && entryLon == exitLon) { // we have narrowed the search down to a single Digital Elevation Model pixel - return tile.pixelIntersection(entry, ellipsoid.convertLos(entry, los), exitLat, exitLon); + GeodeticPoint intersection = tile.pixelIntersection(entry, ellipsoid.convertLos(entry, los), exitLat, exitLon); + if (intersection != null) { + // improve the point, by projecting it back on the 3D line, fixing the small body curvature at pixel level + final Vector3D delta = ellipsoid.transform(intersection).subtract(position); + final double s = Vector3D.dotProduct(delta, los) / los.getNormSq(); + final GeodeticPoint projected = ellipsoid.transform(new Vector3D(1, position, s, los), + ellipsoid.getBodyFrame(), null); + intersection = tile.pixelIntersection(projected, ellipsoid.convertLos(projected, los), exitLat, exitLon); + } + return intersection; } // find the deepest level in the min/max kd-tree at which entry and exit share a sub-tile diff --git a/rugged-core/src/test/java/org/orekit/rugged/core/AbstractAlgorithmTest.java b/rugged-core/src/test/java/org/orekit/rugged/core/AbstractAlgorithmTest.java new file mode 100644 index 0000000000000000000000000000000000000000..204260fa2b0ea85d24f9a667eca18b812d322c08 --- /dev/null +++ b/rugged-core/src/test/java/org/orekit/rugged/core/AbstractAlgorithmTest.java @@ -0,0 +1,274 @@ +/* Copyright 2002-2014 CS Systèmes d'Information + * Licensed to CS Systèmes d'Information (CS) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. + * CS licenses this file to You under the Apache License, Version 2.0 + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.orekit.rugged.core; + + +import java.io.File; +import java.net.URISyntaxException; + +import org.apache.commons.math3.geometry.euclidean.threed.Rotation; +import org.apache.commons.math3.geometry.euclidean.threed.Vector3D; +import org.apache.commons.math3.util.FastMath; +import org.junit.After; +import org.junit.Assert; +import org.junit.Before; +import org.junit.Test; +import org.orekit.attitudes.Attitude; +import org.orekit.bodies.GeodeticPoint; +import org.orekit.data.DataProvidersManager; +import org.orekit.data.DirectoryCrawler; +import org.orekit.errors.OrekitException; +import org.orekit.frames.FramesFactory; +import org.orekit.frames.Transform; +import org.orekit.orbits.CartesianOrbit; +import org.orekit.propagation.SpacecraftState; +import org.orekit.rugged.api.RuggedException; +import org.orekit.rugged.api.TileUpdater; +import org.orekit.rugged.core.duvenhage.MinMaxTreeTile; +import org.orekit.rugged.core.duvenhage.MinMaxTreeTileFactory; +import org.orekit.rugged.core.raster.CliffsElevationUpdater; +import org.orekit.rugged.core.raster.IntersectionAlgorithm; +import org.orekit.rugged.core.raster.VolcanicConeElevationUpdater; +import org.orekit.time.AbsoluteDate; +import org.orekit.time.TimeScalesFactory; +import org.orekit.utils.Constants; +import org.orekit.utils.IERSConventions; +import org.orekit.utils.PVCoordinates; + +public abstract class AbstractAlgorithmTest { + + protected abstract IntersectionAlgorithm createAlgorithm(); + + @Test + public void testMayonVolcanoOnSubTileCorner() + throws RuggedException, OrekitException { + + setUpMayonVolcanoContext(); + + // test point approximately 1.6km North-North-West and 800 meters below volcano summit + // note that this test point is EXACTLY at a pixel corner, and even at corners of + // middle level (12 and above) sub-tiles + final double latitude = FastMath.toRadians(13.27); + final double longitude = FastMath.toRadians(123.68); + MinMaxTreeTile tile = new MinMaxTreeTileFactory().createTile(); + updater.updateTile(latitude, longitude, tile); + double altitude = tile.interpolateElevation(latitude, longitude); + final GeodeticPoint groundGP = new GeodeticPoint(latitude, longitude, altitude); + Vector3D groundP = earth.transform(groundGP); + + final IntersectionAlgorithm algorithm = createAlgorithm(); + algorithm.setUpTilesManagement(updater, 8); + + // preliminary check: the point has been chosen in the spacecraft (YZ) plane + Transform earthToSpacecraft = new Transform(state.getDate(), + earth.getBodyFrame().getTransformTo(state.getFrame(), state.getDate()), + state.toTransform()); + Vector3D pointInSpacecraftFrame = earthToSpacecraft.transformPosition(groundP); + Assert.assertEquals( 0.000, pointInSpacecraftFrame.getX(), 1.0e-3); + Assert.assertEquals(-87754.914, pointInSpacecraftFrame.getY(), 1.0e-3); + Assert.assertEquals(790330.254, pointInSpacecraftFrame.getZ(), 1.0e-3); + + // test direct localization + Vector3D position = state.getPVCoordinates(earth.getBodyFrame()).getPosition(); + Vector3D los = groundP.subtract(position); + GeodeticPoint result = algorithm.intersection(earth, position, los); + Assert.assertEquals(0.0, groundP.distance(earth.transform(result)), 2.0e-9); + + } + + @Test + public void testMayonVolcanoWithinPixel() + throws RuggedException, OrekitException { + + setUpMayonVolcanoContext(); + + final double latitude = FastMath.toRadians(13.2696); + final double longitude = FastMath.toRadians(123.6803); + MinMaxTreeTile tile = new MinMaxTreeTileFactory().createTile(); + updater.updateTile(latitude, longitude, tile); + double altitude = tile.interpolateElevation(latitude, longitude); + final GeodeticPoint groundGP = new GeodeticPoint(latitude, longitude, altitude); + Vector3D groundP = earth.transform(groundGP); + + final IntersectionAlgorithm algorithm = createAlgorithm(); + algorithm.setUpTilesManagement(updater, 8); + + // test direct localization + Vector3D position = state.getPVCoordinates(earth.getBodyFrame()).getPosition(); + Vector3D los = groundP.subtract(position); + GeodeticPoint result = algorithm.intersection(earth, position, los); + Assert.assertEquals(0.0, groundP.distance(earth.transform(result)), 2.0e-9); + + } + + @Test + public void testCliffsOfMoher() + throws RuggedException, OrekitException { + + setUpCliffsOfMoherContext(); + + // test point on top the cliffs, roughly 15m East of edge (inland) + final double latitude = FastMath.toRadians(52.98045); + final double longitude = FastMath.toRadians(-9.421826199814143); + MinMaxTreeTile tile = new MinMaxTreeTileFactory().createTile(); + updater.updateTile(latitude, longitude, tile); + double altitude = tile.interpolateElevation(latitude, longitude); + final GeodeticPoint groundGP = new GeodeticPoint(latitude, longitude, altitude); + Vector3D groundP = earth.transform(groundGP); + + final IntersectionAlgorithm algorithm = createAlgorithm(); + algorithm.setUpTilesManagement(updater, 8); + + // preliminary check: the point has been chosen in the spacecraft (YZ) plane + Transform earthToSpacecraft = new Transform(state.getDate(), + earth.getBodyFrame().getTransformTo(state.getFrame(), state.getDate()), + state.toTransform()); + Vector3D pointInSpacecraftFrame = earthToSpacecraft.transformPosition(groundP); + Assert.assertEquals( 0.000, pointInSpacecraftFrame.getX(), 1.0e-3); + Assert.assertEquals( 66702.419, pointInSpacecraftFrame.getY(), 1.0e-3); + Assert.assertEquals(796873.178, pointInSpacecraftFrame.getZ(), 1.0e-3); + + Vector3D position = state.getPVCoordinates(earth.getBodyFrame()).getPosition(); + Vector3D los = groundP.subtract(position); + GeodeticPoint result = algorithm.intersection(earth, position, los); + Assert.assertEquals(0.0, groundP.distance(earth.transform(result)), 2.0e-9); + + } + + private void setUpMayonVolcanoContext() + throws RuggedException, OrekitException { + + // Mayon Volcano location according to Wikipedia: 13°15′24″N 123°41′6″E + GeodeticPoint summit = + new GeodeticPoint(FastMath.toRadians(13.25667), FastMath.toRadians(123.685), 2463.0); + updater = new VolcanicConeElevationUpdater(summit, + FastMath.toRadians(30.0), 16.0, + FastMath.toRadians(1.0), 1201); + + // some orbital parameters have been computed using Orekit + // tutorial about phasing, using the following configuration: + // + // orbit.date = 2012-01-01T00:00:00.000 + // phasing.orbits.number = 143 + // phasing.days.number = 10 + // sun.synchronous.reference.latitude = 0 + // sun.synchronous.reference.ascending = false + // sun.synchronous.mean.solar.time = 10:30:00 + // gravity.field.degree = 12 + // gravity.field.order = 12 + // + // the resulting phased orbit has then been propagated to a date corresponding + // to test point lying in the spacecraft (YZ) plane (with nadir pointing and yaw compensation) + AbsoluteDate crossing = new AbsoluteDate("2012-01-06T02:27:15.942757185", TimeScalesFactory.getUTC()); + state = new SpacecraftState(new CartesianOrbit(new PVCoordinates(new Vector3D( -649500.423763743, + -6943715.537565755, + 1657929.137063380), + new Vector3D(-1305.453711368668, + -1600.627551928136, + -7167.286855869801)), + FramesFactory.getEME2000(), + crossing, + Constants.EIGEN5C_EARTH_MU), + new Attitude(crossing, + FramesFactory.getEME2000(), + new Rotation(-0.40904880353552850, + 0.46125295378582530, + -0.63525007056319790, + -0.46516893361386025, + true), + new Vector3D(-7.048568391860185e-05, + -1.043582650222194e-03, + 1.700400341147713e-05))); + + } + + private void setUpCliffsOfMoherContext() + throws RuggedException, OrekitException { + + // cliffs of Moher location according to Wikipedia: 52°56′10″N 9°28′15″ W + GeodeticPoint north = new GeodeticPoint(FastMath.toRadians(52.9984), + FastMath.toRadians(-9.4072), + 120.0); + GeodeticPoint south = new GeodeticPoint(FastMath.toRadians(52.9625), + FastMath.toRadians(-9.4369), + 120.0); + + // pixels are about 10m x 10m here and a tile covers 1km x 1km + updater = new CliffsElevationUpdater(north, south, + 120.0, 0.0, + FastMath.toRadians(0.015), 101); + + // some orbital parameters have been computed using Orekit + // tutorial about phasing, using the following configuration: + // + // orbit.date = 2012-01-01T00:00:00.000 + // phasing.orbits.number = 143 + // phasing.days.number = 10 + // sun.synchronous.reference.latitude = 0 + // sun.synchronous.reference.ascending = false + // sun.synchronous.mean.solar.time = 10:30:00 + // gravity.field.degree = 12 + // gravity.field.order = 12 + // + // the resulting phased orbit has then been propagated to a date corresponding + // to test point lying in the spacecraft (YZ) plane (with nadir pointing and yaw compensation) + AbsoluteDate crossing = new AbsoluteDate("2012-01-07T11:50:04.935272115", TimeScalesFactory.getUTC()); + state = new SpacecraftState(new CartesianOrbit(new PVCoordinates(new Vector3D( 412324.544397459, + -4325872.329311633, + 5692124.593989491), + new Vector3D(-1293.174701214779, + -5900.764863603793, + -4378.671036383179)), + FramesFactory.getEME2000(), + crossing, + Constants.EIGEN5C_EARTH_MU), + new Attitude(crossing, + FramesFactory.getEME2000(), + new Rotation(-0.17806699079182878, + 0.60143347387211290, + -0.73251248177468900, + -0.26456641385623986, + true), + new Vector3D(-4.289600857433520e-05, + -1.039151496480297e-03, + 5.811423736843181e-05))); + + } + + @Before + public void setUp() + throws OrekitException, URISyntaxException { + String path = getClass().getClassLoader().getResource("orekit-data").toURI().getPath(); + DataProvidersManager.getInstance().addProvider(new DirectoryCrawler(new File(path))); + earth = new ExtendedEllipsoid(Constants.WGS84_EARTH_EQUATORIAL_RADIUS, + Constants.WGS84_EARTH_FLATTENING, + FramesFactory.getITRF(IERSConventions.IERS_2010, true)); + + } + + @After + public void tearDown() { + earth = null; + updater = null; + state = null; + } + + private ExtendedEllipsoid earth; + private TileUpdater updater; + private SpacecraftState state; + +} diff --git a/rugged-core/src/test/java/org/orekit/rugged/core/BasicScanAlgorithmTest.java b/rugged-core/src/test/java/org/orekit/rugged/core/BasicScanAlgorithmTest.java index dbf7e3f36d2c1e0f1f32ede17540584d3dee0738..4181b8408074bb94ad92bceea3aac89c89e71ab2 100644 --- a/rugged-core/src/test/java/org/orekit/rugged/core/BasicScanAlgorithmTest.java +++ b/rugged-core/src/test/java/org/orekit/rugged/core/BasicScanAlgorithmTest.java @@ -17,205 +17,12 @@ package org.orekit.rugged.core; -import java.io.File; -import java.net.URISyntaxException; +import org.orekit.rugged.core.raster.IntersectionAlgorithm; -import org.apache.commons.math3.geometry.euclidean.threed.Rotation; -import org.apache.commons.math3.geometry.euclidean.threed.Vector3D; -import org.apache.commons.math3.util.FastMath; -import org.junit.Assert; -import org.junit.Test; -import org.orekit.attitudes.Attitude; -import org.orekit.bodies.GeodeticPoint; -import org.orekit.data.DataProvidersManager; -import org.orekit.data.DirectoryCrawler; -import org.orekit.errors.OrekitException; -import org.orekit.frames.FramesFactory; -import org.orekit.frames.Transform; -import org.orekit.orbits.CartesianOrbit; -import org.orekit.propagation.SpacecraftState; -import org.orekit.rugged.api.RuggedException; -import org.orekit.rugged.api.TileUpdater; -import org.orekit.rugged.core.ExtendedEllipsoid; -import org.orekit.rugged.core.raster.CliffsElevationUpdater; -import org.orekit.rugged.core.raster.SimpleTile; -import org.orekit.rugged.core.raster.SimpleTileFactory; -import org.orekit.rugged.core.raster.VolcanicConeElevationUpdater; -import org.orekit.time.AbsoluteDate; -import org.orekit.time.TimeScalesFactory; -import org.orekit.utils.Constants; -import org.orekit.utils.IERSConventions; -import org.orekit.utils.PVCoordinates; - -public class BasicScanAlgorithmTest { - - @Test - public void testMayonVolcano() - throws RuggedException, OrekitException, URISyntaxException { - - String path = getClass().getClassLoader().getResource("orekit-data").toURI().getPath(); - DataProvidersManager.getInstance().addProvider(new DirectoryCrawler(new File(path))); - ExtendedEllipsoid earth = new ExtendedEllipsoid(Constants.WGS84_EARTH_EQUATORIAL_RADIUS, - Constants.WGS84_EARTH_FLATTENING, - FramesFactory.getITRF(IERSConventions.IERS_2010, true)); - - // Mayon Volcano location according to Wikipedia: 13°15′24″N 123°41′6″E - GeodeticPoint summit = - new GeodeticPoint(FastMath.toRadians(13.25667), FastMath.toRadians(123.685), 2463.0); - TileUpdater updater = new VolcanicConeElevationUpdater(summit, - FastMath.toRadians(30.0), 16.0, - FastMath.toRadians(1.0), 1201); - - // test point approximately 1.6km North-North-West and 800 meters below volcano summit - double latitude = FastMath.toRadians(13.27); - double longitude = FastMath.toRadians(123.68); - SimpleTile tile = new SimpleTileFactory().createTile(); - updater.updateTile(latitude, longitude, tile); - double altitude = tile.interpolateElevation(latitude, longitude); - GeodeticPoint groundGP = new GeodeticPoint(latitude, longitude, altitude); - Vector3D groundP = earth.transform(groundGP); - - BasicScanAlgorithm basicScan = new BasicScanAlgorithm(); - basicScan.setUpTilesManagement(updater, 8); - - // some orbital parameters have been computed using Orekit - // tutorial about phasing, using the following configuration: - // - // orbit.date = 2012-01-01T00:00:00.000 - // phasing.orbits.number = 143 - // phasing.days.number = 10 - // sun.synchronous.reference.latitude = 0 - // sun.synchronous.reference.ascending = false - // sun.synchronous.mean.solar.time = 10:30:00 - // gravity.field.degree = 12 - // gravity.field.order = 12 - // - // the resulting phased orbit has then been propagated to a date corresponding - // to test point lying in the spacecraft (YZ) plane (with nadir pointing and yaw compensation) - AbsoluteDate crossing = new AbsoluteDate("2012-01-06T02:27:15.942757185", TimeScalesFactory.getUTC()); - SpacecraftState state = - new SpacecraftState(new CartesianOrbit(new PVCoordinates(new Vector3D( -649500.423763743, - -6943715.537565755, - 1657929.137063380), - new Vector3D(-1305.453711368668, - -1600.627551928136, - -7167.286855869801)), - FramesFactory.getEME2000(), - crossing, - Constants.EIGEN5C_EARTH_MU), - new Attitude(crossing, - FramesFactory.getEME2000(), - new Rotation(-0.40904880353552850, - 0.46125295378582530, - -0.63525007056319790, - -0.46516893361386025, - true), - new Vector3D(-7.048568391860185e-05, - -1.043582650222194e-03, - 1.700400341147713e-05))); - - // preliminary check: the point has been chosen in the spacecraft (YZ) plane - Transform earthToSpacecraft = new Transform(state.getDate(), - earth.getBodyFrame().getTransformTo(state.getFrame(), state.getDate()), - state.toTransform()); - Vector3D pointInSpacecraftFrame = earthToSpacecraft.transformPosition(groundP); - Assert.assertEquals( 0.000, pointInSpacecraftFrame.getX(), 1.0e-3); - Assert.assertEquals(-87754.914, pointInSpacecraftFrame.getY(), 1.0e-3); - Assert.assertEquals(790330.254, pointInSpacecraftFrame.getZ(), 1.0e-3); - - Vector3D position = state.getPVCoordinates(earth.getBodyFrame()).getPosition(); - Vector3D los = groundP.subtract(position); - GeodeticPoint result = basicScan.intersection(earth, position, los); - Assert.assertEquals(0.0, groundP.distance(earth.transform(result)), 0.03); - - } - - @Test - public void testCliffsOfMoher() - throws RuggedException, OrekitException, URISyntaxException { - - String path = getClass().getClassLoader().getResource("orekit-data").toURI().getPath(); - DataProvidersManager.getInstance().addProvider(new DirectoryCrawler(new File(path))); - ExtendedEllipsoid earth = new ExtendedEllipsoid(Constants.WGS84_EARTH_EQUATORIAL_RADIUS, - Constants.WGS84_EARTH_FLATTENING, - FramesFactory.getITRF(IERSConventions.IERS_2010, true)); - - // cliffs of Moher location according to Wikipedia: 52°56′10″N 9°28′15″ W - GeodeticPoint north = new GeodeticPoint(FastMath.toRadians(52.9984), - FastMath.toRadians(-9.4072), - 120.0); - GeodeticPoint south = new GeodeticPoint(FastMath.toRadians(52.9625), - FastMath.toRadians(-9.4369), - 120.0); - - // pixels are about 10m x 10m here and a tile covers 1km x 1km - TileUpdater updater = new CliffsElevationUpdater(north, south, - 120.0, 0.0, - FastMath.toRadians(0.015), 101); - - // test point on top the cliffs, roughly 15m East of edge (inland) - double latitude = 0.5 * (north.getLatitude() + south.getLatitude()); - double longitude = 0.5 * (north.getLongitude() + south.getLongitude()) + - 15.0 / (Constants.WGS84_EARTH_EQUATORIAL_RADIUS * FastMath.cos(latitude)); - SimpleTile tile = new SimpleTileFactory().createTile(); - updater.updateTile(latitude, longitude, tile); - double altitude = tile.interpolateElevation(latitude, longitude); - GeodeticPoint groundGP = new GeodeticPoint(latitude, longitude, altitude); - Vector3D groundP = earth.transform(groundGP); - - BasicScanAlgorithm basicScan = new BasicScanAlgorithm(); - basicScan.setUpTilesManagement(updater, 8); - - // some orbital parameters have been computed using Orekit - // tutorial about phasing, using the following configuration: - // - // orbit.date = 2012-01-01T00:00:00.000 - // phasing.orbits.number = 143 - // phasing.days.number = 10 - // sun.synchronous.reference.latitude = 0 - // sun.synchronous.reference.ascending = false - // sun.synchronous.mean.solar.time = 10:30:00 - // gravity.field.degree = 12 - // gravity.field.order = 12 - // - // the resulting phased orbit has then been propagated to a date corresponding - // to test point lying in the spacecraft (YZ) plane (with nadir pointing and yaw compensation) - AbsoluteDate crossing = new AbsoluteDate("2012-01-07T11:50:04.935272115", TimeScalesFactory.getUTC()); - SpacecraftState state = - new SpacecraftState(new CartesianOrbit(new PVCoordinates(new Vector3D( 412324.544397459, - -4325872.329311633, - 5692124.593989491), - new Vector3D(-1293.174701214779, - -5900.764863603793, - -4378.671036383179)), - FramesFactory.getEME2000(), - crossing, - Constants.EIGEN5C_EARTH_MU), - new Attitude(crossing, - FramesFactory.getEME2000(), - new Rotation(-0.17806699079182878, - 0.60143347387211290, - -0.73251248177468900, - -0.26456641385623986, - true), - new Vector3D(-4.289600857433520e-05, - -1.039151496480297e-03, - 5.811423736843181e-05))); - - // preliminary check: the point has been chosen in the spacecraft (YZ) plane - Transform earthToSpacecraft = new Transform(state.getDate(), - earth.getBodyFrame().getTransformTo(state.getFrame(), state.getDate()), - state.toTransform()); - Vector3D pointInSpacecraftFrame = earthToSpacecraft.transformPosition(groundP); - Assert.assertEquals( 0.000, pointInSpacecraftFrame.getX(), 1.0e-3); - Assert.assertEquals( 66702.419, pointInSpacecraftFrame.getY(), 1.0e-3); - Assert.assertEquals(796873.178, pointInSpacecraftFrame.getZ(), 1.0e-3); - - Vector3D position = state.getPVCoordinates(earth.getBodyFrame()).getPosition(); - Vector3D los = groundP.subtract(position); - GeodeticPoint result = basicScan.intersection(earth, position, los); - Assert.assertEquals(0.0, groundP.distance(earth.transform(result)), 1.0e-10); +public class BasicScanAlgorithmTest extends AbstractAlgorithmTest { + public IntersectionAlgorithm createAlgorithm() { + return new BasicScanAlgorithm(); } } diff --git a/rugged-core/src/test/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithmTest.java b/rugged-core/src/test/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithmTest.java index eb9b75cd1d0bcf2181d91e49fcf8a87e650eefa5..535bc08a67995dbc9ab82e059c67865c0d485865 100644 --- a/rugged-core/src/test/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithmTest.java +++ b/rugged-core/src/test/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithmTest.java @@ -17,209 +17,13 @@ package org.orekit.rugged.core.duvenhage; -import java.io.File; -import java.net.URISyntaxException; +import org.orekit.rugged.core.AbstractAlgorithmTest; +import org.orekit.rugged.core.raster.IntersectionAlgorithm; -import org.apache.commons.math3.geometry.euclidean.threed.Rotation; -import org.apache.commons.math3.geometry.euclidean.threed.Vector3D; -import org.apache.commons.math3.util.FastMath; -import org.junit.Assert; -import org.junit.Test; -import org.orekit.attitudes.Attitude; -import org.orekit.bodies.GeodeticPoint; -import org.orekit.data.DataProvidersManager; -import org.orekit.data.DirectoryCrawler; -import org.orekit.errors.OrekitException; -import org.orekit.frames.FramesFactory; -import org.orekit.frames.Transform; -import org.orekit.orbits.CartesianOrbit; -import org.orekit.propagation.SpacecraftState; -import org.orekit.rugged.api.RuggedException; -import org.orekit.rugged.api.TileUpdater; -import org.orekit.rugged.core.ExtendedEllipsoid; -import org.orekit.rugged.core.raster.CliffsElevationUpdater; -import org.orekit.rugged.core.raster.VolcanicConeElevationUpdater; -import org.orekit.time.AbsoluteDate; -import org.orekit.time.TimeScalesFactory; -import org.orekit.utils.Constants; -import org.orekit.utils.IERSConventions; -import org.orekit.utils.PVCoordinates; - -public class DuvenhageAlgorithmTest { - - @Test - public void testMayonVolcano() - throws RuggedException, OrekitException, URISyntaxException { - - String path = getClass().getClassLoader().getResource("orekit-data").toURI().getPath(); - DataProvidersManager.getInstance().addProvider(new DirectoryCrawler(new File(path))); - ExtendedEllipsoid earth = new ExtendedEllipsoid(Constants.WGS84_EARTH_EQUATORIAL_RADIUS, - Constants.WGS84_EARTH_FLATTENING, - FramesFactory.getITRF(IERSConventions.IERS_2010, true)); - - // Mayon Volcano location according to Wikipedia: 13°15′24″N 123°41′6″E - GeodeticPoint summit = - new GeodeticPoint(FastMath.toRadians(13.25667), FastMath.toRadians(123.685), 2463.0); - TileUpdater updater = new VolcanicConeElevationUpdater(summit, - FastMath.toRadians(30.0), 16.0, - FastMath.toRadians(1.0), 1201); - - // test point approximately 1.6km North-North-West and 800 meters below volcano summit - // note that this test point is EXACTLY at a pixel corner, and even at corners of - // middle level (12 and above) sub-tiles - double latitude = FastMath.toRadians(13.27); - double longitude = FastMath.toRadians(123.68); - MinMaxTreeTile tile = new MinMaxTreeTileFactory().createTile(); - updater.updateTile(latitude, longitude, tile); - double altitude = tile.interpolateElevation(latitude, longitude); - GeodeticPoint groundGP = new GeodeticPoint(latitude, longitude, altitude); - Vector3D groundP = earth.transform(groundGP); - - DuvenhageAlgorithm duvenhage = new DuvenhageAlgorithm(); - duvenhage.setUpTilesManagement(updater, 8); - - // some orbital parameters have been computed using Orekit - // tutorial about phasing, using the following configuration: - // - // orbit.date = 2012-01-01T00:00:00.000 - // phasing.orbits.number = 143 - // phasing.days.number = 10 - // sun.synchronous.reference.latitude = 0 - // sun.synchronous.reference.ascending = false - // sun.synchronous.mean.solar.time = 10:30:00 - // gravity.field.degree = 12 - // gravity.field.order = 12 - // - // the resulting phased orbit has then been propagated to a date corresponding - // to test point lying in the spacecraft (YZ) plane (with nadir pointing and yaw compensation) - AbsoluteDate crossing = new AbsoluteDate("2012-01-06T02:27:15.942757185", TimeScalesFactory.getUTC()); - SpacecraftState state = - new SpacecraftState(new CartesianOrbit(new PVCoordinates(new Vector3D( -649500.423763743, - -6943715.537565755, - 1657929.137063380), - new Vector3D(-1305.453711368668, - -1600.627551928136, - -7167.286855869801)), - FramesFactory.getEME2000(), - crossing, - Constants.EIGEN5C_EARTH_MU), - new Attitude(crossing, - FramesFactory.getEME2000(), - new Rotation(-0.40904880353552850, - 0.46125295378582530, - -0.63525007056319790, - -0.46516893361386025, - true), - new Vector3D(-7.048568391860185e-05, - -1.043582650222194e-03, - 1.700400341147713e-05))); - - // preliminary check: the point has been chosen in the spacecraft (YZ) plane - Transform earthToSpacecraft = new Transform(state.getDate(), - earth.getBodyFrame().getTransformTo(state.getFrame(), state.getDate()), - state.toTransform()); - Vector3D pointInSpacecraftFrame = earthToSpacecraft.transformPosition(groundP); - Assert.assertEquals( 0.000, pointInSpacecraftFrame.getX(), 1.0e-3); - Assert.assertEquals(-87754.914, pointInSpacecraftFrame.getY(), 1.0e-3); - Assert.assertEquals(790330.254, pointInSpacecraftFrame.getZ(), 1.0e-3); - - Vector3D position = state.getPVCoordinates(earth.getBodyFrame()).getPosition(); - Vector3D los = groundP.subtract(position); - GeodeticPoint result = duvenhage.intersection(earth, position, los); - Assert.assertEquals(groundGP.getLatitude(), result.getLatitude(), 1.0e-10); - Assert.assertEquals(groundGP.getLongitude(), result.getLongitude(), 1.0e-10); - Assert.assertEquals(groundGP.getAltitude(), result.getAltitude(), 1.0e-9); - - } - - @Test - public void testCliffsOfMoher() - throws RuggedException, OrekitException, URISyntaxException { - - String path = getClass().getClassLoader().getResource("orekit-data").toURI().getPath(); - DataProvidersManager.getInstance().addProvider(new DirectoryCrawler(new File(path))); - ExtendedEllipsoid earth = new ExtendedEllipsoid(Constants.WGS84_EARTH_EQUATORIAL_RADIUS, - Constants.WGS84_EARTH_FLATTENING, - FramesFactory.getITRF(IERSConventions.IERS_2010, true)); - - // cliffs of Moher location according to Wikipedia: 52°56′10″N 9°28′15″ W - GeodeticPoint north = new GeodeticPoint(FastMath.toRadians(52.9984), - FastMath.toRadians(-9.4072), - 120.0); - GeodeticPoint south = new GeodeticPoint(FastMath.toRadians(52.9625), - FastMath.toRadians(-9.4369), - 120.0); - - // pixels are about 10m x 10m here and a tile covers 1km x 1km - TileUpdater updater = new CliffsElevationUpdater(north, south, - 120.0, 0.0, - FastMath.toRadians(0.015), 101); - - // test point on top the cliffs, roughly 15m East of edge (inland) - double latitude = 0.5 * (north.getLatitude() + south.getLatitude()); - double longitude = 0.5 * (north.getLongitude() + south.getLongitude()) + - 15.0 / (Constants.WGS84_EARTH_EQUATORIAL_RADIUS * FastMath.cos(latitude)); - MinMaxTreeTile tile = new MinMaxTreeTileFactory().createTile(); - updater.updateTile(latitude, longitude, tile); - double altitude = tile.interpolateElevation(latitude, longitude); - GeodeticPoint groundGP = new GeodeticPoint(latitude, longitude, altitude); - Vector3D groundP = earth.transform(groundGP); - - DuvenhageAlgorithm duvenhage = new DuvenhageAlgorithm(); - duvenhage.setUpTilesManagement(updater, 8); - - // some orbital parameters have been computed using Orekit - // tutorial about phasing, using the following configuration: - // - // orbit.date = 2012-01-01T00:00:00.000 - // phasing.orbits.number = 143 - // phasing.days.number = 10 - // sun.synchronous.reference.latitude = 0 - // sun.synchronous.reference.ascending = false - // sun.synchronous.mean.solar.time = 10:30:00 - // gravity.field.degree = 12 - // gravity.field.order = 12 - // - // the resulting phased orbit has then been propagated to a date corresponding - // to test point lying in the spacecraft (YZ) plane (with nadir pointing and yaw compensation) - AbsoluteDate crossing = new AbsoluteDate("2012-01-07T11:50:04.935272115", TimeScalesFactory.getUTC()); - SpacecraftState state = - new SpacecraftState(new CartesianOrbit(new PVCoordinates(new Vector3D( 412324.544397459, - -4325872.329311633, - 5692124.593989491), - new Vector3D(-1293.174701214779, - -5900.764863603793, - -4378.671036383179)), - FramesFactory.getEME2000(), - crossing, - Constants.EIGEN5C_EARTH_MU), - new Attitude(crossing, - FramesFactory.getEME2000(), - new Rotation(-0.17806699079182878, - 0.60143347387211290, - -0.73251248177468900, - -0.26456641385623986, - true), - new Vector3D(-4.289600857433520e-05, - -1.039151496480297e-03, - 5.811423736843181e-05))); - - // preliminary check: the point has been chosen in the spacecraft (YZ) plane - Transform earthToSpacecraft = new Transform(state.getDate(), - earth.getBodyFrame().getTransformTo(state.getFrame(), state.getDate()), - state.toTransform()); - Vector3D pointInSpacecraftFrame = earthToSpacecraft.transformPosition(groundP); - Assert.assertEquals( 0.000, pointInSpacecraftFrame.getX(), 1.0e-3); - Assert.assertEquals( 66702.419, pointInSpacecraftFrame.getY(), 1.0e-3); - Assert.assertEquals(796873.178, pointInSpacecraftFrame.getZ(), 1.0e-3); - - Vector3D position = state.getPVCoordinates(earth.getBodyFrame()).getPosition(); - Vector3D los = groundP.subtract(position); - GeodeticPoint result = duvenhage.intersection(earth, position, los); - Assert.assertEquals(groundGP.getLatitude(), result.getLatitude(), 1.0e-10); - Assert.assertEquals(groundGP.getLongitude(), result.getLongitude(), 1.0e-10); - Assert.assertEquals(groundGP.getAltitude(), result.getAltitude(), 1.0e-9); +public class DuvenhageAlgorithmTest extends AbstractAlgorithmTest { + protected IntersectionAlgorithm createAlgorithm() { + return new DuvenhageAlgorithm(); } }