LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/ili - ili2reader.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 313 455 68.8 %
Date: 2025-01-18 12:42:00 Functions: 29 32 90.6 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  Interlis 2 Reader
       4             :  * Purpose:  Implementation of ILI2Reader class.
       5             :  * Author:   Markus Schnider, Sourcepole AG
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2004, Pirmin Kalberer, Sourcepole AG
       9             :  * Copyright (c) 2008-2012, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "ili2readerp.h"
      15             : #include "ogr_ili2.h"
      16             : #include "cpl_conv.h"
      17             : #include "cpl_string.h"
      18             : 
      19             : #include "ili2reader.h"
      20             : 
      21             : using namespace std;
      22             : 
      23             : //
      24             : // constants
      25             : //
      26             : static const char *const ILI2_TID = "TID";
      27             : static const XMLCh xmlch_ILI2_TID[] = {'T', 'I', 'D', '\0'};
      28             : static const XMLCh ILI2_REF[] = {'R', 'E', 'F', '\0'};
      29             : 
      30             : constexpr int ILI2_STRING_TYPE = 0;
      31             : constexpr int ILI2_COORD_TYPE = 1;
      32             : constexpr int ILI2_ARC_TYPE = 2;
      33             : constexpr int ILI2_POLYLINE_TYPE = 4;
      34             : constexpr int ILI2_BOUNDARY_TYPE = 8;
      35             : constexpr int ILI2_AREA_TYPE = 16;  // also SURFACE
      36             : constexpr int ILI2_GEOMCOLL_TYPE = 32;
      37             : 
      38             : static const char *const ILI2_COORD = "COORD";
      39             : static const char *const ILI2_ARC = "ARC";
      40             : static const char *const ILI2_POLYLINE = "POLYLINE";
      41             : static const char *const ILI2_BOUNDARY = "BOUNDARY";
      42             : static const char *const ILI2_AREA = "AREA";
      43             : static const char *const ILI2_SURFACE = "SURFACE";
      44             : 
      45             : namespace gdal
      46             : {
      47             : namespace ili2
      48             : {
      49             : //
      50             : // helper functions
      51             : //
      52       37750 : int cmpStr(const string &s1, const string &s2)
      53             : {
      54       37750 :     string::const_iterator p1 = s1.begin();
      55       37750 :     string::const_iterator p2 = s2.begin();
      56             : 
      57      129883 :     while (p1 != s1.end() && p2 != s2.end())
      58             :     {
      59      121749 :         if (CPLToupper(static_cast<unsigned char>(*p1)) !=
      60      121749 :             CPLToupper(static_cast<unsigned char>(*p2)))
      61       29616 :             return (CPLToupper(static_cast<unsigned char>(*p1)) <
      62       29616 :                     CPLToupper(static_cast<unsigned char>(*p2)))
      63       29616 :                        ? -1
      64       29616 :                        : 1;
      65       92133 :         ++p1;
      66       92133 :         ++p2;
      67             :     }
      68             : 
      69        8134 :     return (s2.size() == s1.size()) ? 0 : (s1.size() < s2.size()) ? -1 : 1;
      70             : }
      71             : 
      72        6686 : string ltrim(const string &tmpstr)
      73             : {
      74        6686 :     size_t i = 0;
      75       29407 :     while (i < tmpstr.length() && (tmpstr[i] == ' ' || tmpstr[i] == '\t' ||
      76        6686 :                                    tmpstr[i] == '\r' || tmpstr[i] == '\n'))
      77       16035 :         ++i;
      78        6686 :     return i > 0 ? tmpstr.substr(i, tmpstr.length() - i) : tmpstr;
      79             : }
      80             : 
      81        6686 : string rtrim(const string &tmpstr)
      82             : {
      83        6686 :     if (tmpstr.empty())
      84        1053 :         return tmpstr;
      85        5633 :     size_t i = tmpstr.length() - 1U;
      86       11266 :     while (tmpstr[i] == ' ' || tmpstr[i] == '\t' || tmpstr[i] == '\r' ||
      87        5633 :            tmpstr[i] == '\n')
      88           0 :         --i;
      89        5633 :     return i < tmpstr.length() - 1 ? tmpstr.substr(0, i + 1) : tmpstr;
      90             : }
      91             : 
      92        6686 : string trim(const string &tmpstr)
      93             : {
      94        6686 :     auto ret = ltrim(tmpstr);
      95        6686 :     ret = rtrim(ret);
      96        6686 :     return ret;
      97             : }
      98             : }  // namespace ili2
      99             : }  // namespace gdal
     100             : 
     101             : using namespace gdal::ili2;
     102             : 
     103        4041 : static int getGeometryTypeOfElem(DOMElement *elem)
     104             : {
     105        4041 :     int type = ILI2_STRING_TYPE;
     106        4041 :     if (elem == nullptr)
     107           0 :         return type;
     108        4041 :     char *pszTagName = XMLString::transcode(elem->getTagName());
     109             : 
     110        4041 :     if (elem->getNodeType() == DOMNode::ELEMENT_NODE)
     111             :     {
     112        4041 :         if (cmpStr(ILI2_COORD, pszTagName) == 0)
     113             :         {
     114          50 :             type = ILI2_COORD_TYPE;
     115             :         }
     116        3991 :         else if (cmpStr(ILI2_ARC, pszTagName) == 0)
     117             :         {
     118           0 :             type = ILI2_ARC_TYPE;
     119             :         }
     120        3991 :         else if (cmpStr(ILI2_POLYLINE, pszTagName) == 0)
     121             :         {
     122          43 :             type = ILI2_POLYLINE_TYPE;
     123             :         }
     124        3948 :         else if (cmpStr(ILI2_BOUNDARY, pszTagName) == 0)
     125             :         {
     126           0 :             type = ILI2_BOUNDARY_TYPE;
     127             :         }
     128        3948 :         else if (cmpStr(ILI2_AREA, pszTagName) == 0)
     129             :         {
     130           0 :             type = ILI2_AREA_TYPE;
     131             :         }
     132        3948 :         else if (cmpStr(ILI2_SURFACE, pszTagName) == 0)
     133             :         {
     134         145 :             type = ILI2_AREA_TYPE;
     135             :         }
     136             :     }
     137        4041 :     XMLString::release(&pszTagName);
     138        4041 :     return type;
     139             : }
     140             : 
     141        6100 : static char *getObjValue(DOMElement *elem)
     142             : {
     143        6100 :     DOMNode *child = elem->getFirstChild();
     144        6100 :     if ((child != nullptr) && (child->getNodeType() == DOMNode::TEXT_NODE))
     145             :     {
     146        5300 :         return CPLStrdup(transcode(child->getNodeValue()));
     147             :     }
     148             : 
     149         800 :     return nullptr;
     150             : }
     151             : 
     152         800 : static char *getREFValue(DOMElement *elem)
     153             : {
     154        1600 :     CPLString osREFValue(transcode(elem->getAttribute(ILI2_REF)));
     155        1600 :     return CPLStrdup(osREFValue);
     156             : }
     157             : 
     158        2370 : static OGRPoint *getPoint(DOMElement *elem)
     159             : {
     160             :     // elem -> COORD (or ARC)
     161        2370 :     DOMElement *coordElem = dynamic_cast<DOMElement *>(elem->getFirstChild());
     162        2370 :     if (coordElem == nullptr)
     163           0 :         return nullptr;
     164        2370 :     OGRPoint *pt = new OGRPoint();
     165             : 
     166        7110 :     while (coordElem != nullptr)
     167             :     {
     168        4740 :         char *pszTagName = XMLString::transcode(coordElem->getTagName());
     169        4740 :         char *pszObjValue = getObjValue(coordElem);
     170        4740 :         if (pszObjValue)
     171             :         {
     172        4740 :             if (cmpStr("C1", pszTagName) == 0)
     173        2370 :                 pt->setX(CPLAtof(pszObjValue));
     174        2370 :             else if (cmpStr("C2", pszTagName) == 0)
     175        2370 :                 pt->setY(CPLAtof(pszObjValue));
     176           0 :             else if (cmpStr("C3", pszTagName) == 0)
     177           0 :                 pt->setZ(CPLAtof(pszObjValue));
     178             :         }
     179        4740 :         CPLFree(pszObjValue);
     180        4740 :         XMLString::release(&pszTagName);
     181        4740 :         coordElem = dynamic_cast<DOMElement *>(coordElem->getNextSibling());
     182             :     }
     183        2370 :     pt->flattenTo2D();
     184        2370 :     return pt;
     185             : }
     186             : 
     187           0 : OGRCircularString *ILI2Reader::getArc(DOMElement *elem)
     188             : {
     189             :     // previous point -> start point
     190           0 :     auto elemPrev = dynamic_cast<DOMElement *>(elem->getPreviousSibling());
     191           0 :     if (elemPrev == nullptr)
     192           0 :         return nullptr;
     193           0 :     OGRPoint *ptStart = getPoint(elemPrev);  // COORD or ARC
     194           0 :     if (ptStart == nullptr)
     195           0 :         return nullptr;
     196             : 
     197             :     // elem -> ARC
     198           0 :     OGRCircularString *arc = new OGRCircularString();
     199             :     // end point
     200           0 :     OGRPoint *ptEnd = new OGRPoint();
     201             :     // point on the arc
     202           0 :     OGRPoint *ptOnArc = new OGRPoint();
     203             :     // double radius = 0; // radius
     204             : 
     205           0 :     DOMElement *arcElem = dynamic_cast<DOMElement *>(elem->getFirstChild());
     206           0 :     while (arcElem != nullptr)
     207             :     {
     208           0 :         char *pszTagName = XMLString::transcode(arcElem->getTagName());
     209           0 :         char *pszObjValue = getObjValue(arcElem);
     210           0 :         if (pszObjValue)
     211             :         {
     212           0 :             if (cmpStr("C1", pszTagName) == 0)
     213           0 :                 ptEnd->setX(CPLAtof(pszObjValue));
     214           0 :             else if (cmpStr("C2", pszTagName) == 0)
     215           0 :                 ptEnd->setY(CPLAtof(pszObjValue));
     216           0 :             else if (cmpStr("C3", pszTagName) == 0)
     217           0 :                 ptEnd->setZ(CPLAtof(pszObjValue));
     218           0 :             else if (cmpStr("A1", pszTagName) == 0)
     219           0 :                 ptOnArc->setX(CPLAtof(pszObjValue));
     220           0 :             else if (cmpStr("A2", pszTagName) == 0)
     221           0 :                 ptOnArc->setY(CPLAtof(pszObjValue));
     222           0 :             else if (cmpStr("A3", pszTagName) == 0)
     223           0 :                 ptOnArc->setZ(CPLAtof(pszObjValue));
     224           0 :             else if (cmpStr("R", pszTagName) == 0)
     225             :             {
     226             :                 // radius = CPLAtof(pszObjValue);
     227             :             }
     228             :         }
     229           0 :         CPLFree(pszObjValue);
     230           0 :         XMLString::release(&pszTagName);
     231           0 :         arcElem = dynamic_cast<DOMElement *>(arcElem->getNextSibling());
     232             :     }
     233           0 :     arc->addPoint(ptStart);
     234           0 :     arc->addPoint(ptOnArc);
     235           0 :     arc->addPoint(ptEnd);
     236           0 :     delete ptStart;
     237           0 :     delete ptOnArc;
     238           0 :     delete ptEnd;
     239           0 :     return arc;
     240             : }
     241             : 
     242         103 : static OGRCompoundCurve *getPolyline(DOMElement *elem)
     243             : {
     244             :     // elem -> POLYLINE
     245         103 :     OGRCompoundCurve *ogrCurve = new OGRCompoundCurve();
     246         103 :     OGRLineString *ls = new OGRLineString();
     247             : 
     248         103 :     DOMElement *lineElem = dynamic_cast<DOMElement *>(elem->getFirstChild());
     249        2494 :     while (lineElem != nullptr)
     250             :     {
     251        2391 :         char *pszTagName = XMLString::transcode(lineElem->getTagName());
     252        2391 :         if (cmpStr(ILI2_COORD, pszTagName) == 0)
     253             :         {
     254        2346 :             OGRPoint *poPoint = getPoint(lineElem);
     255        2346 :             if (poPoint)
     256             :             {
     257        2346 :                 ls->addPoint(poPoint);
     258        2346 :                 delete poPoint;
     259             :             }
     260             :         }
     261          45 :         else if (cmpStr(ILI2_ARC, pszTagName) == 0)
     262             :         {
     263             :             // Finish line and start arc
     264           0 :             if (ls->getNumPoints() > 1)
     265             :             {
     266           0 :                 ogrCurve->addCurveDirectly(ls);
     267           0 :                 ls = new OGRLineString();
     268             :             }
     269             :             else
     270             :             {
     271           0 :                 ls->empty();
     272             :             }
     273           0 :             OGRCircularString *arc = new OGRCircularString();
     274             :             // end point
     275           0 :             OGRPoint *ptEnd = new OGRPoint();
     276             :             // point on the arc
     277           0 :             OGRPoint *ptOnArc = new OGRPoint();
     278             :             // radius
     279             :             // double radius = 0;
     280             : 
     281             :             DOMElement *arcElem =
     282           0 :                 dynamic_cast<DOMElement *>(lineElem->getFirstChild());
     283           0 :             while (arcElem != nullptr)
     284             :             {
     285           0 :                 char *pszTagName2 = XMLString::transcode(arcElem->getTagName());
     286           0 :                 char *pszObjValue = getObjValue(arcElem);
     287           0 :                 if (pszObjValue)
     288             :                 {
     289           0 :                     if (cmpStr("C1", pszTagName2) == 0)
     290           0 :                         ptEnd->setX(CPLAtof(pszObjValue));
     291           0 :                     else if (cmpStr("C2", pszTagName2) == 0)
     292           0 :                         ptEnd->setY(CPLAtof(pszObjValue));
     293           0 :                     else if (cmpStr("C3", pszTagName2) == 0)
     294           0 :                         ptEnd->setZ(CPLAtof(pszObjValue));
     295           0 :                     else if (cmpStr("A1", pszTagName2) == 0)
     296           0 :                         ptOnArc->setX(CPLAtof(pszObjValue));
     297           0 :                     else if (cmpStr("A2", pszTagName2) == 0)
     298           0 :                         ptOnArc->setY(CPLAtof(pszObjValue));
     299           0 :                     else if (cmpStr("A3", pszTagName2) == 0)
     300           0 :                         ptOnArc->setZ(CPLAtof(pszObjValue));
     301           0 :                     else if (cmpStr("R", pszTagName2) == 0)
     302             :                     {
     303             :                         // radius = CPLAtof(pszObjValue);
     304             :                     }
     305             :                 }
     306           0 :                 CPLFree(pszObjValue);
     307           0 :                 XMLString::release(&pszTagName2);
     308             : 
     309           0 :                 arcElem = dynamic_cast<DOMElement *>(arcElem->getNextSibling());
     310             :             }
     311             : 
     312             :             auto elemPrev =
     313           0 :                 dynamic_cast<DOMElement *>(lineElem->getPreviousSibling());
     314           0 :             if (elemPrev)
     315             :             {
     316           0 :                 OGRPoint *ptStart = getPoint(elemPrev);  // COORD or ARC
     317           0 :                 if (ptStart)
     318           0 :                     arc->addPoint(ptStart);
     319           0 :                 delete ptStart;
     320             :             }
     321           0 :             arc->addPoint(ptOnArc);
     322           0 :             arc->addPoint(ptEnd);
     323           0 :             ogrCurve->addCurveDirectly(arc);
     324             : 
     325             :             // Add arc endpoint as next start point, if COORD sequence follows.
     326             :             DOMElement *nextElem =
     327           0 :                 dynamic_cast<DOMElement *>(lineElem->getNextSibling());
     328           0 :             if (nextElem)
     329             :             {
     330             :                 char *nextTagName =
     331           0 :                     XMLString::transcode(nextElem->getTagName());
     332           0 :                 if (cmpStr(ILI2_COORD, nextTagName) == 0)
     333             :                 {
     334           0 :                     ls->addPoint(ptEnd);
     335             :                 }
     336           0 :                 XMLString::release(&nextTagName);
     337             :             }
     338             : 
     339           0 :             delete ptEnd;
     340           0 :             delete ptOnArc;
     341             :         } /* else { // TODO: StructureValue in Polyline not yet supported
     342             :         } */
     343        2391 :         XMLString::release(&pszTagName);
     344             : 
     345        2391 :         lineElem = dynamic_cast<DOMElement *>(lineElem->getNextSibling());
     346             :     }
     347             : 
     348         103 :     if (ls->getNumPoints() > 1)
     349             :     {
     350         103 :         ogrCurve->addCurveDirectly(ls);
     351             :     }
     352             :     else
     353             :     {
     354           0 :         delete ls;
     355             :     }
     356         103 :     return ogrCurve;
     357             : }
     358             : 
     359          82 : static OGRCompoundCurve *getBoundary(DOMElement *elem)
     360             : {
     361             : 
     362          82 :     DOMElement *lineElem = dynamic_cast<DOMElement *>(elem->getFirstChild());
     363          82 :     if (lineElem != nullptr)
     364             :     {
     365          82 :         char *pszTagName = XMLString::transcode(lineElem->getTagName());
     366          82 :         if (cmpStr(ILI2_POLYLINE, pszTagName) == 0)
     367             :         {
     368          82 :             XMLString::release(&pszTagName);
     369          82 :             return getPolyline(lineElem);
     370             :         }
     371           0 :         XMLString::release(&pszTagName);
     372             :     }
     373             : 
     374           0 :     return new OGRCompoundCurve();
     375             : }
     376             : 
     377          72 : static OGRCurvePolygon *getPolygon(DOMElement *elem)
     378             : {
     379          72 :     OGRCurvePolygon *pg = new OGRCurvePolygon();
     380             : 
     381             :     DOMElement *boundaryElem =
     382          72 :         dynamic_cast<DOMElement *>(elem->getFirstChild());  // outer boundary
     383         154 :     while (boundaryElem != nullptr)
     384             :     {
     385          82 :         char *pszTagName = XMLString::transcode(boundaryElem->getTagName());
     386          82 :         if (cmpStr(ILI2_BOUNDARY, pszTagName) == 0)
     387             :         {
     388          82 :             OGRCompoundCurve *poCC = getBoundary(boundaryElem);
     389          82 :             if (pg->addRingDirectly(poCC) != OGRERR_NONE)
     390             :             {
     391           0 :                 delete poCC;
     392             :             }
     393             :         }
     394          82 :         XMLString::release(&pszTagName);
     395          82 :         boundaryElem = dynamic_cast<DOMElement *>(
     396          82 :             boundaryElem->getNextSibling());  // inner boundaries
     397             :     }
     398             : 
     399          72 :     return pg;
     400             : }
     401             : 
     402         117 : OGRGeometry *ILI2Reader::getGeometry(DOMElement *elem, int type)
     403             : {
     404         117 :     OGRGeometryCollection *gm = new OGRGeometryCollection();
     405             : 
     406         117 :     DOMElement *childElem = elem;
     407         117 :     while (childElem != nullptr)
     408             :     {
     409         117 :         char *pszTagName = XMLString::transcode(childElem->getTagName());
     410         117 :         switch (type)
     411             :         {
     412          24 :             case ILI2_COORD_TYPE:
     413          24 :                 if (cmpStr(ILI2_COORD, pszTagName) == 0)
     414             :                 {
     415          24 :                     delete gm;
     416          24 :                     XMLString::release(&pszTagName);
     417         117 :                     return getPoint(childElem);
     418             :                 }
     419           0 :                 break;
     420           0 :             case ILI2_ARC_TYPE:
     421             :                 // is it possible here? It have to be a ARC or COORD before
     422             :                 // (getPreviousSibling)
     423           0 :                 if (cmpStr(ILI2_ARC, pszTagName) == 0)
     424             :                 {
     425           0 :                     delete gm;
     426           0 :                     XMLString::release(&pszTagName);
     427           0 :                     return getArc(childElem);
     428             :                 }
     429           0 :                 break;
     430          21 :             case ILI2_POLYLINE_TYPE:
     431          21 :                 if (cmpStr(ILI2_POLYLINE, pszTagName) == 0)
     432             :                 {
     433          21 :                     delete gm;
     434          21 :                     XMLString::release(&pszTagName);
     435          21 :                     return getPolyline(childElem);
     436             :                 }
     437           0 :                 break;
     438           0 :             case ILI2_BOUNDARY_TYPE:
     439           0 :                 if (cmpStr(ILI2_BOUNDARY, pszTagName) == 0)
     440             :                 {
     441           0 :                     delete gm;
     442           0 :                     XMLString::release(&pszTagName);
     443           0 :                     return getPolyline(childElem);
     444             :                 }
     445           0 :                 break;
     446          72 :             case ILI2_AREA_TYPE:
     447         216 :                 if ((cmpStr(ILI2_AREA, pszTagName) == 0) ||
     448         144 :                     (cmpStr(ILI2_SURFACE, pszTagName) == 0))
     449             :                 {
     450          72 :                     delete gm;
     451          72 :                     XMLString::release(&pszTagName);
     452          72 :                     return getPolygon(childElem);
     453             :                 }
     454           0 :                 break;
     455           0 :             default:
     456           0 :                 if (type >= ILI2_GEOMCOLL_TYPE)
     457             :                 {
     458           0 :                     int subType = getGeometryTypeOfElem(childElem);  //????
     459           0 :                     OGRGeometry *poSubGeom = getGeometry(childElem, subType);
     460           0 :                     if (poSubGeom)
     461           0 :                         gm->addGeometryDirectly(poSubGeom);
     462             :                 }
     463           0 :                 break;
     464             :         }
     465           0 :         XMLString::release(&pszTagName);
     466             : 
     467             :         // GEOMCOLL
     468           0 :         childElem = dynamic_cast<DOMElement *>(childElem->getNextSibling());
     469             :     }
     470             : 
     471           0 :     return gm;
     472             : }
     473             : 
     474           3 : int ILI2Reader::ReadModel(OGRILI2DataSource *poDS, ImdReader *poImdReader,
     475             :                           const char *modelFilename)
     476             : {
     477           3 :     poImdReader->ReadModel(modelFilename);
     478          38 :     for (FeatureDefnInfos::const_iterator it =
     479           3 :              poImdReader->featureDefnInfos.begin();
     480          79 :          it != poImdReader->featureDefnInfos.end(); ++it)
     481             :     {
     482             :         OGRLayer *layer =
     483          38 :             new OGRILI2Layer(it->GetTableDefnRef(), it->poGeomFieldInfos, poDS);
     484          38 :         m_listLayer.push_back(layer);
     485             :     }
     486           3 :     return 0;
     487             : }
     488             : 
     489             : // Detect field name of value element
     490        1728 : static char *fieldName(DOMElement *elem)
     491             : {
     492        1728 :     DOMNode *node = elem;
     493        1728 :     if (getGeometryTypeOfElem(elem))
     494             :     {
     495         117 :         int depth = 0;  // Depth of value elem node
     496         702 :         for (node = elem; node; node = node->getParentNode())
     497         585 :             ++depth;
     498             :         // Field name is on level 4
     499         117 :         node = elem;
     500         234 :         for (int d = 0; d < depth - 4; ++d)
     501         117 :             node = node->getParentNode();
     502             :     }
     503        1728 :     if (node == nullptr)
     504             :     {
     505           0 :         CPLError(CE_Failure, CPLE_AssertionFailed, "node == NULL");
     506           0 :         return CPLStrdup("***bug***");
     507             :     }
     508        1728 :     return CPLStrdup(transcode(node->getNodeName()));
     509             : }
     510             : 
     511          14 : void ILI2Reader::setFieldDefn(OGRFeatureDefn *featureDef, DOMElement *elem)
     512             : {
     513          14 :     int type = 0;
     514             :     // recursively search children
     515          14 :     for (DOMNode *childNode = elem->getFirstChild();
     516          64 :          type == 0 && childNode &&
     517          25 :          childNode->getNodeType() == DOMNode::ELEMENT_NODE;
     518          25 :          childNode = childNode->getNextSibling())
     519             :     {
     520          25 :         DOMElement *childElem = dynamic_cast<DOMElement *>(childNode);
     521          25 :         CPLAssert(childElem);
     522          25 :         type = getGeometryTypeOfElem(childElem);
     523          25 :         if (type == 0)
     524             :         {
     525          39 :             if (childElem->getFirstChild() &&
     526          18 :                 childElem->getFirstChild()->getNodeType() ==
     527             :                     DOMNode::ELEMENT_NODE)
     528             :             {
     529           8 :                 setFieldDefn(featureDef, childElem);
     530             :             }
     531             :             else
     532             :             {
     533          13 :                 char *fName = fieldName(childElem);
     534          13 :                 if (featureDef->GetFieldIndex(fName) == -1)
     535             :                 {
     536          13 :                     CPLDebug("OGR_ILI", "AddFieldDefn: %s", fName);
     537          26 :                     OGRFieldDefn oFieldDefn(fName, OFTString);
     538          13 :                     featureDef->AddFieldDefn(&oFieldDefn);
     539             :                 }
     540          13 :                 CPLFree(fName);
     541             :             }
     542             :         }
     543             :     }
     544          14 : }
     545             : 
     546        1095 : void ILI2Reader::SetFieldValues(OGRFeature *feature, DOMElement *elem)
     547             : {
     548        1095 :     int type = 0;
     549             :     // recursively search children
     550        1095 :     for (DOMNode *childNode = elem->getFirstChild();
     551        5671 :          type == 0 && childNode &&
     552        2288 :          childNode->getNodeType() == DOMNode::ELEMENT_NODE;
     553        2288 :          childNode = childNode->getNextSibling())
     554             :     {
     555        2288 :         DOMElement *childElem = dynamic_cast<DOMElement *>(childNode);
     556        2288 :         CPLAssert(childElem);
     557        2288 :         type = getGeometryTypeOfElem(childElem);
     558        2288 :         if (type == 0)
     559             :         {
     560        3542 :             if (childElem->getFirstChild() &&
     561        1371 :                 childElem->getFirstChild()->getNodeType() ==
     562             :                     DOMNode::ELEMENT_NODE)
     563             :             {
     564         573 :                 SetFieldValues(feature, childElem);
     565             :             }
     566             :             else
     567             :             {
     568        1598 :                 char *fName = fieldName(childElem);
     569        1598 :                 int fIndex = feature->GetFieldIndex(fName);
     570        1598 :                 if (fIndex != -1)
     571             :                 {
     572        1360 :                     char *objVal = getObjValue(childElem);
     573        1360 :                     if (objVal == nullptr)
     574         800 :                         objVal = getREFValue(childElem);  // only to try
     575        1360 :                     feature->SetField(fIndex, objVal);
     576        1360 :                     CPLFree(objVal);
     577             :                 }
     578             :                 else
     579             :                 {
     580         238 :                     CPLDebug("OGR_ILI", "Attribute '%s' not found", fName);
     581         238 :                     m_missAttrs.push_back(fName);
     582             :                 }
     583        1598 :                 CPLFree(fName);
     584             :             }
     585             :         }
     586             :         else
     587             :         {
     588         117 :             char *fName = fieldName(childElem);
     589         117 :             int fIndex = feature->GetGeomFieldIndex(fName);
     590         117 :             OGRGeometry *geom = getGeometry(childElem, type);
     591         117 :             if (geom)
     592             :             {
     593         117 :                 if (fIndex == -1)
     594             :                 {  // Unknown model
     595          27 :                     feature->SetGeometryDirectly(geom);
     596             :                 }
     597             :                 else
     598             :                 {
     599             :                     OGRwkbGeometryType geomType =
     600          90 :                         feature->GetGeomFieldDefnRef(fIndex)->GetType();
     601          90 :                     if (geomType == wkbMultiLineString ||
     602             :                         geomType == wkbPolygon)
     603             :                     {
     604          38 :                         feature->SetGeomFieldDirectly(
     605          38 :                             fIndex, geom->getLinearGeometry());
     606          38 :                         delete geom;
     607             :                     }
     608             :                     else
     609             :                     {
     610          52 :                         feature->SetGeomFieldDirectly(fIndex, geom);
     611             :                     }
     612             :                 }
     613             :             }
     614         117 :             CPLFree(fName);
     615             :         }
     616             :     }
     617        1095 : }
     618             : 
     619             : //
     620             : // ILI2Reader
     621             : //
     622           4 : IILI2Reader::~IILI2Reader()
     623             : {
     624           4 : }
     625             : 
     626           4 : ILI2Reader::ILI2Reader()
     627             :     : m_pszFilename(nullptr), m_poILI2Handler(nullptr), m_poSAXReader(nullptr),
     628           4 :       m_bReadStarted(FALSE), m_bXercesInitialized(false)
     629             : {
     630           4 :     SetupParser();
     631           4 : }
     632             : 
     633           8 : ILI2Reader::~ILI2Reader()
     634             : {
     635           4 :     CPLFree(m_pszFilename);
     636             : 
     637           4 :     CleanupParser();
     638             : 
     639           4 :     if (m_bXercesInitialized)
     640           4 :         OGRDeinitializeXerces();
     641             : 
     642           4 :     list<OGRLayer *>::const_iterator layerIt = m_listLayer.begin();
     643          48 :     while (layerIt != m_listLayer.end())
     644             :     {
     645          44 :         OGRILI2Layer *tmpLayer = (OGRILI2Layer *)*layerIt;
     646          44 :         delete tmpLayer;
     647          44 :         ++layerIt;
     648             :     }
     649           8 : }
     650             : 
     651           4 : void ILI2Reader::SetSourceFile(const char *pszFilename)
     652             : {
     653           4 :     CPLFree(m_pszFilename);
     654           4 :     m_pszFilename = CPLStrdup(pszFilename);
     655           4 : }
     656             : 
     657           4 : int ILI2Reader::SetupParser()
     658             : {
     659             : 
     660           4 :     if (!m_bXercesInitialized)
     661             :     {
     662           4 :         if (!OGRInitializeXerces())
     663           0 :             return FALSE;
     664           4 :         m_bXercesInitialized = true;
     665             :     }
     666             : 
     667             :     // Cleanup any old parser.
     668           4 :     if (m_poSAXReader != nullptr)
     669           0 :         CleanupParser();
     670             : 
     671             :     // Create and initialize parser.
     672           4 :     m_poSAXReader = XMLReaderFactory::createXMLReader();
     673             : 
     674           4 :     m_poILI2Handler = new ILI2Handler(this);
     675             : 
     676           4 :     m_poSAXReader->setContentHandler(m_poILI2Handler);
     677           4 :     m_poSAXReader->setErrorHandler(m_poILI2Handler);
     678           4 :     m_poSAXReader->setLexicalHandler(m_poILI2Handler);
     679           4 :     m_poSAXReader->setEntityResolver(m_poILI2Handler);
     680           4 :     m_poSAXReader->setDTDHandler(m_poILI2Handler);
     681           4 :     m_poSAXReader->setFeature(XMLUni::fgXercesDisableDefaultEntityResolution,
     682           4 :                               true);
     683             : 
     684             :     /* No Validation
     685             :     #if (OGR_ILI2_VALIDATION)
     686             :         m_poSAXReader->setFeature(
     687             :             XMLString::transcode("http://xml.org/sax/features/validation"),
     688             :     true); m_poSAXReader->setFeature(
     689             :             XMLString::transcode("http://xml.org/sax/features/namespaces"),
     690             :     true);
     691             : 
     692             :         m_poSAXReader->setFeature( XMLUni::fgSAX2CoreNameSpaces, true );
     693             :         m_poSAXReader->setFeature( XMLUni::fgXercesSchema, true );
     694             : 
     695             :     //    m_poSAXReader->setDoSchema(true);
     696             :     //    m_poSAXReader->setValidationSchemaFullChecking(true);
     697             :     #else
     698             :     */
     699             :     XMLCh *tmpCh =
     700           4 :         XMLString::transcode("http://xml.org/sax/features/validation");
     701           4 :     m_poSAXReader->setFeature(tmpCh, false);
     702           4 :     XMLString::release(&tmpCh);
     703           4 :     tmpCh = XMLString::transcode("http://xml.org/sax/features/namespaces");
     704           4 :     m_poSAXReader->setFeature(tmpCh, false);
     705           4 :     XMLString::release(&tmpCh);
     706             :     // #endif
     707             : 
     708           4 :     m_bReadStarted = FALSE;
     709             : 
     710           4 :     return TRUE;
     711             : }
     712             : 
     713           4 : void ILI2Reader::CleanupParser()
     714             : {
     715           4 :     if (m_poSAXReader == nullptr)
     716           0 :         return;
     717             : 
     718           4 :     delete m_poSAXReader;
     719           4 :     m_poSAXReader = nullptr;
     720             : 
     721           4 :     delete m_poILI2Handler;
     722           4 :     m_poILI2Handler = nullptr;
     723             : 
     724           4 :     m_bReadStarted = FALSE;
     725             : }
     726             : 
     727           4 : int ILI2Reader::SaveClasses(const char *pszFile = nullptr)
     728             : {
     729             : 
     730             :     // Add logic later to determine reasonable default schema file.
     731           4 :     if (pszFile == nullptr)
     732           0 :         return FALSE;
     733             : 
     734           4 :     VSILFILE *fp = VSIFOpenL(pszFile, "rb");
     735           4 :     if (fp == nullptr)
     736           0 :         return FALSE;
     737             : 
     738           4 :     InputSource *is = OGRCreateXercesInputSource(fp);
     739             : 
     740             :     // parse and create layers and features
     741             :     try
     742             :     {
     743           4 :         CPLDebug("OGR_ILI", "Parsing %s", pszFile);
     744           4 :         m_poSAXReader->parse(*is);
     745           4 :         VSIFCloseL(fp);
     746           4 :         OGRDestroyXercesInputSource(is);
     747             :     }
     748           0 :     catch (const DOMException &toCatch)
     749             :     {
     750             :         // Can happen with createElement() in ILI2Handler::startElement()
     751           0 :         CPLError(CE_Failure, CPLE_AppDefined, "DOMException: %s\n",
     752           0 :                  transcode(toCatch.getMessage()).c_str());
     753           0 :         VSIFCloseL(fp);
     754           0 :         OGRDestroyXercesInputSource(is);
     755           0 :         return FALSE;
     756             :     }
     757           0 :     catch (const SAXException &toCatch)
     758             :     {
     759           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Parsing failed: %s\n",
     760           0 :                  transcode(toCatch.getMessage()).c_str());
     761           0 :         VSIFCloseL(fp);
     762           0 :         OGRDestroyXercesInputSource(is);
     763           0 :         return FALSE;
     764             :     }
     765             : 
     766           4 :     if (!m_missAttrs.empty())
     767             :     {
     768           1 :         m_missAttrs.sort();
     769           1 :         m_missAttrs.unique();
     770           2 :         string attrs = "";
     771           1 :         list<string>::const_iterator it;
     772           3 :         for (it = m_missAttrs.begin(); it != m_missAttrs.end(); ++it)
     773           2 :             attrs += *it + ", ";
     774             : 
     775           1 :         CPLError(CE_Warning, CPLE_NotSupported,
     776             :                  "Failed to add new definition to existing layers, attributes "
     777             :                  "not saved: %s",
     778             :                  attrs.c_str());
     779             :     }
     780             : 
     781           4 :     return TRUE;
     782             : }
     783             : 
     784           4 : list<OGRLayer *> ILI2Reader::GetLayers()
     785             : {
     786           4 :     return m_listLayer;
     787             : }
     788             : 
     789           0 : int ILI2Reader::GetLayerCount()
     790             : {
     791           0 :     return static_cast<int>(m_listLayer.size());
     792             : }
     793             : 
     794         522 : OGRLayer *ILI2Reader::GetLayer(const char *pszName)
     795             : {
     796        3848 :     for (list<OGRLayer *>::reverse_iterator layerIt = m_listLayer.rbegin();
     797        7174 :          layerIt != m_listLayer.rend(); ++layerIt)
     798             :     {
     799        3842 :         OGRFeatureDefn *fDef = (*layerIt)->GetLayerDefn();
     800        3842 :         if (cmpStr(fDef->GetName(), pszName) == 0)
     801             :         {
     802         516 :             return *layerIt;
     803             :         }
     804             :     }
     805           6 :     return nullptr;
     806             : }
     807             : 
     808         522 : int ILI2Reader::AddFeature(DOMElement *elem)
     809             : {
     810         522 :     CPLString osName(transcode(elem->getTagName()));
     811             :     // CPLDebug( "OGR_ILI", "Reading layer: %s", osName.c_str() );
     812             : 
     813             :     // test if this layer exist
     814         522 :     OGRILI2Layer *curLayer = dynamic_cast<OGRILI2Layer *>(GetLayer(osName));
     815         522 :     bool newLayer = (curLayer == nullptr);
     816             : 
     817             :     // add a layer
     818         522 :     if (newLayer)
     819             :     {
     820           6 :         CPLDebug("OGR_ILI", "Adding layer: %s", osName.c_str());
     821           6 :         OGRFeatureDefn *poFeatureDefn = new OGRFeatureDefn(osName);
     822           6 :         poFeatureDefn->SetGeomType(wkbUnknown);
     823           6 :         GeomFieldInfos oGeomFieldInfos;
     824           6 :         curLayer = new OGRILI2Layer(poFeatureDefn, oGeomFieldInfos, nullptr);
     825           6 :         m_listLayer.push_back(curLayer);
     826             :     }
     827             : 
     828             :     // the feature and field definition
     829         522 :     OGRFeatureDefn *featureDef = curLayer->GetLayerDefn();
     830         522 :     if (newLayer)
     831             :     {
     832             :         // add TID field
     833          12 :         OGRFieldDefn ofieldDefn(ILI2_TID, OFTString);
     834           6 :         featureDef->AddFieldDefn(&ofieldDefn);
     835             : 
     836           6 :         setFieldDefn(featureDef, elem);
     837             :     }
     838             : 
     839             :     // add the features
     840         522 :     OGRFeature *feature = new OGRFeature(featureDef);
     841             : 
     842             :     // assign TID
     843         522 :     int fIndex = feature->GetFieldIndex(ILI2_TID);
     844         522 :     if (fIndex != -1)
     845             :     {
     846         522 :         feature->SetField(
     847        1044 :             fIndex, transcode(elem->getAttribute(xmlch_ILI2_TID)).c_str());
     848             :     }
     849             :     else
     850             :     {
     851           0 :         CPLDebug("OGR_ILI", "'%s' not found", ILI2_TID);
     852             :     }
     853             : 
     854         522 :     SetFieldValues(feature, elem);
     855         522 :     curLayer->AddFeature(feature);
     856             : 
     857        1044 :     return 0;
     858             : }
     859             : 
     860           4 : IILI2Reader *CreateILI2Reader()
     861             : {
     862           4 :     return new ILI2Reader();
     863             : }
     864             : 
     865          81 : void DestroyILI2Reader(IILI2Reader *reader)
     866             : {
     867          81 :     if (reader)
     868           4 :         delete reader;
     869          81 : }

Generated by: LCOV version 1.14