LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/gmlutils - parsexsd.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 442 532 83.1 %
Date: 2025-01-18 12:42:00 Functions: 15 15 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  GML Reader
       4             :  * Purpose:  Implementation of GMLParseXSD()
       5             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2005, Frank Warmerdam
       9             :  * Copyright (c) 2010-2014, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "cpl_port.h"
      15             : #include "parsexsd.h"
      16             : 
      17             : #include <cmath>
      18             : #include <cstdlib>
      19             : #include <cstring>
      20             : #include <set>
      21             : #include <string>
      22             : #include <utility>
      23             : 
      24             : #include "cpl_conv.h"
      25             : #include "cpl_error.h"
      26             : #include "cpl_http.h"
      27             : #include "cpl_minixml.h"
      28             : #include "cpl_string.h"
      29             : #include "ogr_core.h"
      30             : 
      31             : /************************************************************************/
      32             : /*                              StripNS()                               */
      33             : /*                                                                      */
      34             : /*      Return potentially shortened form of string with namespace      */
      35             : /*      stripped off if there is one.  Returns pointer into             */
      36             : /*      original string.                                                */
      37             : /************************************************************************/
      38        2961 : static const char *StripNS(const char *pszFullValue)
      39             : 
      40             : {
      41        2961 :     const char *pszColon = strstr(pszFullValue, ":");
      42        2961 :     if (pszColon == nullptr)
      43         219 :         return pszFullValue;
      44             :     else
      45        2742 :         return pszColon + 1;
      46             : }
      47             : 
      48             : /************************************************************************/
      49             : /*                   GetSimpleTypeProperties()                          */
      50             : /************************************************************************/
      51             : 
      52         300 : static bool GetSimpleTypeProperties(CPLXMLNode *psTypeNode,
      53             :                                     GMLPropertyType *pGMLType, int *pnWidth,
      54             :                                     int *pnPrecision)
      55             : {
      56             :     const char *pszBase =
      57         300 :         StripNS(CPLGetXMLValue(psTypeNode, "restriction.base", ""));
      58             : 
      59         300 :     if (EQUAL(pszBase, "decimal"))
      60             :     {
      61          91 :         *pGMLType = GMLPT_Real;
      62             :         const char *pszWidth =
      63          91 :             CPLGetXMLValue(psTypeNode, "restriction.totalDigits.value", "0");
      64             :         const char *pszPrecision =
      65          91 :             CPLGetXMLValue(psTypeNode, "restriction.fractionDigits.value", "0");
      66          91 :         *pnWidth = atoi(pszWidth);
      67          91 :         *pnPrecision = atoi(pszPrecision);
      68          91 :         return true;
      69             :     }
      70             : 
      71         209 :     else if (EQUAL(pszBase, "float"))
      72             :     {
      73           1 :         *pGMLType = GMLPT_Float;
      74           1 :         return true;
      75             :     }
      76             : 
      77         208 :     else if (EQUAL(pszBase, "double"))
      78             :     {
      79           0 :         *pGMLType = GMLPT_Real;
      80           0 :         return true;
      81             :     }
      82             : 
      83         208 :     else if (EQUAL(pszBase, "integer"))
      84             :     {
      85          71 :         *pGMLType = GMLPT_Integer;
      86             :         const char *pszWidth =
      87          71 :             CPLGetXMLValue(psTypeNode, "restriction.totalDigits.value", "0");
      88          71 :         *pnWidth = atoi(pszWidth);
      89          71 :         return true;
      90             :     }
      91             : 
      92         137 :     else if (EQUAL(pszBase, "long"))
      93             :     {
      94          12 :         *pGMLType = GMLPT_Integer64;
      95             :         const char *pszWidth =
      96          12 :             CPLGetXMLValue(psTypeNode, "restriction.totalDigits.value", "0");
      97          12 :         *pnWidth = atoi(pszWidth);
      98          12 :         return true;
      99             :     }
     100             : 
     101         125 :     else if (EQUAL(pszBase, "unsignedLong"))
     102             :     {
     103             :         // Optimistically map to signed integer...
     104           0 :         *pGMLType = GMLPT_Integer64;
     105             :         const char *pszWidth =
     106           0 :             CPLGetXMLValue(psTypeNode, "restriction.totalDigits.value", "0");
     107           0 :         *pnWidth = atoi(pszWidth);
     108           0 :         return true;
     109             :     }
     110             : 
     111         125 :     else if (EQUAL(pszBase, "string"))
     112             :     {
     113         121 :         *pGMLType = GMLPT_String;
     114             :         const char *pszWidth =
     115         121 :             CPLGetXMLValue(psTypeNode, "restriction.maxLength.value", "0");
     116         121 :         *pnWidth = atoi(pszWidth);
     117         121 :         return true;
     118             :     }
     119             : 
     120           4 :     else if (EQUAL(pszBase, "date"))
     121             :     {
     122           0 :         *pGMLType = GMLPT_Date;
     123           0 :         return true;
     124             :     }
     125             : 
     126           4 :     else if (EQUAL(pszBase, "time"))
     127             :     {
     128           0 :         *pGMLType = GMLPT_Time;
     129           0 :         return true;
     130             :     }
     131             : 
     132           4 :     else if (EQUAL(pszBase, "dateTime"))
     133             :     {
     134           0 :         *pGMLType = GMLPT_DateTime;
     135           0 :         return true;
     136             :     }
     137             : 
     138           4 :     else if (EQUAL(pszBase, "boolean"))
     139             :     {
     140           3 :         *pGMLType = GMLPT_Boolean;
     141           3 :         return true;
     142             :     }
     143             : 
     144           1 :     else if (EQUAL(pszBase, "short"))
     145             :     {
     146           1 :         *pGMLType = GMLPT_Short;
     147           1 :         return true;
     148             :     }
     149             : 
     150           0 :     return false;
     151             : }
     152             : 
     153             : /************************************************************************/
     154             : /*                      LookForSimpleType()                             */
     155             : /************************************************************************/
     156             : 
     157          11 : static bool LookForSimpleType(CPLXMLNode *psSchemaNode,
     158             :                               const char *pszStrippedNSType,
     159             :                               GMLPropertyType *pGMLType, int *pnWidth,
     160             :                               int *pnPrecision)
     161             : {
     162          11 :     CPLXMLNode *psThis = psSchemaNode->psChild;
     163         102 :     for (; psThis != nullptr; psThis = psThis->psNext)
     164             :     {
     165         220 :         if (psThis->eType == CXT_Element &&
     166          95 :             EQUAL(psThis->pszValue, "simpleType") &&
     167           2 :             EQUAL(CPLGetXMLValue(psThis, "name", ""), pszStrippedNSType))
     168             :         {
     169           2 :             break;
     170             :         }
     171             :     }
     172          11 :     if (psThis == nullptr)
     173           9 :         return false;
     174             : 
     175           2 :     return GetSimpleTypeProperties(psThis, pGMLType, pnWidth, pnPrecision);
     176             : }
     177             : 
     178             : /************************************************************************/
     179             : /*                      GetSingleChildElement()                         */
     180             : /************************************************************************/
     181             : 
     182             : /* Returns the child element whose name is pszExpectedValue only if */
     183             : /* there is only one child that is an element. */
     184           3 : static CPLXMLNode *GetSingleChildElement(CPLXMLNode *psNode,
     185             :                                          const char *pszExpectedValue)
     186             : {
     187           3 :     if (psNode == nullptr)
     188           0 :         return nullptr;
     189             : 
     190           3 :     CPLXMLNode *psIter = psNode->psChild;
     191           3 :     if (psIter == nullptr)
     192           0 :         return nullptr;
     193             : 
     194           3 :     CPLXMLNode *psChild = nullptr;
     195           9 :     while (psIter != nullptr)
     196             :     {
     197           6 :         if (psIter->eType == CXT_Element)
     198             :         {
     199           3 :             if (psChild != nullptr)
     200           0 :                 return nullptr;
     201           3 :             if (pszExpectedValue != nullptr &&
     202           3 :                 strcmp(psIter->pszValue, pszExpectedValue) != 0)
     203           0 :                 return nullptr;
     204           3 :             psChild = psIter;
     205             :         }
     206           6 :         psIter = psIter->psNext;
     207             :     }
     208           3 :     return psChild;
     209             : }
     210             : 
     211             : /************************************************************************/
     212             : /*                      CheckMinMaxOccursCardinality()                  */
     213             : /************************************************************************/
     214             : 
     215           2 : static int CheckMinMaxOccursCardinality(CPLXMLNode *psNode)
     216             : {
     217           2 :     const char *pszMinOccurs = CPLGetXMLValue(psNode, "minOccurs", nullptr);
     218           2 :     const char *pszMaxOccurs = CPLGetXMLValue(psNode, "maxOccurs", nullptr);
     219           1 :     return (pszMinOccurs == nullptr || EQUAL(pszMinOccurs, "0") ||
     220           4 :             EQUAL(pszMinOccurs, "1")) &&
     221           3 :            (pszMaxOccurs == nullptr || EQUAL(pszMaxOccurs, "1"));
     222             : }
     223             : 
     224             : /************************************************************************/
     225             : /*                     GetListTypeFromSingleType()                      */
     226             : /************************************************************************/
     227             : 
     228          11 : static GMLPropertyType GetListTypeFromSingleType(GMLPropertyType eType)
     229             : {
     230          11 :     if (eType == GMLPT_String)
     231           2 :         return GMLPT_StringList;
     232           9 :     if (eType == GMLPT_Integer || eType == GMLPT_Short)
     233           2 :         return GMLPT_IntegerList;
     234           7 :     if (eType == GMLPT_Integer64)
     235           2 :         return GMLPT_Integer64List;
     236           5 :     if (eType == GMLPT_Real || eType == GMLPT_Float)
     237           2 :         return GMLPT_RealList;
     238           3 :     if (eType == GMLPT_Boolean)
     239           1 :         return GMLPT_BooleanList;
     240           2 :     if (eType == GMLPT_FeatureProperty)
     241           2 :         return GMLPT_FeaturePropertyList;
     242           0 :     return eType;
     243             : }
     244             : 
     245             : /************************************************************************/
     246             : /*                      ParseFeatureType()                              */
     247             : /************************************************************************/
     248             : 
     249             : typedef struct
     250             : {
     251             :     const char *pszName;
     252             :     OGRwkbGeometryType eType;
     253             : } AssocNameType;
     254             : 
     255             : static const AssocNameType apsPropertyTypes[] = {
     256             :     {"GeometryPropertyType", wkbUnknown},
     257             :     {"PointPropertyType", wkbPoint},
     258             :     {"LineStringPropertyType", wkbLineString},
     259             :     {"CurvePropertyType", wkbCompoundCurve},
     260             :     {"PolygonPropertyType", wkbPolygon},
     261             :     {"SurfacePropertyType", wkbCurvePolygon},
     262             :     {"MultiPointPropertyType", wkbMultiPoint},
     263             :     {"MultiLineStringPropertyType", wkbMultiLineString},
     264             :     {"MultiCurvePropertyType", wkbMultiCurve},
     265             :     {"MultiPolygonPropertyType", wkbMultiPolygon},
     266             :     {"MultiSurfacePropertyType", wkbMultiSurface},
     267             :     {"MultiGeometryPropertyType", wkbGeometryCollection},
     268             :     {"GeometryAssociationType", wkbUnknown},
     269             :     {nullptr, wkbUnknown},
     270             : };
     271             : 
     272             : /* Found in FME .xsd  (e.g. <element ref="gml:curveProperty" minOccurs="0"/>) */
     273             : static const AssocNameType apsRefTypes[] = {
     274             :     {"pointProperty", wkbPoint},
     275             :     {"curveProperty", wkbLineString},  // Should we promote to wkbCompoundCurve?
     276             :     {"surfaceProperty", wkbPolygon},   // Should we promote to wkbCurvePolygon?
     277             :     {"multiPointProperty", wkbMultiPoint},
     278             :     {"multiCurveProperty", wkbMultiLineString},
     279             :     // Should we promote to wkbMultiSurface?
     280             :     {"multiSurfaceProperty", wkbMultiPolygon},
     281             :     {nullptr, wkbUnknown},
     282             : };
     283             : 
     284             : static GMLFeatureClass *GMLParseFeatureType(CPLXMLNode *psSchemaNode,
     285             :                                             const char *pszName,
     286             :                                             CPLXMLNode *psThis);
     287             : 
     288         580 : static GMLFeatureClass *GMLParseFeatureType(CPLXMLNode *psSchemaNode,
     289             :                                             const char *pszName,
     290             :                                             const char *pszType)
     291             : {
     292         580 :     CPLXMLNode *psThis = psSchemaNode->psChild;
     293        5191 :     for (; psThis != nullptr; psThis = psThis->psNext)
     294             :     {
     295       12886 :         if (psThis->eType == CXT_Element &&
     296        6161 :             EQUAL(psThis->pszValue, "complexType") &&
     297         970 :             EQUAL(CPLGetXMLValue(psThis, "name", ""), pszType))
     298             :         {
     299         580 :             break;
     300             :         }
     301             :     }
     302         580 :     if (psThis == nullptr)
     303           0 :         return nullptr;
     304             : 
     305         580 :     return GMLParseFeatureType(psSchemaNode, pszName, psThis);
     306             : }
     307             : 
     308         594 : static GMLFeatureClass *GMLParseFeatureType(CPLXMLNode *psSchemaNode,
     309             :                                             const char *pszName,
     310             :                                             CPLXMLNode *psComplexType)
     311             : {
     312             : 
     313             :     /* -------------------------------------------------------------------- */
     314             :     /*      Grab the sequence of extensions greatgrandchild.                */
     315             :     /* -------------------------------------------------------------------- */
     316             :     CPLXMLNode *psAttrSeq =
     317         594 :         CPLGetXMLNode(psComplexType, "complexContent.extension.sequence");
     318             : 
     319         594 :     if (psAttrSeq == nullptr)
     320             :     {
     321           0 :         return nullptr;
     322             :     }
     323             : 
     324             :     /* -------------------------------------------------------------------- */
     325             :     /*      We are pretty sure this going to be a valid Feature class       */
     326             :     /*      now, so create it.                                              */
     327             :     /* -------------------------------------------------------------------- */
     328         594 :     GMLFeatureClass *poClass = new GMLFeatureClass(pszName);
     329             : 
     330             :     /* -------------------------------------------------------------------- */
     331             :     /*      Loop over each of the attribute elements being defined for      */
     332             :     /*      this feature class.                                             */
     333             :     /* -------------------------------------------------------------------- */
     334         594 :     int nAttributeIndex = 0;
     335             : 
     336         594 :     bool bGotUnrecognizedType = false;
     337             : 
     338         594 :     CPLXMLNode *psAttrDef = psAttrSeq->psChild;
     339        2913 :     for (; psAttrDef != nullptr; psAttrDef = psAttrDef->psNext)
     340             :     {
     341        2328 :         if (strcmp(psAttrDef->pszValue, "group") == 0)
     342             :         {
     343             :             /* Too complex schema for us. Aborts parsing */
     344           0 :             delete poClass;
     345           9 :             return nullptr;
     346             :         }
     347             : 
     348             :         /* Parse stuff like:
     349             :         <xs:choice>
     350             :             <xs:element ref="gml:polygonProperty"/>
     351             :             <xs:element ref="gml:multiPolygonProperty"/>
     352             :         </xs:choice>
     353             :         as found in
     354             :         https://downloadagiv.blob.core.windows.net/overstromingsgebieden-en-oeverzones/2014_01/Overstromingsgebieden_en_oeverzones_2014_01_GML.zip
     355             :         */
     356        2328 :         if (strcmp(psAttrDef->pszValue, "choice") == 0)
     357             :         {
     358           1 :             CPLXMLNode *psChild = psAttrDef->psChild;
     359           1 :             bool bPolygon = false;
     360           1 :             bool bMultiPolygon = false;
     361           3 :             for (; psChild; psChild = psChild->psNext)
     362             :             {
     363           2 :                 if (psChild->eType != CXT_Element)
     364           0 :                     continue;
     365           2 :                 if (strcmp(psChild->pszValue, "element") == 0)
     366             :                 {
     367             :                     const char *pszRef =
     368           2 :                         CPLGetXMLValue(psChild, "ref", nullptr);
     369           2 :                     if (pszRef != nullptr)
     370             :                     {
     371           2 :                         if (strcmp(pszRef, "gml:polygonProperty") == 0)
     372             :                         {
     373           1 :                             bPolygon = true;
     374             :                         }
     375           1 :                         else if (strcmp(pszRef, "gml:multiPolygonProperty") ==
     376             :                                  0)
     377             :                         {
     378           1 :                             bMultiPolygon = true;
     379             :                         }
     380             :                         else
     381             :                         {
     382           0 :                             delete poClass;
     383           0 :                             return nullptr;
     384             :                         }
     385             :                     }
     386             :                     else
     387             :                     {
     388           0 :                         delete poClass;
     389           0 :                         return nullptr;
     390             :                     }
     391             :                 }
     392             :             }
     393           1 :             if (bPolygon && bMultiPolygon)
     394             :             {
     395           2 :                 poClass->AddGeometryProperty(new GMLGeometryPropertyDefn(
     396           1 :                     "", "", wkbMultiPolygon, nAttributeIndex, true));
     397             : 
     398           1 :                 nAttributeIndex++;
     399             :             }
     400        2021 :             continue;
     401             :         }
     402             : 
     403        2327 :         if (!EQUAL(psAttrDef->pszValue, "element"))
     404          93 :             continue;
     405             : 
     406             :         // MapServer WFS writes element type as an attribute of element
     407             :         // not as a simpleType definition.
     408        2234 :         const char *pszType = CPLGetXMLValue(psAttrDef, "type", nullptr);
     409        2234 :         const char *pszElementName = CPLGetXMLValue(psAttrDef, "name", nullptr);
     410             :         bool bNullable =
     411        2234 :             EQUAL(CPLGetXMLValue(psAttrDef, "minOccurs", "1"), "0");
     412             :         const char *pszMaxOccurs =
     413        2234 :             CPLGetXMLValue(psAttrDef, "maxOccurs", nullptr);
     414        2234 :         if (pszType != nullptr)
     415             :         {
     416        1917 :             const char *pszStrippedNSType = StripNS(pszType);
     417        1917 :             int nWidth = 0;
     418        1917 :             int nPrecision = 0;
     419             : 
     420        1917 :             GMLPropertyType gmlType = GMLPT_Untyped;
     421        1917 :             if (EQUAL(pszStrippedNSType, "string") ||
     422        1411 :                 EQUAL(pszStrippedNSType, "Character"))
     423         506 :                 gmlType = GMLPT_String;
     424        1411 :             else if (EQUAL(pszStrippedNSType, "date"))
     425          63 :                 gmlType = GMLPT_Date;
     426        1348 :             else if (EQUAL(pszStrippedNSType, "time"))
     427           2 :                 gmlType = GMLPT_Time;
     428        1346 :             else if (EQUAL(pszStrippedNSType, "dateTime"))
     429         151 :                 gmlType = GMLPT_DateTime;
     430        1195 :             else if (EQUAL(pszStrippedNSType, "real") ||
     431        1195 :                      EQUAL(pszStrippedNSType, "double") ||
     432        1074 :                      EQUAL(pszStrippedNSType, "decimal"))
     433         153 :                 gmlType = GMLPT_Real;
     434        1042 :             else if (EQUAL(pszStrippedNSType, "float"))
     435         112 :                 gmlType = GMLPT_Float;
     436         930 :             else if (EQUAL(pszStrippedNSType, "int") ||
     437         799 :                      EQUAL(pszStrippedNSType, "integer"))
     438         132 :                 gmlType = GMLPT_Integer;
     439         798 :             else if (EQUAL(pszStrippedNSType, "long"))
     440           9 :                 gmlType = GMLPT_Integer64;
     441         789 :             else if (EQUAL(pszStrippedNSType, "unsignedLong"))
     442             :             {
     443             :                 // Optimistically map to signed integer
     444           0 :                 gmlType = GMLPT_Integer64;
     445             :             }
     446         789 :             else if (EQUAL(pszStrippedNSType, "short"))
     447         116 :                 gmlType = GMLPT_Short;
     448         673 :             else if (EQUAL(pszStrippedNSType, "boolean"))
     449         112 :                 gmlType = GMLPT_Boolean;
     450             :             // TODO: Would be nice to have a binary type.
     451         561 :             else if (EQUAL(pszStrippedNSType, "hexBinary"))
     452           0 :                 gmlType = GMLPT_String;
     453         561 :             else if (strcmp(pszType, "gml:FeaturePropertyType") == 0)
     454             :             {
     455           3 :                 gmlType = GMLPT_FeatureProperty;
     456             :             }
     457         558 :             else if (STARTS_WITH(pszType, "gml:"))
     458             :             {
     459         547 :                 const AssocNameType *psIter = apsPropertyTypes;
     460        1646 :                 while (psIter->pszName)
     461             :                 {
     462        1646 :                     if (strncmp(pszType + 4, psIter->pszName,
     463        1646 :                                 strlen(psIter->pszName)) == 0)
     464             :                     {
     465         547 :                         OGRwkbGeometryType eType = psIter->eType;
     466        1094 :                         std::string osSRSName;
     467             : 
     468             :                         // Look if there's a comment restricting to subclasses.
     469         547 :                         for (const CPLXMLNode *psIter2 = psAttrDef->psNext;
     470        1023 :                              psIter2 != nullptr; psIter2 = psIter2->psNext)
     471             :                         {
     472         476 :                             if (psIter2->eType == CXT_Comment)
     473             :                             {
     474         102 :                                 if (strstr(psIter2->pszValue,
     475             :                                            "restricted to Polygon"))
     476          17 :                                     eType = wkbPolygon;
     477          85 :                                 else if (strstr(psIter2->pszValue,
     478             :                                                 "restricted to LineString"))
     479          25 :                                     eType = wkbLineString;
     480          60 :                                 else if (strstr(psIter2->pszValue,
     481             :                                                 "restricted to MultiPolygon"))
     482          13 :                                     eType = wkbMultiPolygon;
     483          47 :                                 else if (strstr(
     484          47 :                                              psIter2->pszValue,
     485             :                                              "restricted to MultiLineString"))
     486          11 :                                     eType = wkbMultiLineString;
     487             :                                 else
     488             :                                 {
     489             :                                     const char *pszSRSName =
     490          36 :                                         strstr(psIter2->pszValue, "srsName=\"");
     491          36 :                                     if (pszSRSName)
     492             :                                     {
     493             :                                         osSRSName =
     494          32 :                                             pszSRSName + strlen("srsName=\"");
     495          32 :                                         const auto nPos = osSRSName.find('"');
     496          32 :                                         if (nPos != std::string::npos)
     497          32 :                                             osSRSName.resize(nPos);
     498             :                                         else
     499           0 :                                             osSRSName.clear();
     500             :                                     }
     501             :                                 }
     502             :                             }
     503             :                         }
     504             : 
     505             :                         // Try to get coordinate precision from a construct like:
     506             :                         /*
     507             :                             <xs:element name="wkb_geometry" type="gml:SurfacePropertyType" nillable="true" minOccurs="0" maxOccurs="1">
     508             :                                 <xs:annotation>
     509             :                                   <xs:appinfo source="http://ogr.maptools.org/">
     510             :                                     <ogr:xy_coordinate_resolution>8.9e-9</ogr:xy_coordinate_resolution>
     511             :                                     <ogr:z_coordinate_resolution>1e-3</ogr:z_coordinate_resolution>
     512             :                                     <ogr:m_coordinate_resolution>1e-3</ogr:m_coordinate_resolution>
     513             :                                   </xs:appinfo>
     514             :                                 </xs:annotation>
     515             :                             </xs:element>
     516             :                         */
     517        1094 :                         OGRGeomCoordinatePrecision oGeomCoordPrec;
     518             :                         const auto psAnnotation =
     519         547 :                             CPLGetXMLNode(psAttrDef, "annotation");
     520         547 :                         if (psAnnotation)
     521             :                         {
     522           1 :                             for (const CPLXMLNode *psIterAppinfo =
     523             :                                      psAnnotation->psChild;
     524           2 :                                  psIterAppinfo;
     525           1 :                                  psIterAppinfo = psIterAppinfo->psNext)
     526             :                             {
     527           3 :                                 if (psIterAppinfo->eType == CXT_Element &&
     528           1 :                                     strcmp(psIterAppinfo->pszValue,
     529           2 :                                            "appinfo") == 0 &&
     530           1 :                                     strcmp(CPLGetXMLValue(psIterAppinfo,
     531             :                                                           "source", ""),
     532             :                                            "http://ogr.maptools.org/") == 0)
     533             :                                 {
     534           1 :                                     if (const char *pszXYRes = CPLGetXMLValue(
     535             :                                             psIterAppinfo,
     536             :                                             "xy_coordinate_resolution",
     537             :                                             nullptr))
     538             :                                     {
     539           1 :                                         const double dfVal = CPLAtof(pszXYRes);
     540           1 :                                         if (dfVal > 0 && std::isfinite(dfVal))
     541           1 :                                             oGeomCoordPrec.dfXYResolution =
     542             :                                                 dfVal;
     543             :                                     }
     544           1 :                                     if (const char *pszZRes = CPLGetXMLValue(
     545             :                                             psIterAppinfo,
     546             :                                             "z_coordinate_resolution", nullptr))
     547             :                                     {
     548           1 :                                         const double dfVal = CPLAtof(pszZRes);
     549           1 :                                         if (dfVal > 0 && std::isfinite(dfVal))
     550           1 :                                             oGeomCoordPrec.dfZResolution =
     551             :                                                 dfVal;
     552             :                                     }
     553             :                                 }
     554             :                             }
     555             :                         }
     556             : 
     557             :                         GMLGeometryPropertyDefn *poDefn =
     558             :                             new GMLGeometryPropertyDefn(
     559             :                                 pszElementName, pszElementName, eType,
     560         547 :                                 nAttributeIndex, bNullable, oGeomCoordPrec);
     561         547 :                         poDefn->SetSRSName(osSRSName);
     562             : 
     563         547 :                         if (poClass->AddGeometryProperty(poDefn) < 0)
     564           0 :                             delete poDefn;
     565             :                         else
     566         547 :                             nAttributeIndex++;
     567             : 
     568         547 :                         break;
     569             :                     }
     570             : 
     571        1099 :                     psIter++;
     572             :                 }
     573             : 
     574         547 :                 if (psIter->pszName == nullptr)
     575             :                 {
     576             :                     // Can be a non geometry gml type.
     577             :                     // Too complex schema for us. Aborts parsing.
     578           0 :                     delete poClass;
     579           9 :                     return nullptr;
     580             :                 }
     581             : 
     582         547 :                 if (poClass->GetGeometryPropertyCount() == 0)
     583           0 :                     bGotUnrecognizedType = true;
     584             : 
     585        1908 :                 continue;
     586             :             }
     587             : 
     588             :             /* Integraph stuff */
     589          11 :             else if (strcmp(pszType, "G:Point_MultiPointPropertyType") == 0 ||
     590          11 :                      strcmp(pszType, "gmgml:Point_MultiPointPropertyType") == 0)
     591             :             {
     592             :                 GMLGeometryPropertyDefn *poDefn = new GMLGeometryPropertyDefn(
     593             :                     pszElementName, pszElementName, wkbMultiPoint,
     594           0 :                     nAttributeIndex, bNullable);
     595             : 
     596           0 :                 if (poClass->AddGeometryProperty(poDefn) < 0)
     597           0 :                     delete poDefn;
     598             :                 else
     599           0 :                     nAttributeIndex++;
     600             : 
     601           0 :                 continue;
     602             :             }
     603          11 :             else if (strcmp(pszType,
     604          11 :                             "G:LineString_MultiLineStringPropertyType") == 0 ||
     605          11 :                      strcmp(pszType,
     606             :                             "gmgml:LineString_MultiLineStringPropertyType") ==
     607             :                          0)
     608             :             {
     609             :                 GMLGeometryPropertyDefn *poDefn = new GMLGeometryPropertyDefn(
     610             :                     pszElementName, pszElementName, wkbMultiLineString,
     611           0 :                     nAttributeIndex, bNullable);
     612             : 
     613           0 :                 if (poClass->AddGeometryProperty(poDefn) < 0)
     614           0 :                     delete poDefn;
     615             :                 else
     616           0 :                     nAttributeIndex++;
     617             : 
     618           0 :                 continue;
     619             :             }
     620          11 :             else if (strcmp(pszType, "G:Polygon_MultiPolygonPropertyType") ==
     621          11 :                          0 ||
     622          11 :                      strcmp(pszType,
     623          11 :                             "gmgml:Polygon_MultiPolygonPropertyType") == 0 ||
     624          11 :                      strcmp(pszType, "gmgml:Polygon_Surface_MultiSurface_"
     625             :                                      "CompositeSurfacePropertyType") == 0)
     626             :             {
     627             :                 GMLGeometryPropertyDefn *poDefn = new GMLGeometryPropertyDefn(
     628             :                     pszElementName, pszElementName, wkbMultiPolygon,
     629           0 :                     nAttributeIndex, bNullable);
     630             : 
     631           0 :                 if (poClass->AddGeometryProperty(poDefn) < 0)
     632           0 :                     delete poDefn;
     633             :                 else
     634           0 :                     nAttributeIndex++;
     635             : 
     636           0 :                 continue;
     637             :             }
     638             : 
     639             :             // ERDAS Apollo stufflike in
     640             :             // http://apollo.erdas.com/erdas-apollo/vector/WORLDWIDE?SERVICE=WFS&VERSION=1.0.0&REQUEST=DescribeFeatureType&TYPENAME=wfs:cntry98)
     641          11 :             else if (strcmp(pszType, "wfs:MixedPolygonPropertyType") == 0)
     642             :             {
     643             :                 GMLGeometryPropertyDefn *poDefn = new GMLGeometryPropertyDefn(
     644             :                     pszElementName, pszElementName, wkbMultiPolygon,
     645           0 :                     nAttributeIndex, bNullable);
     646             : 
     647           0 :                 if (poClass->AddGeometryProperty(poDefn) < 0)
     648           0 :                     delete poDefn;
     649             :                 else
     650           0 :                     nAttributeIndex++;
     651             : 
     652           0 :                 continue;
     653             :             }
     654             : 
     655             :             else
     656             :             {
     657          11 :                 gmlType = GMLPT_Untyped;
     658          11 :                 if (!LookForSimpleType(psSchemaNode, pszStrippedNSType,
     659             :                                        &gmlType, &nWidth, &nPrecision))
     660             :                 {
     661             :                     // Too complex schema for us. Aborts parsing.
     662           9 :                     delete poClass;
     663           9 :                     return nullptr;
     664             :                 }
     665             :             }
     666             : 
     667        1361 :             if (pszElementName == nullptr)
     668           0 :                 pszElementName = "unnamed";
     669        1361 :             const char *pszPropertyName = pszElementName;
     670        1361 :             if (gmlType == GMLPT_FeatureProperty)
     671             :             {
     672           3 :                 pszPropertyName = CPLSPrintf("%s_href", pszElementName);
     673             :             }
     674             :             GMLPropertyDefn *poProp =
     675        1361 :                 new GMLPropertyDefn(pszPropertyName, pszElementName);
     676             : 
     677        1361 :             if (pszMaxOccurs != nullptr && strcmp(pszMaxOccurs, "1") != 0)
     678           2 :                 gmlType = GetListTypeFromSingleType(gmlType);
     679             : 
     680        1361 :             poProp->SetType(gmlType);
     681        1361 :             poProp->SetWidth(nWidth);
     682        1361 :             poProp->SetPrecision(nPrecision);
     683        1361 :             poProp->SetNullable(bNullable);
     684             : 
     685             :             const CPLXMLNode *psAnnotation =
     686        1361 :                 CPLGetXMLNode(psAttrDef, "annotation");
     687        1361 :             if (psAnnotation)
     688             :             {
     689             :                 const char *pszDocumentation =
     690           3 :                     CPLGetXMLValue(psAnnotation, "documentation", nullptr);
     691           3 :                 if (pszDocumentation)
     692           3 :                     poProp->SetDocumentation(pszDocumentation);
     693             :             }
     694             : 
     695        1361 :             if (poClass->AddProperty(poProp) < 0)
     696           0 :                 delete poProp;
     697             :             else
     698        1361 :                 nAttributeIndex++;
     699             : 
     700        1361 :             continue;
     701             :         }
     702             : 
     703             :         // For now we skip geometries.  Fixup later.
     704         317 :         CPLXMLNode *psSimpleType = CPLGetXMLNode(psAttrDef, "simpleType");
     705         317 :         if (psSimpleType == nullptr)
     706             :         {
     707          19 :             const char *pszRef = CPLGetXMLValue(psAttrDef, "ref", nullptr);
     708             : 
     709             :             // FME .xsd
     710          19 :             if (pszRef != nullptr && STARTS_WITH(pszRef, "gml:"))
     711             :             {
     712          18 :                 const AssocNameType *psIter = apsRefTypes;
     713          63 :                 while (psIter->pszName)
     714             :                 {
     715          63 :                     if (strncmp(pszRef + 4, psIter->pszName,
     716          63 :                                 strlen(psIter->pszName)) == 0)
     717             :                     {
     718          18 :                         if (poClass->GetGeometryPropertyCount() > 0)
     719             :                         {
     720           9 :                             OGRwkbGeometryType eNewType = psIter->eType;
     721             :                             OGRwkbGeometryType eOldType =
     722             :                                 static_cast<OGRwkbGeometryType>(
     723           9 :                                     poClass->GetGeometryProperty(0)->GetType());
     724             : 
     725           9 :                             if ((eNewType == wkbMultiPoint &&
     726           6 :                                  eOldType == wkbPoint) ||
     727           3 :                                 (eNewType == wkbMultiLineString &&
     728           3 :                                  eOldType == wkbLineString) ||
     729           3 :                                 (eNewType == wkbMultiPolygon &&
     730             :                                  eOldType == wkbPolygon))
     731             :                             {
     732           9 :                                 poClass->GetGeometryProperty(0)->SetType(
     733             :                                     eNewType);
     734             :                             }
     735             :                             else
     736             :                             {
     737           0 :                                 CPLDebug("GML",
     738             :                                          "Geometry field already found ! "
     739             :                                          "Ignoring the following ones");
     740             :                             }
     741             :                         }
     742             :                         else
     743             :                         {
     744             :                             GMLGeometryPropertyDefn *poDefn =
     745             :                                 new GMLGeometryPropertyDefn(
     746             :                                     pszElementName, pszElementName,
     747           9 :                                     psIter->eType, nAttributeIndex, true);
     748             : 
     749           9 :                             if (poClass->AddGeometryProperty(poDefn) < 0)
     750           0 :                                 delete poDefn;
     751             :                             else
     752           9 :                                 nAttributeIndex++;
     753             :                         }
     754             : 
     755          18 :                         break;
     756             :                     }
     757             : 
     758          45 :                     psIter++;
     759             :                 }
     760             : 
     761          18 :                 if (psIter->pszName == nullptr)
     762             :                 {
     763             :                     // Can be a non geometry gml type .
     764             :                     // Too complex schema for us. Aborts parsing.
     765           0 :                     delete poClass;
     766           0 :                     return nullptr;
     767             :                 }
     768             : 
     769          18 :                 if (poClass->GetGeometryPropertyCount() == 0)
     770           0 :                     bGotUnrecognizedType = true;
     771             : 
     772          18 :                 continue;
     773             :             }
     774             : 
     775             :             /* Parse stuff like the following found in
     776             :             http://199.29.1.81:8181/miwfs/GetFeature.ashx?REQUEST=GetFeature&MAXFEATURES=1&SERVICE=WFS&VERSION=1.0.0&TYPENAME=miwfs:World
     777             :             : <xs:element name="Obj" minOccurs="0" maxOccurs="1">
     778             :                 <xs:complexType>
     779             :                     <xs:sequence>
     780             :                         <xs:element ref="gml:_Geometry"/>
     781             :                     </xs:sequence>
     782             :                 </xs:complexType>
     783             :             </xs:element>
     784             :             */
     785             :             CPLXMLNode *l_psComplexType =
     786           1 :                 GetSingleChildElement(psAttrDef, "complexType");
     787             :             CPLXMLNode *psComplexTypeSequence =
     788           1 :                 GetSingleChildElement(l_psComplexType, "sequence");
     789             :             CPLXMLNode *psComplexTypeSequenceElement =
     790           1 :                 GetSingleChildElement(psComplexTypeSequence, "element");
     791             : 
     792           2 :             if (pszElementName != nullptr &&
     793           1 :                 CheckMinMaxOccursCardinality(psAttrDef) &&
     794           1 :                 psComplexTypeSequenceElement != nullptr &&
     795           3 :                 CheckMinMaxOccursCardinality(psComplexTypeSequence) &&
     796           1 :                 strcmp(CPLGetXMLValue(psComplexTypeSequenceElement, "ref", ""),
     797             :                        "gml:_Geometry") == 0)
     798             :             {
     799             :                 GMLGeometryPropertyDefn *poDefn = new GMLGeometryPropertyDefn(
     800             :                     pszElementName, pszElementName, wkbUnknown, nAttributeIndex,
     801           1 :                     bNullable);
     802             : 
     803           1 :                 if (poClass->AddGeometryProperty(poDefn) < 0)
     804           0 :                     delete poDefn;
     805             :                 else
     806           1 :                     nAttributeIndex++;
     807             : 
     808           1 :                 continue;
     809             :             }
     810             :             else
     811             :             {
     812             :                 // Too complex schema for us. Aborts parsing.
     813           0 :                 delete poClass;
     814           0 :                 return nullptr;
     815             :             }
     816             :         }
     817             : 
     818         298 :         if (pszElementName == nullptr)
     819           0 :             pszElementName = "unnamed";
     820             :         GMLPropertyDefn *poProp =
     821         298 :             new GMLPropertyDefn(pszElementName, pszElementName);
     822             : 
     823         298 :         GMLPropertyType eType = GMLPT_Untyped;
     824         298 :         int nWidth = 0;
     825         298 :         int nPrecision = 0;
     826         298 :         GetSimpleTypeProperties(psSimpleType, &eType, &nWidth, &nPrecision);
     827             : 
     828         298 :         if (pszMaxOccurs != nullptr && strcmp(pszMaxOccurs, "1") != 0)
     829           9 :             eType = GetListTypeFromSingleType(eType);
     830             : 
     831         298 :         poProp->SetType(eType);
     832         298 :         poProp->SetWidth(nWidth);
     833         298 :         poProp->SetPrecision(nPrecision);
     834         298 :         poProp->SetNullable(bNullable);
     835             : 
     836         298 :         const CPLXMLNode *psAnnotation = CPLGetXMLNode(psAttrDef, "annotation");
     837         298 :         if (psAnnotation)
     838             :         {
     839             :             const char *pszDocumentation =
     840           8 :                 CPLGetXMLValue(psAnnotation, "documentation", nullptr);
     841           8 :             if (pszDocumentation)
     842           8 :                 poProp->SetDocumentation(pszDocumentation);
     843             :         }
     844             : 
     845         298 :         if (poClass->AddProperty(poProp) < 0)
     846           0 :             delete poProp;
     847             :         else
     848         298 :             nAttributeIndex++;
     849             :     }
     850             : 
     851             :     // If we have found an unknown types, let's be on the side of caution and
     852             :     // create a geometry field.
     853         585 :     if (poClass->GetGeometryPropertyCount() == 0 && bGotUnrecognizedType)
     854             :     {
     855           0 :         poClass->AddGeometryProperty(
     856           0 :             new GMLGeometryPropertyDefn("", "", wkbUnknown, -1, true));
     857             :     }
     858             : 
     859             :     /* -------------------------------------------------------------------- */
     860             :     /*      Class complete, add to reader class list.                       */
     861             :     /* -------------------------------------------------------------------- */
     862         585 :     poClass->SetSchemaLocked(true);
     863             : 
     864         585 :     return poClass;
     865             : }
     866             : 
     867             : /************************************************************************/
     868             : /*                         ExcludeBaseGMLSchemas()                      */
     869             : /************************************************************************/
     870             : 
     871          12 : static bool ExcludeBaseGMLSchemas(const std::string &osFilename)
     872             : {
     873             :     // List of substrings to exclude
     874             :     const std::vector<std::string> excludedBaseGMLReferencedSchemasList = {
     875          96 :         "/gml/3.2.1/", "gml/3.1.1/", "/gml/2.1.2/", "/gmlsfProfile/"};
     876          28 :     for (const auto &pattern : excludedBaseGMLReferencedSchemasList)
     877             :     {
     878          24 :         if (osFilename.find(pattern) != std::string::npos)
     879             :         {
     880           8 :             return false;  // Found one of the excluded base GML referenced schema
     881             :         }
     882             :     }
     883           4 :     return true;  // None of the base GML referenced schemas were found
     884             : }
     885             : 
     886             : /************************************************************************/
     887             : /*                         GMLParseXMLFile()                            */
     888             : /************************************************************************/
     889             : 
     890         469 : static CPLXMLNode *GMLParseXMLFile(const char *pszFilename)
     891             : {
     892         469 :     if (STARTS_WITH(pszFilename, "http://") ||
     893         468 :         STARTS_WITH(pszFilename, "https://"))
     894             :     {
     895           1 :         CPLXMLNode *psRet = nullptr;
     896           1 :         CPLHTTPResult *psResult = CPLHTTPFetch(pszFilename, nullptr);
     897           1 :         if (psResult != nullptr)
     898             :         {
     899           1 :             if (psResult->pabyData != nullptr)
     900             :             {
     901           1 :                 psRet = CPLParseXMLString(
     902           1 :                     reinterpret_cast<const char *>(psResult->pabyData));
     903             :             }
     904           1 :             CPLHTTPDestroyResult(psResult);
     905             :         }
     906           1 :         return psRet;
     907             :     }
     908             :     else
     909             :     {
     910         468 :         return CPLParseXMLFile(pszFilename);
     911             :     }
     912             : }
     913             : 
     914             : /************************************************************************/
     915             : /*                       CPLGetFirstChildNode()                         */
     916             : /************************************************************************/
     917             : 
     918          12 : static CPLXMLNode *CPLGetFirstChildNode(CPLXMLNode *psNode)
     919             : {
     920          12 :     if (psNode == nullptr)
     921           0 :         return nullptr;
     922          12 :     CPLXMLNode *psIter = psNode->psChild;
     923          94 :     while (psIter != nullptr)
     924             :     {
     925          94 :         if (psIter->eType == CXT_Element)
     926          12 :             return psIter;
     927          82 :         psIter = psIter->psNext;
     928             :     }
     929           0 :     return nullptr;
     930             : }
     931             : 
     932             : /************************************************************************/
     933             : /*                          CPLGetLastNode()                            */
     934             : /************************************************************************/
     935             : 
     936          12 : static CPLXMLNode *CPLGetLastNode(CPLXMLNode *psNode)
     937             : {
     938          12 :     CPLXMLNode *psIter = psNode;
     939          23 :     while (psIter->psNext != nullptr)
     940          11 :         psIter = psIter->psNext;
     941          12 :     return psIter;
     942             : }
     943             : 
     944             : /************************************************************************/
     945             : /*                       CPLXMLSchemaResolveInclude()                   */
     946             : /************************************************************************/
     947             : 
     948         457 : static void CPLXMLSchemaResolveInclude(const char *pszMainSchemaLocation,
     949             :                                        CPLXMLNode *psSchemaNode,
     950             :                                        bool bUseSchemaImports)
     951             : {
     952         914 :     std::set<CPLString> osAlreadyIncluded;
     953             : 
     954             :     bool bTryAgain;
     955         465 :     do
     956             :     {
     957         465 :         CPLXMLNode *psLast = nullptr;
     958         465 :         bTryAgain = false;
     959             : 
     960         465 :         CPLXMLNode *psThis = psSchemaNode->psChild;
     961        5015 :         for (; psThis != nullptr; psThis = psThis->psNext)
     962             :         {
     963             :             std::string osSchemaLocation =
     964        4550 :                 CPLGetXMLValue(psThis, "schemaLocation", "");
     965             : 
     966        6782 :             if (psThis->eType == CXT_Element &&
     967        2232 :                 (EQUAL(psThis->pszValue, "include") ||
     968        2223 :                  (bUseSchemaImports == TRUE &&
     969          20 :                   EQUAL(psThis->pszValue, "import") &&
     970          12 :                   ExcludeBaseGMLSchemas(osSchemaLocation))))
     971             :             {
     972             : 
     973          26 :                 if (!osSchemaLocation.empty() &&
     974          26 :                     osAlreadyIncluded.count(osSchemaLocation) == 0)
     975             :                 {
     976          12 :                     osAlreadyIncluded.insert(osSchemaLocation);
     977             : 
     978          12 :                     if (!STARTS_WITH(osSchemaLocation.c_str(), "http://") &&
     979          24 :                         !STARTS_WITH(osSchemaLocation.c_str(), "https://") &&
     980          12 :                         CPLIsFilenameRelative(osSchemaLocation.c_str()))
     981             :                     {
     982          36 :                         osSchemaLocation = CPLFormFilenameSafe(
     983          24 :                             CPLGetPathSafe(pszMainSchemaLocation).c_str(),
     984          12 :                             osSchemaLocation.c_str(), nullptr);
     985             :                     }
     986             : 
     987             :                     CPLXMLNode *psIncludedXSDTree =
     988          12 :                         GMLParseXMLFile(osSchemaLocation.c_str());
     989          12 :                     if (psIncludedXSDTree != nullptr)
     990             :                     {
     991          12 :                         CPLStripXMLNamespace(psIncludedXSDTree, nullptr, TRUE);
     992             :                         CPLXMLNode *psIncludedSchemaNode =
     993          12 :                             CPLGetXMLNode(psIncludedXSDTree, "=schema");
     994          12 :                         if (psIncludedSchemaNode != nullptr)
     995             :                         {
     996             :                             // Substitute de <include> node by its content.
     997             :                             CPLXMLNode *psFirstChildElement =
     998          12 :                                 CPLGetFirstChildNode(psIncludedSchemaNode);
     999          12 :                             if (psFirstChildElement != nullptr)
    1000             :                             {
    1001             :                                 CPLXMLNode *psCopy =
    1002          12 :                                     CPLCloneXMLTree(psFirstChildElement);
    1003          12 :                                 if (psLast != nullptr)
    1004          12 :                                     psLast->psNext = psCopy;
    1005             :                                 else
    1006           0 :                                     psSchemaNode->psChild = psCopy;
    1007          12 :                                 CPLXMLNode *psNext = psThis->psNext;
    1008          12 :                                 psThis->psNext = nullptr;
    1009          12 :                                 CPLDestroyXMLNode(psThis);
    1010          12 :                                 psThis = CPLGetLastNode(psCopy);
    1011          12 :                                 psThis->psNext = psNext;
    1012             : 
    1013             :                                 // In case the included schema also contains
    1014             :                                 // includes.
    1015          12 :                                 bTryAgain = true;
    1016             :                             }
    1017             :                         }
    1018          12 :                         CPLDestroyXMLNode(psIncludedXSDTree);
    1019             :                     }
    1020             :                 }
    1021             :             }
    1022             : 
    1023        4550 :             psLast = psThis;
    1024             :         }
    1025             :     } while (bTryAgain);
    1026             : 
    1027             :     const char *pszSchemaOutputName =
    1028         457 :         CPLGetConfigOption("GML_SCHEMA_OUTPUT_NAME", nullptr);
    1029         457 :     if (pszSchemaOutputName != nullptr)
    1030             :     {
    1031           0 :         CPLSerializeXMLTreeToFile(psSchemaNode, pszSchemaOutputName);
    1032             :     }
    1033         457 : }
    1034             : 
    1035             : /************************************************************************/
    1036             : /*                       GetUniqueConstraints()                         */
    1037             : /************************************************************************/
    1038             : 
    1039             : static std::set<std::pair<std::string, std::string>>
    1040         144 : GetUniqueConstraints(const CPLXMLNode *psNode)
    1041             : {
    1042             :     /* Parse
    1043             :         <xs:unique name="uniqueConstraintpolyeas_id">
    1044             :             <xs:selector xpath="ogr:featureMember/ogr:poly"/>
    1045             :             <xs:field xpath="ogr:eas_id"/>
    1046             :         </xs:unique>
    1047             :     */
    1048         144 :     std::set<std::pair<std::string, std::string>> oSet;
    1049         576 :     for (const auto *psIter = psNode->psChild; psIter != nullptr;
    1050         432 :          psIter = psIter->psNext)
    1051             :     {
    1052         432 :         if (psIter->eType == CXT_Element && EQUAL(psIter->pszValue, "unique"))
    1053             :         {
    1054             :             const char *pszSelector =
    1055           0 :                 CPLGetXMLValue(psIter, "selector.xpath", nullptr);
    1056             :             const char *pszField =
    1057           0 :                 CPLGetXMLValue(psIter, "field.xpath", nullptr);
    1058           0 :             if (pszSelector && pszField && pszField[0] != '@')
    1059             :             {
    1060           0 :                 const char *pszSlash = strchr(pszSelector, '/');
    1061           0 :                 if (pszSlash)
    1062             :                 {
    1063             :                     oSet.insert(
    1064           0 :                         std::pair(StripNS(pszSlash + 1), StripNS(pszField)));
    1065             :                 }
    1066             :             }
    1067             :         }
    1068             :     }
    1069         144 :     return oSet;
    1070             : }
    1071             : 
    1072             : /************************************************************************/
    1073             : /*                          GMLParseXSD()                               */
    1074             : /************************************************************************/
    1075             : 
    1076         457 : bool GMLParseXSD(const char *pszFile, bool bUseSchemaImports,
    1077             :                  std::vector<GMLFeatureClass *> &aosClasses,
    1078             :                  bool &bFullyUnderstood)
    1079             : 
    1080             : {
    1081         457 :     bFullyUnderstood = false;
    1082             : 
    1083         457 :     if (pszFile == nullptr)
    1084           0 :         return false;
    1085             : 
    1086             :     /* -------------------------------------------------------------------- */
    1087             :     /*      Load the raw XML file.                                          */
    1088             :     /* -------------------------------------------------------------------- */
    1089         457 :     CPLXMLNode *psXSDTree = GMLParseXMLFile(pszFile);
    1090             : 
    1091         457 :     if (psXSDTree == nullptr)
    1092           0 :         return false;
    1093             : 
    1094             :     /* -------------------------------------------------------------------- */
    1095             :     /*      Strip off any namespace qualifiers.                             */
    1096             :     /* -------------------------------------------------------------------- */
    1097         457 :     CPLStripXMLNamespace(psXSDTree, nullptr, TRUE);
    1098             : 
    1099             :     /* -------------------------------------------------------------------- */
    1100             :     /*      Find <schema> root element.                                     */
    1101             :     /* -------------------------------------------------------------------- */
    1102         457 :     CPLXMLNode *psSchemaNode = CPLGetXMLNode(psXSDTree, "=schema");
    1103         457 :     if (psSchemaNode == nullptr)
    1104             :     {
    1105           0 :         CPLDestroyXMLNode(psXSDTree);
    1106           0 :         return false;
    1107             :     }
    1108             : 
    1109             :     /* ==================================================================== */
    1110             :     /*      Process each include directive.                                 */
    1111             :     /* ==================================================================== */
    1112         457 :     CPLXMLSchemaResolveInclude(pszFile, psSchemaNode, bUseSchemaImports);
    1113             : 
    1114             :     // CPLSerializeXMLTreeToFile(psSchemaNode, "/vsistdout/");
    1115             : 
    1116         457 :     bFullyUnderstood = true;
    1117             : 
    1118             :     /* ==================================================================== */
    1119             :     /*      Process each feature class definition.                          */
    1120             :     /* ==================================================================== */
    1121         457 :     CPLXMLNode *psThis = psSchemaNode->psChild;
    1122             : 
    1123         457 :     std::set<std::pair<std::string, std::string>> oSetUniqueConstraints;
    1124             : 
    1125        4947 :     for (; psThis != nullptr; psThis = psThis->psNext)
    1126             :     {
    1127             :         /* --------------------------------------------------------------------
    1128             :          */
    1129             :         /*      Check for <xs:element> node. */
    1130             :         /* --------------------------------------------------------------------
    1131             :          */
    1132        4490 :         if (psThis->eType != CXT_Element || !EQUAL(psThis->pszValue, "element"))
    1133        3910 :             continue;
    1134             : 
    1135             :         /* --------------------------------------------------------------------
    1136             :          */
    1137             :         /*      Get name */
    1138             :         /* --------------------------------------------------------------------
    1139             :          */
    1140         744 :         const char *pszName = CPLGetXMLValue(psThis, "name", nullptr);
    1141         744 :         if (pszName == nullptr)
    1142             :         {
    1143           0 :             continue;
    1144             :         }
    1145             : 
    1146             :         /* --------------------------------------------------------------------
    1147             :          */
    1148             :         /*      Check the substitution group. */
    1149             :         /* --------------------------------------------------------------------
    1150             :          */
    1151             :         const char *pszSubGroup =
    1152         744 :             StripNS(CPLGetXMLValue(psThis, "substitutionGroup", ""));
    1153             : 
    1154         744 :         if (EQUAL(pszName, "FeatureCollection") &&
    1155         144 :             (EQUAL(pszSubGroup, "_FeatureCollection") ||
    1156         104 :              EQUAL(pszSubGroup, "_GML") ||
    1157          94 :              EQUAL(pszSubGroup, "AbstractFeature")))
    1158             :         {
    1159         144 :             oSetUniqueConstraints = GetUniqueConstraints(psThis);
    1160         144 :             continue;
    1161             :         }
    1162             : 
    1163             :         // AbstractFeature used by GML 3.2.
    1164         600 :         if (!EQUAL(pszSubGroup, "_Feature") &&
    1165         138 :             !EQUAL(pszSubGroup, "AbstractFeature"))
    1166             :         {
    1167           0 :             continue;
    1168             :         }
    1169             : 
    1170             :         /* --------------------------------------------------------------------
    1171             :          */
    1172             :         /*      Get type and verify relationship with name. */
    1173             :         /* --------------------------------------------------------------------
    1174             :          */
    1175         600 :         const char *pszType = CPLGetXMLValue(psThis, "type", nullptr);
    1176         600 :         if (pszType == nullptr)
    1177             :         {
    1178          14 :             CPLXMLNode *psComplexType = CPLGetXMLNode(psThis, "complexType");
    1179          14 :             if (psComplexType)
    1180             :             {
    1181             :                 GMLFeatureClass *poClass =
    1182          14 :                     GMLParseFeatureType(psSchemaNode, pszName, psComplexType);
    1183          14 :                 if (poClass)
    1184          13 :                     aosClasses.push_back(poClass);
    1185             :                 else
    1186           1 :                     bFullyUnderstood = false;
    1187             :             }
    1188          14 :             continue;
    1189             :         }
    1190         586 :         if (strstr(pszType, ":") != nullptr)
    1191         578 :             pszType = strstr(pszType, ":") + 1;
    1192         586 :         if (EQUAL(pszType, pszName))
    1193             :         {
    1194             :             // A few WFS servers return a type name which is the element name
    1195             :             // without any _Type or Type suffix
    1196             :             // e.g.:
    1197             :             // http://apollo.erdas.com/erdas-apollo/vector/Cherokee?SERVICE=WFS&VERSION=1.0.0&REQUEST=DescribeFeatureType&TYPENAME=iwfs:Air
    1198             :             // */
    1199             : 
    1200             :             // TODO(schwehr): What was supposed to go here?
    1201             :         }
    1202             : 
    1203             :         // <element name="RekisteriyksikonPalstanTietoja"
    1204             :         // type="ktjkiiwfs:PalstanTietojaType" substitutionGroup="gml:_Feature"
    1205             :         // />
    1206         586 :         else if (strlen(pszType) > 4 &&
    1207         586 :                  strcmp(pszType + strlen(pszType) - 4, "Type") == 0 &&
    1208         586 :                  strlen(pszName) > strlen(pszType) - 4 &&
    1209           0 :                  strncmp(pszName + strlen(pszName) - (strlen(pszType) - 4),
    1210           0 :                          pszType, strlen(pszType) - 4) == 0)
    1211             :         {
    1212             :         }
    1213             : 
    1214         586 :         else if (!EQUALN(pszType, pszName, strlen(pszName)) ||
    1215         586 :                  !(EQUAL(pszType + strlen(pszName), "_Type") ||
    1216         404 :                    EQUAL(pszType + strlen(pszName), "Type") ||
    1217           4 :                    EQUAL(pszType + strlen(pszName), "FeatureType")))
    1218             :         {
    1219           3 :             continue;
    1220             :         }
    1221             : 
    1222             :         // CanVec .xsd contains weird types that are not used in the related
    1223             :         // GML.
    1224         583 :         if (STARTS_WITH(pszName, "XyZz") || STARTS_WITH(pszName, "XyZ1") ||
    1225         580 :             STARTS_WITH(pszName, "XyZ2"))
    1226           3 :             continue;
    1227             : 
    1228             :         GMLFeatureClass *poClass =
    1229         580 :             GMLParseFeatureType(psSchemaNode, pszName, pszType);
    1230         580 :         if (poClass)
    1231         572 :             aosClasses.push_back(poClass);
    1232             :         else
    1233           8 :             bFullyUnderstood = false;
    1234             :     }
    1235             : 
    1236         457 :     CPLDestroyXMLNode(psXSDTree);
    1237             : 
    1238             :     // Attach unique constraints to fields
    1239         457 :     for (const auto &typeFieldPair : oSetUniqueConstraints)
    1240             :     {
    1241           0 :         for (const auto *poClass : aosClasses)
    1242             :         {
    1243           0 :             if (poClass->GetName() == typeFieldPair.first)
    1244             :             {
    1245             :                 auto poProperty =
    1246           0 :                     poClass->GetProperty(typeFieldPair.second.c_str());
    1247           0 :                 if (poProperty)
    1248             :                 {
    1249           0 :                     poProperty->SetUnique(true);
    1250             :                 }
    1251           0 :                 break;
    1252             :             }
    1253             :         }
    1254             :     }
    1255             : 
    1256         457 :     return !aosClasses.empty();
    1257             : }

Generated by: LCOV version 1.14