Skip to content

Commit

Permalink
Merge pull request #4139 from OSGeo/backport-4124-to-9.4
Browse files Browse the repository at this point in the history
[Backport 9.4] DerivedProjectedCRS: deal with lack of explicit CS in BASEPROJCRS in WKT
  • Loading branch information
rouault authored May 9, 2024
2 parents f00f01d + ee6a519 commit 5e9c999
Show file tree
Hide file tree
Showing 7 changed files with 1,402 additions and 1,049 deletions.
31 changes: 31 additions & 0 deletions src/iso19111/crs.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -6740,6 +6740,37 @@ void DerivedProjectedCRS::_exportToWKT(io::WKTFormatter *formatter) const {
formatter->endNode();

l_baseProjCRS->derivingConversionRef()->_exportToWKT(formatter);

const auto &baseCSAxisList =
l_baseProjCRS->coordinateSystem()->axisList();
// Current WKT grammar (as of WKT2 18-010r11) does not allow a
// BASEPROJCRS.CS node, but in situations where this is ambiguous, emit
// one. Cf WKTParser::Private::buildProjectedCRS() for more details
if (!baseCSAxisList.empty() &&
baseCSAxisList[0]->unit() != common::UnitOfMeasure::METRE &&
l_baseProjCRS->identifiers().empty()) {
bool knownBaseCRS = false;
auto &dbContext = formatter->databaseContext();
if (dbContext) {
auto authFactory = io::AuthorityFactory::create(
NN_NO_CHECK(dbContext), std::string());
auto res = authFactory->createObjectsFromName(
l_baseProjCRS->nameStr(),
{io::AuthorityFactory::ObjectType::PROJECTED_CRS}, false,
2);
if (res.size() == 1) {
knownBaseCRS = true;
}
}
if (!knownBaseCRS) {
l_baseProjCRS->coordinateSystem()->_exportToWKT(formatter);
}
}

if (identifiers().empty() && !l_baseProjCRS->identifiers().empty()) {
l_baseProjCRS->formatID(formatter);
}

formatter->endNode();
}

Expand Down
74 changes: 72 additions & 2 deletions src/iso19111/io.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -4460,10 +4460,80 @@ WKTParser::Private::buildProjectedCRS(const WKTNodeNNPtr &node) {
!ci_equal(nodeValue, WKTConstants::BASEPROJCRS)) {
ThrowMissing(WKTConstants::CS_);
}
auto cs = buildCS(csNode, node, UnitOfMeasure::NONE);
auto cartesianCS = nn_dynamic_pointer_cast<CartesianCS>(cs);

const std::string projCRSName = stripQuotes(nodeP->children()[0]);

auto cs = [this, &projCRSName, &nodeP, &csNode, &node, &nodeValue,
&conversionNode]() -> CoordinateSystemNNPtr {
if (isNull(csNode) && ci_equal(nodeValue, WKTConstants::BASEPROJCRS) &&
!isNull(conversionNode)) {
// A BASEPROJCRS (as of WKT2 18-010r11) normally lacks an explicit
// CS[] which cause issues to properly instanciate it. So we first
// start by trying to identify the BASEPROJCRS by its id or name.
// And fallback to exploring the conversion parameters to infer the
// CS AXIS unit from the linear parameter unit... Not fully bullet
// proof.
if (dbContext_) {
// Get official name from database if ID is present
auto &idNode = nodeP->lookForChild(WKTConstants::ID);
if (!isNull(idNode)) {
try {
auto id = buildId(idNode, false, false);
auto authFactory = AuthorityFactory::create(
NN_NO_CHECK(dbContext_), *id->codeSpace());
auto projCRS =
authFactory->createProjectedCRS(id->code());
return projCRS->coordinateSystem();
} catch (const std::exception &) {
}
}

auto authFactory = AuthorityFactory::create(
NN_NO_CHECK(dbContext_), std::string());
auto res = authFactory->createObjectsFromName(
projCRSName, {AuthorityFactory::ObjectType::PROJECTED_CRS},
false, 2);
if (res.size() == 1) {
auto projCRS =
dynamic_cast<const ProjectedCRS *>(res.front().get());
if (projCRS) {
return projCRS->coordinateSystem();
}
}
}

auto conv = buildConversion(conversionNode, UnitOfMeasure::METRE,
UnitOfMeasure::DEGREE);
UnitOfMeasure linearUOM = UnitOfMeasure::NONE;
for (const auto &genOpParamvalue : conv->parameterValues()) {
auto opParamvalue =
dynamic_cast<const operation::OperationParameterValue *>(
genOpParamvalue.get());
if (opParamvalue) {
const auto &parameterValue = opParamvalue->parameterValue();
if (parameterValue->type() ==
operation::ParameterValue::Type::MEASURE) {
const auto &measure = parameterValue->value();
const auto &unit = measure.unit();
if (unit.type() == UnitOfMeasure::Type::LINEAR) {
if (linearUOM == UnitOfMeasure::NONE) {
linearUOM = unit;
} else if (linearUOM != unit) {
linearUOM = UnitOfMeasure::NONE;
break;
}
}
}
}
}
if (linearUOM != UnitOfMeasure::NONE) {
return CartesianCS::createEastingNorthing(linearUOM);
}
}
return buildCS(csNode, node, UnitOfMeasure::NONE);
}();
auto cartesianCS = nn_dynamic_pointer_cast<CartesianCS>(cs);

if (esriStyle_ && dbContext_) {
if (cartesianCS) {
std::string outTableName;
Expand Down
2 changes: 1 addition & 1 deletion src/lib_proj.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -344,7 +344,7 @@ add_custom_target(check_wkt2_grammar_md5 ALL
COMMAND ${CMAKE_COMMAND}
"-DIN_FILE=wkt2_grammar.y"
"-DTARGET=generate_wkt2_parser"
"-DEXPECTED_MD5SUM=289572eebe9dab3c7225bd48c445c287"
"-DEXPECTED_MD5SUM=94337ca471caf74688db939009652ddb"
-P "${CMAKE_CURRENT_SOURCE_DIR}/check_md5sum.cmake"
WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}"
DEPENDS "${CMAKE_CURRENT_SOURCE_DIR}/wkt2_grammar.y"
Expand Down
Loading

0 comments on commit 5e9c999

Please sign in to comment.