Skip to content

Commit

Permalink
Merge pull request #3707 from rouault/fix_crs_to_crs_warn_only_best
Browse files Browse the repository at this point in the history
proj_create_crs_to_crs(): restore behaviour of PROJ 9.1
  • Loading branch information
rouault authored Apr 19, 2023
2 parents 977ce13 + 78d563c commit 700721e
Show file tree
Hide file tree
Showing 6 changed files with 154 additions and 42 deletions.
115 changes: 79 additions & 36 deletions src/4D_api.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1878,25 +1878,29 @@ pj_create_prepared_operations(PJ_CONTEXT *ctx, const PJ *source_crs,
double north_lat = 0.0;

const char *areaName = nullptr;
if (proj_get_area_of_use(ctx, op, &west_lon, &south_lat, &east_lon,
&north_lat, &areaName)) {
const bool isOffshore =
areaName && strstr(areaName, "- offshore");
if (west_lon <= east_lon) {
op = add_coord_op_to_list(
i, op, west_lon, south_lat, east_lon, north_lat,
pjGeogToSrc, pjGeogToDst, isOffshore, preparedOpList);
} else {
auto op_clone = proj_clone(ctx, op);

op = add_coord_op_to_list(
i, op, west_lon, south_lat, 180, north_lat, pjGeogToSrc,
pjGeogToDst, isOffshore, preparedOpList);
op_clone = add_coord_op_to_list(
i, op_clone, -180, south_lat, east_lon, north_lat,
pjGeogToSrc, pjGeogToDst, isOffshore, preparedOpList);
proj_destroy(op_clone);
}
if (!proj_get_area_of_use(ctx, op, &west_lon, &south_lat, &east_lon,
&north_lat, &areaName)) {
west_lon = -180;
south_lat = -90;
east_lon = 180;
north_lat = 90;
}

const bool isOffshore = areaName && strstr(areaName, "- offshore");
if (west_lon <= east_lon) {
op = add_coord_op_to_list(i, op, west_lon, south_lat, east_lon,
north_lat, pjGeogToSrc, pjGeogToDst,
isOffshore, preparedOpList);
} else {
auto op_clone = proj_clone(ctx, op);

op = add_coord_op_to_list(i, op, west_lon, south_lat, 180,
north_lat, pjGeogToSrc, pjGeogToDst,
isOffshore, preparedOpList);
op_clone = add_coord_op_to_list(
i, op_clone, -180, south_lat, east_lon, north_lat,
pjGeogToSrc, pjGeogToDst, isOffshore, preparedOpList);
proj_destroy(op_clone);
}

proj_destroy(op);
Expand Down Expand Up @@ -2056,27 +2060,48 @@ PJ *proj_create_crs_to_crs_from_pj(PJ_CONTEXT *ctx, const PJ *source_crs,
P->skipNonInstantiable = warnIfBestTransformationNotAvailable;
}

if (P == nullptr || op_count == 1 ||
proj_get_type(source_crs) == PJ_TYPE_GEOCENTRIC_CRS ||
proj_get_type(target_crs) == PJ_TYPE_GEOCENTRIC_CRS) {
const bool mayNeedToReRunWithDiscardMissing =
(errorIfBestTransformationNotAvailable ||
warnIfBestTransformationNotAvailable) &&
!proj_context_is_network_enabled(ctx);
int singleOpIsInstanciable = -1;
if (P != nullptr && op_count == 1 && mayNeedToReRunWithDiscardMissing) {
singleOpIsInstanciable = proj_coordoperation_is_instantiable(ctx, P);
}

const auto backup_errno = proj_context_errno(ctx);
if ((P == nullptr ||
(op_count == 1 &&
(!mayNeedToReRunWithDiscardMissing ||
errorIfBestTransformationNotAvailable ||
singleOpIsInstanciable == static_cast<int>(true))) ||
proj_get_type(source_crs) == PJ_TYPE_GEOCENTRIC_CRS ||
proj_get_type(target_crs) == PJ_TYPE_GEOCENTRIC_CRS)) {
proj_list_destroy(op_list);
ctx->forceOver = false;

if (P != nullptr &&
(errorIfBestTransformationNotAvailable ||
warnIfBestTransformationNotAvailable) &&
!proj_coordoperation_is_instantiable(ctx, P)) {
warnAboutMissingGrid(P);
if (errorIfBestTransformationNotAvailable) {
proj_destroy(P);
return nullptr;
if (P != nullptr && (errorIfBestTransformationNotAvailable ||
warnIfBestTransformationNotAvailable)) {
if (singleOpIsInstanciable < 0) {
singleOpIsInstanciable =
proj_coordoperation_is_instantiable(ctx, P);
}
if (!singleOpIsInstanciable) {
warnAboutMissingGrid(P);
if (errorIfBestTransformationNotAvailable) {
proj_destroy(P);
return nullptr;
}
}
}

if (P != nullptr) {
P->over = forceOver;
}
return P;
} else if (op_count == 1 && mayNeedToReRunWithDiscardMissing &&
!singleOpIsInstanciable) {
warnAboutMissingGrid(P);
}

if (errorIfBestTransformationNotAvailable ||
Expand All @@ -2094,10 +2119,6 @@ PJ *proj_create_crs_to_crs_from_pj(PJ_CONTEXT *ctx, const PJ *source_crs,
return nullptr;
}

const bool mayNeedToReRunWithDiscardMissing =
(errorIfBestTransformationNotAvailable ||
warnIfBestTransformationNotAvailable) &&
!proj_context_is_network_enabled(ctx);
bool foundInstanciableAndNonBallpark = false;

for (auto &op : preparedOpList) {
Expand Down Expand Up @@ -2152,6 +2173,8 @@ PJ *proj_create_crs_to_crs_from_pj(PJ_CONTEXT *ctx, const PJ *source_crs,

op_list = proj_create_operations(ctx, source_crs, target_crs,
operation_ctx);
proj_operation_factory_context_destroy(operation_ctx);

if (op_list) {
ctx->forceOver = forceOver;
ctx->debug_level = PJ_LOG_NONE;
Expand Down Expand Up @@ -2179,10 +2202,30 @@ PJ *proj_create_crs_to_crs_from_pj(PJ_CONTEXT *ctx, const PJ *source_crs,
newOpList.emplace_back(std::move(op));
}
preparedOpList = std::move(newOpList);
} else {
// We get there in "cs2cs --only-best --no-ballpark
// EPSG:4326+3855 EPSG:4979" use case, where the initial
// create_operations returned 1 operation, and the retry
// with
// PROJ_GRID_AVAILABILITY_DISCARD_OPERATION_IF_MISSING_GRID
// returned 0.
if (op_count == 1 &&
(errorIfBestTransformationNotAvailable ||
warnIfBestTransformationNotAvailable)) {
if (singleOpIsInstanciable < 0) {
singleOpIsInstanciable =
proj_coordoperation_is_instantiable(ctx, P);
}
if (!singleOpIsInstanciable) {
if (errorIfBestTransformationNotAvailable) {
proj_destroy(P);
proj_context_errno_set(ctx, backup_errno);
return nullptr;
}
}
}
}
}

proj_operation_factory_context_destroy(operation_ctx);
}
}

Expand Down
11 changes: 11 additions & 0 deletions src/iso19111/factory.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3250,6 +3250,17 @@ bool DatabaseContext::lookForGridInfo(
bool &directDownload, bool &openLicense, bool &gridAvailable) const {
Private::GridInfoCache info;

if (projFilename == "null") {
// Special case for implicit "null" grid.
fullFilename.clear();
packageName.clear();
url.clear();
directDownload = false;
openLicense = true;
gridAvailable = true;
return true;
}

auto ctxt = d->pjCtxt();
if (ctxt == nullptr) {
ctxt = pj_get_default_ctx();
Expand Down
21 changes: 17 additions & 4 deletions src/iso19111/operation/coordinateoperationfactory.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -4860,14 +4860,27 @@ void CoordinateOperationFactory::Private::createOperationsVertToVert(
if (convDst == 0)
throw InvalidOperation("Conversion factor of target unit is 0");
const double factor = convSrc / convDst;

const auto &sourceCRSExtent = getExtent(sourceCRS);
const auto &targetCRSExtent = getExtent(targetCRS);
const bool sameExtent =
sourceCRSExtent && targetCRSExtent &&
sourceCRSExtent->_isEquivalentTo(
targetCRSExtent.get(), util::IComparable::Criterion::EQUIVALENT);

util::PropertyMap map;
map.set(common::ObjectUsage::DOMAIN_OF_VALIDITY_KEY,
sameExtent ? NN_NO_CHECK(sourceCRSExtent)
: metadata::Extent::WORLD);

if (!equivalentVDatum) {
auto name = buildTransfName(sourceCRS->nameStr(), targetCRS->nameStr());
name += " (";
name += BALLPARK_VERTICAL_TRANSFORMATION;
name += ')';
auto conv = Transformation::createChangeVerticalUnit(
util::PropertyMap().set(common::IdentifiedObject::NAME_KEY, name),
sourceCRS, targetCRS,
map.set(common::IdentifiedObject::NAME_KEY, name), sourceCRS,
targetCRS,
// In case of a height depth reversal, we should probably have
// 2 steps instead of putting a negative factor...
common::Scale(heightDepthReversal ? -factor : factor), {});
Expand All @@ -4876,7 +4889,7 @@ void CoordinateOperationFactory::Private::createOperationsVertToVert(
} else if (convSrc != convDst || !heightDepthReversal) {
auto name = buildConvName(sourceCRS->nameStr(), targetCRS->nameStr());
auto conv = Conversion::createChangeVerticalUnit(
util::PropertyMap().set(common::IdentifiedObject::NAME_KEY, name),
map.set(common::IdentifiedObject::NAME_KEY, name),
// In case of a height depth reversal, we should probably have
// 2 steps instead of putting a negative factor...
common::Scale(heightDepthReversal ? -factor : factor));
Expand All @@ -4885,7 +4898,7 @@ void CoordinateOperationFactory::Private::createOperationsVertToVert(
} else {
auto name = buildConvName(sourceCRS->nameStr(), targetCRS->nameStr());
auto conv = Conversion::createHeightDepthReversal(
util::PropertyMap().set(common::IdentifiedObject::NAME_KEY, name));
map.set(common::IdentifiedObject::NAME_KEY, name));
conv->setCRSs(sourceCRS, targetCRS, nullptr);
res.push_back(conv);
}
Expand Down
30 changes: 30 additions & 0 deletions test/cli/testvarious
Original file line number Diff line number Diff line change
Expand Up @@ -1217,6 +1217,36 @@ EOF

rm -rf tmp_dir

echo "##############################################################" >> ${OUT}
echo "Test cs2cs (grid missing) with scenario of https://lists.osgeo.org/pipermail/proj/2023-April/011003.html" >> ${OUT}

mkdir tmp_dir
cp $PROJ_DATA/proj.db tmp_dir

echo 39 -3 0 | PROJ_DATA=tmp_dir PROJ_DEBUG=2 $EXE EPSG:4326+5773 EPSG:4326+5782 -E 2>&1|grep -v pj_open_lib >> ${OUT}

rm -rf tmp_dir

echo "##############################################################" >> ${OUT}
echo "Test cs2cs (grid missing) --only-best=no with scenario of https://lists.osgeo.org/pipermail/proj/2023-April/011003.html" >> ${OUT}

mkdir tmp_dir
cp $PROJ_DATA/proj.db tmp_dir

echo 39 -3 0 | PROJ_DATA=tmp_dir PROJ_DEBUG=2 $EXE --only-best=no EPSG:4326+5773 EPSG:4326+5782 -E 2>&1|grep -v pj_open_lib >> ${OUT}

rm -rf tmp_dir

echo "##############################################################" >> ${OUT}
echo "Test cs2cs (grid missing) --only-best with scenario of https://lists.osgeo.org/pipermail/proj/2023-April/011003.html" >> ${OUT}

mkdir tmp_dir
cp $PROJ_DATA/proj.db tmp_dir

echo 39 -3 0 | PROJ_DISPLAY_PROGRAM_NAME=NO PROJ_DATA=tmp_dir PROJ_DEBUG=2 $EXE --only-best EPSG:4326+5773 EPSG:4326+5782 -E 2>&1|grep -v pj_open_lib >> ${OUT}

rm -rf tmp_dir

echo "##############################################################" >> ${OUT}
echo "Test Similarity Transformation (example from EPSG Guidance Note 7.2)" >> ${OUT}
#
Expand Down
15 changes: 15 additions & 0 deletions test/cli/tv_out.dist
Original file line number Diff line number Diff line change
Expand Up @@ -592,6 +592,21 @@ Test cs2cs (grid missing) with scenario of /~https://github.com/OSGeo/PROJ/issues/
569704.5660295591 4269024.671083651 569720.46 4268813.88 0.00
569704.5660295591 4269024.671083651 569720.46 4268813.88 0.00
##############################################################
Test cs2cs (grid missing) with scenario of https://lists.osgeo.org/pipermail/proj/2023-April/011003.html
Attempt to use coordinate operation Inverse of WGS 84 to EGM96 height (1) + ETRS89 to Alicante height (1) using ETRS89 to WGS 84 (1) failed. Grid es_ign_egm08-rednap.tif is not available. Consult https://proj.org/resource_files.html for guidance. Grid us_nga_egm96_15.tif is not available. Consult https://proj.org/resource_files.html for guidance. This might become an error in a future PROJ major release. Set the ONLY_BEST option to YES or NO. This warning will no longer be emitted (for the current transformation instance).
Using coordinate operation Transformation from EGM96 height to Alicante height (ballpark vertical transformation)
39 -3 0 39.00 -3.00 0.00
##############################################################
Test cs2cs (grid missing) --only-best=no with scenario of https://lists.osgeo.org/pipermail/proj/2023-April/011003.html
39 -3 0 39.00 -3.00 0.00
##############################################################
Test cs2cs (grid missing) --only-best with scenario of https://lists.osgeo.org/pipermail/proj/2023-April/011003.html
Attempt to use coordinate operation Inverse of WGS 84 to EGM96 height (1) + ETRS89 to Alicante height (1) using ETRS89 to WGS 84 (1) failed. Grid es_ign_egm08-rednap.tif is not available. Consult https://proj.org/resource_files.html for guidance. Grid us_nga_egm96_15.tif is not available. Consult https://proj.org/resource_files.html for guidance.

cannot initialize transformation
cause: File not found or invalid
program abnormally terminated
##############################################################
Test Similarity Transformation (example from EPSG Guidance Note 7.2)
300000 4500000 299905.060 4499796.515 0.000
##############################################################
Expand Down
4 changes: 2 additions & 2 deletions test/unit/test_operationfactory.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -5988,7 +5988,7 @@ TEST(operation, vertCRS_to_vertCRS_context) {
authFactory->createCoordinateReferenceSystem("7968"),
// NAVD88 height (1)
authFactory->createCoordinateReferenceSystem("5703"), ctxt);
ASSERT_EQ(list.size(), 3U);
ASSERT_EQ(list.size(), 4U);
EXPECT_EQ(list[0]->nameStr(), "NGVD29 height (m) to NAVD88 height (3)");
EXPECT_EQ(list[0]->exportToPROJString(PROJStringFormatter::create().get()),
"+proj=vgridshift +grids=us_noaa_vertcone.tif +multiplier=1");
Expand All @@ -6005,7 +6005,7 @@ TEST(operation, vertCRS_to_vertCRS_New_Zealand_context) {
authFactory->createCoordinateReferenceSystem("7839"),
// Auckland 1946 height
authFactory->createCoordinateReferenceSystem("5759"), ctxt);
ASSERT_EQ(list.size(), 1U);
ASSERT_EQ(list.size(), 2U);
EXPECT_EQ(list[0]->exportToPROJString(PROJStringFormatter::create().get()),
"+proj=vgridshift +grids=nz_linz_auckht1946-nzvd2016.tif "
"+multiplier=1");
Expand Down

0 comments on commit 700721e

Please sign in to comment.