Line data Source code
1 : /******************************************************************************
2 : * Project: OGR
3 : * Purpose: OGRGMLASDriver implementation
4 : * Author: Even Rouault, <even dot rouault at spatialys dot com>
5 : *
6 : * Initial development funded by the European Earth observation programme
7 : * Copernicus
8 : *
9 : ******************************************************************************
10 : * Copyright (c) 2016, Even Rouault, <even dot rouault at spatialys dot com>
11 : *
12 : * SPDX-License-Identifier: MIT
13 : ****************************************************************************/
14 :
15 : #ifndef OGR_GMLAS_INCLUDED
16 : #define OGR_GMLAS_INCLUDED
17 :
18 : #include "xercesc_headers.h"
19 : #include "ogr_xerces.h"
20 :
21 : #include "cpl_vsi_virtual.h"
22 : #include "gdal_priv.h"
23 : #include "ogrsf_frmts.h"
24 :
25 : #include "ogr_gmlas_consts.h"
26 :
27 : #include <list>
28 : #include <set>
29 : #include <map>
30 : #include <vector>
31 :
32 : typedef std::pair<CPLString, CPLString> PairURIFilename;
33 :
34 : typedef enum
35 : {
36 : GMLAS_SWAP_AUTO,
37 : GMLAS_SWAP_YES,
38 : GMLAS_SWAP_NO,
39 : } GMLASSwapCoordinatesEnum;
40 :
41 : GDALDataset *OGRGMLASDriverCreateCopy(const char *pszFilename,
42 : GDALDataset *poSrcDS, int /*bStrict*/,
43 : char **papszOptions,
44 : GDALProgressFunc pfnProgress,
45 : void *pProgressData);
46 :
47 : /************************************************************************/
48 : /* IGMLASInputSourceClosing */
49 : /************************************************************************/
50 :
51 : class IGMLASInputSourceClosing
52 : {
53 : public:
54 235 : virtual ~IGMLASInputSourceClosing()
55 235 : {
56 235 : }
57 :
58 : virtual void notifyClosing(const CPLString &osFilename) = 0;
59 : };
60 :
61 : /************************************************************************/
62 : /* GMLASResourceCache */
63 : /************************************************************************/
64 :
65 : class GMLASResourceCache
66 : {
67 : protected:
68 : bool m_bHasCheckedCacheDirectory = false;
69 : std::string m_osCacheDirectory{};
70 : bool m_bRefresh = false;
71 : bool m_bAllowDownload = false;
72 : std::set<std::string> m_aoSetRefreshedFiles{};
73 :
74 : static bool
75 : RecursivelyCreateDirectoryIfNeeded(const std::string &osDirname);
76 : bool RecursivelyCreateDirectoryIfNeeded();
77 :
78 : std::string GetCachedFilename(const std::string &osResource);
79 :
80 : public:
81 : void SetCacheDirectory(const std::string &osCacheDirectory);
82 :
83 361 : void SetRefreshMode(bool bRefresh)
84 : {
85 361 : m_bRefresh = bRefresh;
86 361 : }
87 :
88 188 : void SetAllowDownload(bool bVal)
89 : {
90 188 : m_bAllowDownload = bVal;
91 188 : }
92 : };
93 :
94 : /************************************************************************/
95 : /* GMLASXSDCache */
96 : /************************************************************************/
97 :
98 : class GMLASXSDCache final : public GMLASResourceCache
99 : {
100 : bool CacheAllGML321();
101 : bool CacheAllISO20070417();
102 :
103 : public:
104 : VSILFILE *Open(const std::string &osResource, const std::string &osBasePath,
105 : std::string &osOutFilename);
106 : };
107 :
108 : /************************************************************************/
109 : /* GMLASBaseEntityResolver */
110 : /************************************************************************/
111 :
112 : class GMLASBaseEntityResolver : public EntityResolver,
113 : public IGMLASInputSourceClosing
114 : {
115 : protected:
116 : std::vector<CPLString> m_aosPathStack{};
117 : GMLASXSDCache &m_oCache;
118 : CPLString m_osGMLVersionFound{};
119 : std::set<CPLString> m_oSetSchemaURLs{};
120 : bool m_bFoundNonOfficialGMLSchemaLocation = false;
121 : bool m_bSubstituteWithOGCSchemaLocation = false;
122 :
123 : public:
124 : GMLASBaseEntityResolver(const CPLString &osBasePath, GMLASXSDCache &oCache);
125 : virtual ~GMLASBaseEntityResolver();
126 :
127 : void SetBasePath(const CPLString &osBasePath);
128 :
129 174 : const CPLString &GetGMLVersionFound() const
130 : {
131 174 : return m_osGMLVersionFound;
132 : }
133 :
134 174 : const std::set<CPLString> &GetSchemaURLS() const
135 : {
136 174 : return m_oSetSchemaURLs;
137 : }
138 :
139 2 : void SetSubstituteWithOGCSchemaLocation(bool b)
140 : {
141 2 : m_bSubstituteWithOGCSchemaLocation = b;
142 2 : }
143 :
144 2 : bool GetFoundNonOfficialGMLSchemaLocation() const
145 : {
146 2 : return m_bFoundNonOfficialGMLSchemaLocation;
147 : }
148 :
149 : virtual void notifyClosing(const CPLString &osFilename) override;
150 : virtual InputSource *resolveEntity(const XMLCh *const publicId,
151 : const XMLCh *const systemId) override;
152 :
153 : virtual void
154 : DoExtraSchemaProcessing(const CPLString &osFilename,
155 : const std::shared_ptr<VSIVirtualHandle> &fp);
156 : };
157 :
158 : /************************************************************************/
159 : /* GMLASInputSource */
160 : /************************************************************************/
161 :
162 : class GMLASInputSource final : public InputSource
163 : {
164 : std::shared_ptr<VSIVirtualHandle> m_fp{};
165 : int m_nCounter = 0;
166 : int *m_pnCounter = nullptr;
167 : CPLString m_osFilename{};
168 : IGMLASInputSourceClosing *m_cbk = nullptr;
169 :
170 : CPL_DISALLOW_COPY_ASSIGN(GMLASInputSource)
171 :
172 : public:
173 : GMLASInputSource(
174 : const char *pszFilename, const std::shared_ptr<VSIVirtualHandle> &fp,
175 : MemoryManager *const manager = XMLPlatformUtils::fgMemoryManager);
176 : virtual ~GMLASInputSource();
177 :
178 : virtual BinInputStream *makeStream() const override;
179 :
180 : void SetClosingCallback(IGMLASInputSourceClosing *cbk);
181 : };
182 :
183 : /************************************************************************/
184 : /* GMLASErrorHandler */
185 : /************************************************************************/
186 :
187 : class GMLASErrorHandler : public ErrorHandler
188 : {
189 : public:
190 2725 : GMLASErrorHandler() = default;
191 :
192 1645 : void SetSchemaFullCheckingEnabled(bool b)
193 : {
194 1645 : m_bSchemaFullChecking = b;
195 1645 : }
196 :
197 1645 : void SetHandleMultipleImportsEnabled(bool b)
198 : {
199 1645 : m_bHandleMultipleImports = b;
200 1645 : }
201 :
202 259 : void SetHideGMLTypeNotFound(bool b)
203 : {
204 259 : m_bHideGMLTypeNotFound = b;
205 259 : }
206 :
207 6 : const std::string &GetGMLTypeNotFoundError() const
208 : {
209 6 : return m_osGMLTypeNotFoundError;
210 : }
211 :
212 258 : bool hasFailed() const
213 : {
214 258 : return m_bFailed;
215 : }
216 :
217 : virtual void warning(const SAXParseException &e) override;
218 : virtual void error(const SAXParseException &e) override;
219 : virtual void fatalError(const SAXParseException &e) override;
220 :
221 2311 : virtual void resetErrors() override
222 : {
223 2311 : m_bFailed = false;
224 2311 : }
225 :
226 : private:
227 : bool m_bFailed = false;
228 : bool m_bSchemaFullChecking = false;
229 : bool m_bHandleMultipleImports = false;
230 : bool m_bHideGMLTypeNotFound = false;
231 : std::string m_osGMLTypeNotFoundError{};
232 :
233 : void handle(const SAXParseException &e, CPLErr eErr);
234 : };
235 :
236 : /************************************************************************/
237 : /* GMLASXLinkResolutionConf */
238 : /************************************************************************/
239 :
240 : class GMLASXLinkResolutionConf
241 : {
242 : public:
243 : /* See data/gmlasconf.xsd for docomentation of the fields */
244 :
245 : typedef enum
246 : {
247 : RawContent,
248 : FieldsFromXPath
249 : } ResolutionMode;
250 :
251 : int m_nTimeOut = 0;
252 :
253 : int m_nMaxFileSize = MAX_FILE_SIZE_DEFAULT;
254 :
255 : int m_nMaxGlobalResolutionTime = 0;
256 :
257 : CPLString m_osProxyServerPort{};
258 :
259 : CPLString m_osProxyUserPassword{};
260 :
261 : CPLString m_osProxyAuth{};
262 :
263 : CPLString m_osCacheDirectory{};
264 :
265 : bool m_bDefaultResolutionEnabled = DEFAULT_RESOLUTION_ENABLED_DEFAULT;
266 :
267 : bool m_bDefaultAllowRemoteDownload = ALLOW_REMOTE_DOWNLOAD_DEFAULT;
268 :
269 : ResolutionMode m_eDefaultResolutionMode = RawContent;
270 :
271 : int m_nDefaultResolutionDepth = 1;
272 :
273 : bool m_bDefaultCacheResults = CACHE_RESULTS_DEFAULT;
274 :
275 : bool m_bResolveInternalXLinks = INTERNAL_XLINK_RESOLUTION_DEFAULT;
276 :
277 : struct URLSpecificResolution
278 : {
279 : CPLString m_osURLPrefix{};
280 :
281 : std::vector<std::pair<CPLString, CPLString>>
282 : m_aosNameValueHTTPHeaders{};
283 :
284 : bool m_bAllowRemoteDownload = false;
285 :
286 : ResolutionMode m_eResolutionMode = RawContent;
287 :
288 : int m_nResolutionDepth = 1;
289 :
290 : bool m_bCacheResults = false;
291 :
292 : struct XPathDerivedField
293 : {
294 : CPLString m_osName{};
295 :
296 : CPLString m_osType{};
297 :
298 : CPLString m_osXPath{};
299 : };
300 :
301 : std::vector<XPathDerivedField> m_aoFields{};
302 : };
303 :
304 : std::vector<URLSpecificResolution> m_aoURLSpecificRules{};
305 :
306 : GMLASXLinkResolutionConf() = default;
307 :
308 : bool LoadFromXML(CPLXMLNode *psRoot);
309 : };
310 :
311 : /************************************************************************/
312 : /* GMLASConfiguration */
313 : /************************************************************************/
314 :
315 : class GMLASConfiguration
316 : {
317 : public:
318 : /** Whether remote schemas are allowed to be download. */
319 : bool m_bAllowRemoteSchemaDownload = ALLOW_REMOTE_SCHEMA_DOWNLOAD_DEFAULT;
320 :
321 : /** Whether a ogr_pkid attribute should always be generated. */
322 : bool m_bAlwaysGenerateOGRId = ALWAYS_GENERATE_OGR_ID_DEFAULT;
323 :
324 : /** Whether to remove layers found to be unused in initial scan pass */
325 : bool m_bRemoveUnusedLayers = REMOVE_UNUSED_LAYERS_DEFAULT;
326 :
327 : /** Whether to remove fields found to be unused in initial scan pass */
328 : bool m_bRemoveUnusedFields = REMOVE_UNUSED_FIELDS_DEFAULT;
329 :
330 : /** Whether repeated strings, integers, reals should be in corresponding
331 : OGR array types. */
332 : bool m_bUseArrays = USE_ARRAYS_DEFAULT;
333 :
334 : /** Whether OGR field null state should be used. */
335 : bool m_bUseNullState = USE_NULL_STATE_DEFAULT;
336 :
337 : /** Whether geometries should be stored as XML in a OGR string field. */
338 : bool m_bIncludeGeometryXML = INCLUDE_GEOMETRY_XML_DEFAULT;
339 :
340 : /** Whether, when dealing with schemas that import the
341 : GML namespace, and that at least one of them has
342 : elements that derive from gml:_Feature or
343 : gml:AbstractFeatureonly, only such elements should be
344 : instantiated as OGR layers, during the first pass that
345 : iterates over top level elements of the imported
346 : schemas. */
347 : bool m_bInstantiateGMLFeaturesOnly = INSTANTIATE_GML_FEATURES_ONLY_DEFAULT;
348 :
349 : /** Maximum length of layer and field identifiers*/
350 : int m_nIdentifierMaxLength = 0;
351 :
352 : /** Whether case insensitive comparison should be used for identifier
353 : * equality testing */
354 : bool m_bCaseInsensitiveIdentifier = CASE_INSENSITIVE_IDENTIFIER_DEFAULT;
355 :
356 : /** Whether to launder identifiers like postgresql does */
357 : bool m_bPGIdentifierLaundering = PG_IDENTIFIER_LAUNDERING_DEFAULT;
358 :
359 : /* Maximum number of fields in an element considered for flattening. */
360 : int m_nMaximumFieldsForFlattening = MAXIMUM_FIELDS_FLATTENING_DEFAULT;
361 :
362 : /** Whether remote XSD schemas should be locally cached. */
363 : bool m_bAllowXSDCache = ALLOW_XSD_CACHE_DEFAULT;
364 :
365 : /** Cache directory for cached XSD schemas. */
366 : CPLString m_osXSDCacheDirectory{};
367 :
368 : /** Whether to enable schema full checking. */
369 : bool m_bSchemaFullChecking = SCHEMA_FULL_CHECKING_DEFAULT;
370 :
371 : /** Whether to allow multiple imports of the same namespace. */
372 : bool m_bHandleMultipleImports = HANDLE_MULTIPLE_IMPORTS_DEFAULT;
373 :
374 : /** Whether validation of document against schema should be done. */
375 : bool m_bValidate = VALIDATE_DEFAULT;
376 :
377 : /** Whether a validation error should prevent dataset opening. */
378 : bool m_bFailIfValidationError = FAIL_IF_VALIDATION_ERROR_DEFAULT;
379 :
380 : /** Whether technical layers should be exposed. */
381 : bool m_bExposeMetadataLayers = WARN_IF_EXCLUDED_XPATH_FOUND_DEFAULT;
382 :
383 : /** For flatening rules, map prefix namespace to its URI */
384 : std::map<CPLString, CPLString> m_oMapPrefixToURIFlatteningRules{};
385 :
386 : std::vector<CPLString> m_osForcedFlattenedXPath{};
387 :
388 : std::vector<CPLString> m_osDisabledFlattenedXPath{};
389 :
390 : enum SWEActivationMode
391 : {
392 : SWE_ACTIVATE_IF_NAMESPACE_FOUND,
393 : SWE_ACTIVATE_TRUE,
394 : SWE_ACTIVATE_FALSE
395 : };
396 :
397 : /** If and when activate SWE special processings */
398 : SWEActivationMode m_eSWEActivationMode = SWE_ACTIVATE_IF_NAMESPACE_FOUND;
399 :
400 : /** If enabling swe:DataRecord parsing */
401 : bool m_bSWEProcessDataRecord = SWE_PROCESS_DATA_RECORD_DEFAULT;
402 :
403 : /** If enabling swe:DataArray parsing */
404 : bool m_bSWEProcessDataArray = SWE_PROCESS_DATA_ARRAY_DEFAULT;
405 :
406 : /** For ignored xpaths, map prefix namespace to its URI */
407 : std::map<CPLString, CPLString> m_oMapPrefixToURIIgnoredXPaths{};
408 :
409 : /** Ignored xpaths */
410 : std::vector<CPLString> m_aosIgnoredXPaths{};
411 :
412 : /** For type constraints, map prefix namespace to its URI */
413 : std::map<CPLString, CPLString> m_oMapPrefixToURITypeConstraints{};
414 :
415 : /** Map an XPath to a list of potential types for its children */
416 : std::map<CPLString, std::vector<CPLString>>
417 : m_oMapChildrenElementsConstraints{};
418 :
419 : /* Beginning of Writer config */
420 :
421 : /** Number of spaces for indentation */
422 : int m_nIndentSize = INDENT_SIZE_DEFAULT;
423 :
424 : CPLString m_osComment{};
425 :
426 : /** End of line format: "CRLF" or "LR" */
427 : CPLString m_osLineFormat{};
428 :
429 : /** "SHORT", "OGC_URN" or "OGC_URL" */
430 : CPLString m_osSRSNameFormat = szSRSNAME_DEFAULT;
431 :
432 : /** "WFS2_FEATURECOLLECTION" or "GMLAS_FEATURECOLLECTION" */
433 : CPLString m_osWrapping = szWFS2_FEATURECOLLECTION;
434 :
435 : /** XML datetime or empty for current time */
436 : CPLString m_osTimestamp{};
437 :
438 : /** Path or URL to OGC WFS 2.0 schema. */
439 : CPLString m_osWFS20SchemaLocation = szWFS20_SCHEMALOCATION;
440 :
441 : /* End of Writer config */
442 :
443 : /** Whether a warning should be emitted when an element or attribute is
444 : found in the document parsed, but ignored because of the ignored
445 : XPath defined. */
446 : std::map<CPLString, bool> m_oMapIgnoredXPathToWarn{};
447 :
448 : GMLASXLinkResolutionConf m_oXLinkResolution{};
449 :
450 : GMLASConfiguration() = default;
451 :
452 : bool Load(const char *pszFilename);
453 : void Finalize();
454 :
455 : static CPLString GetBaseCacheDirectory();
456 :
457 : static std::string GetDefaultConfFile(bool &bUnlinkAfterUse);
458 : };
459 :
460 : /************************************************************************/
461 : /* GMLASXLinkResolver */
462 : /************************************************************************/
463 :
464 : class GMLASXLinkResolver final : public GMLASResourceCache
465 : {
466 : GMLASXLinkResolutionConf m_oConf{};
467 : int m_nGlobalResolutionTime = 0;
468 :
469 : std::map<CPLString, CPLString> m_oMapURLToContent{};
470 : std::map<size_t, std::vector<CPLString>> m_oMapFileSizeToURLs{};
471 : size_t m_nMaxRAMCacheSize = 0;
472 : size_t m_nCurrentRAMCacheSize = 0;
473 :
474 : CPLString FetchRawContent(const CPLString &osURL, const char *pszHeaders);
475 :
476 : CPLString GetRawContent(const CPLString &osURL, const char *pszHeaders,
477 : bool bAllowRemoteDownload, bool bCacheResults);
478 :
479 : public:
480 : GMLASXLinkResolver();
481 :
482 : void SetConf(const GMLASXLinkResolutionConf &oConf);
483 :
484 19527 : const GMLASXLinkResolutionConf &GetConf() const
485 : {
486 19527 : return m_oConf;
487 : }
488 :
489 : bool IsRawContentResolutionEnabled() const;
490 : int GetMatchingResolutionRule(const CPLString &osURL) const;
491 : CPLString GetRawContent(const CPLString &osURL);
492 : CPLString GetRawContentForRule(const CPLString &osURL, int nIdxRule);
493 : };
494 :
495 : /************************************************************************/
496 : /* GMLASXPathMatcher */
497 : /************************************************************************/
498 :
499 : /** Object to compares a user provided XPath against a set of test XPaths */
500 : class GMLASXPathMatcher
501 : {
502 : struct XPathComponent
503 : {
504 : CPLString m_osValue{};
505 : bool m_bDirectChild = false;
506 : };
507 :
508 : /** For reference xpaths, map prefix namespace to its URI */
509 : std::map<CPLString, CPLString> m_oMapPrefixToURIReferenceXPaths{};
510 :
511 : /** Reference xpaths */
512 : std::vector<CPLString> m_aosReferenceXPathsUncompiled{};
513 :
514 : /** Reference xpaths "compiled" */
515 : std::vector<std::vector<XPathComponent>> m_aosReferenceXPaths{};
516 :
517 : static bool MatchesRefXPath(const CPLString &osXPath,
518 : const std::vector<XPathComponent> &oRefXPath);
519 :
520 : public:
521 : void SetRefXPaths(
522 : const std::map<CPLString, CPLString> &oMapPrefixToURIReferenceXPaths,
523 : const std::vector<CPLString> &aosReferenceXPaths);
524 :
525 : void SetDocumentMapURIToPrefix(
526 : const std::map<CPLString, CPLString> &oMapURIToPrefix);
527 :
528 : /** Return true if osXPath matches one of the XPath of
529 : m_aosReferenceXPaths */
530 : bool MatchesRefXPath(const CPLString &osXPath,
531 : CPLString &osOutMatchedXPath) const;
532 :
533 6 : const std::map<CPLString, CPLString> &GetMapPrefixToURI() const
534 : {
535 6 : return m_oMapPrefixToURIReferenceXPaths;
536 : }
537 : };
538 :
539 : /************************************************************************/
540 : /* GMLASFieldType */
541 : /************************************************************************/
542 :
543 : /** Enumeration for XML primitive types */
544 : typedef enum
545 : {
546 : GMLAS_FT_STRING,
547 : GMLAS_FT_ID,
548 : GMLAS_FT_BOOLEAN,
549 : GMLAS_FT_SHORT,
550 : GMLAS_FT_INT32,
551 : GMLAS_FT_INT64,
552 : GMLAS_FT_FLOAT,
553 : GMLAS_FT_DOUBLE,
554 : GMLAS_FT_DECIMAL,
555 : GMLAS_FT_DATE,
556 : GMLAS_FT_GYEAR,
557 : GMLAS_FT_GYEAR_MONTH,
558 : GMLAS_FT_TIME,
559 : GMLAS_FT_DATETIME,
560 : GMLAS_FT_BASE64BINARY,
561 : GMLAS_FT_HEXBINARY,
562 : GMLAS_FT_ANYURI,
563 : GMLAS_FT_ANYTYPE,
564 : GMLAS_FT_ANYSIMPLETYPE,
565 : GMLAS_FT_GEOMETRY, // this one isn't a XML primitive type.
566 : } GMLASFieldType;
567 :
568 : /************************************************************************/
569 : /* GMLASField */
570 : /************************************************************************/
571 :
572 : class GMLASField
573 : {
574 : public:
575 : typedef enum
576 : {
577 : /** Field that is going to be instantiated as a OGR field */
578 : REGULAR,
579 :
580 : /** Non-instanciable field. The corresponding element to the XPath
581 : is stored in a child layer that will reference back to the
582 : main layer. */
583 : PATH_TO_CHILD_ELEMENT_NO_LINK,
584 :
585 : /** Field that will store the PKID of a child element */
586 : PATH_TO_CHILD_ELEMENT_WITH_LINK,
587 :
588 : /** Non-instanciable field. The corresponding element to the XPath
589 : is stored in a child layer. And the link between both will be
590 : done through a junction table. */
591 : PATH_TO_CHILD_ELEMENT_WITH_JUNCTION_TABLE,
592 :
593 : /** Non-instanciable field. Corresponds to a group of an element. */
594 : GROUP
595 : } Category;
596 :
597 : private:
598 : CPLString m_osName{}; /**< Field name */
599 : GMLASFieldType m_eType = GMLAS_FT_STRING; /**< Field type */
600 : OGRwkbGeometryType m_eGeomType = wkbNone; /**< Field geometry type */
601 : CPLString m_osTypeName{}; /**< Original XSD type */
602 : int m_nWidth = 0; /**< Field width */
603 : bool m_bNotNullable = false; /**< If the field is not nullable */
604 :
605 : /** If the field is an array (from OGR types point of view) */
606 : bool m_bArray = false;
607 : bool m_bList = false; /**< If the field is a list (a xs:list) */
608 :
609 : /** Category of the field. */
610 : Category m_eCategory = REGULAR;
611 :
612 : /** XPath of the field. */
613 : CPLString m_osXPath{};
614 :
615 : /** Set of XPath that are linked to this field.
616 : This is used for cases where a gml:AbstractGeometry element is
617 : referenced. In which case all possible realizations of this
618 : element are listed. Will be used with eType == GMLAS_FT_ANYTYPE
619 : to store XML blob on parsing. */
620 : std::vector<CPLString> m_aosXPath{};
621 :
622 : CPLString m_osFixedValue{}; /**< Value of fixed='' attribute */
623 : CPLString m_osDefaultValue{}; /**< Value of default='' attribute */
624 :
625 : /** Minimum number of occurrences. Might be -1 if unset */
626 : int m_nMinOccurs = -1;
627 :
628 : /** Maximum number of occurrences, or MAXOCCURS_UNLIMITED. Might be
629 : -1 if unset. */
630 : int m_nMaxOccurs = -1;
631 :
632 : /** For a PATH_TO_CHILD_ELEMENT_NO_LINK, whether maxOccurs>1 is on the
633 : sequence rather than on the element */
634 : bool m_bRepetitionOnSequence = false;
635 :
636 : /** In case of m_eType == GMLAS_FT_ANYTYPE whether the current element
637 : must be stored in the XML blob (if false, only its children) */
638 : bool m_bIncludeThisEltInBlob = false;
639 :
640 : /** Only used for PATH_TO_CHILD_ELEMENT_WITH_JUNCTION_TABLE. The XPath
641 : of the abstract element (the concrete XPath is in m_osXPath).
642 : e.g myns:mainElt/myns:subEltAbstract whereas the concrete XPath
643 : is myns:mainElt/myns:subEltRealization */
644 : CPLString m_osAbstractElementXPath{};
645 :
646 : /** Only used for PATH_TO_CHILD_ELEMENT_WITH_LINK and
647 : PATH_TO_CHILD_ELEMENT_WITH_JUNCTION_TABLE (and also for
648 : PATH_TO_CHILD_ELEMENT_NO_LINK and GROUP but for metadata layers only).
649 : The XPath of the child element. */
650 : CPLString m_osRelatedClassXPath{};
651 :
652 : /** Only use for PATH_TO_CHILD_ELEMENT_WITH_JUNCTION_TABLE. Name of
653 : the junction layer to consult for this field. Only used by
654 : writer code. */
655 : CPLString m_osJunctionLayer{};
656 :
657 : /** Dirty hack to register attributes with fixed values, despite being
658 : in the XPath ignored list. Needed to avoid warning when doing validation
659 : */
660 : bool m_bIgnored = false;
661 :
662 : /** Documentation from schema */
663 : CPLString m_osDoc{};
664 :
665 : /** For elements within xs:choice */
666 : bool m_bMayAppearOutOfOrder = false;
667 :
668 : public:
669 216619 : GMLASField() = default;
670 :
671 791639 : void SetName(const CPLString &osName)
672 : {
673 791639 : m_osName = osName;
674 791639 : }
675 :
676 : void SetType(GMLASFieldType eType, const char *pszTypeName);
677 :
678 371 : void SetGeomType(OGRwkbGeometryType eGeomType)
679 : {
680 371 : m_eGeomType = eGeomType;
681 371 : }
682 :
683 195025 : void SetWidth(int nWidth)
684 : {
685 195025 : m_nWidth = nWidth;
686 195025 : }
687 :
688 169791 : void SetNotNullable(bool bNotNullable)
689 : {
690 169791 : m_bNotNullable = bNotNullable;
691 169791 : }
692 :
693 4187 : void SetArray(bool bArray)
694 : {
695 4187 : m_bArray = bArray;
696 4187 : }
697 :
698 2460 : void SetList(bool bList)
699 : {
700 2460 : m_bList = bList;
701 2460 : }
702 :
703 215834 : void SetXPath(const CPLString &osXPath)
704 : {
705 215834 : m_osXPath = osXPath;
706 215834 : }
707 :
708 761 : void AddAlternateXPath(const CPLString &osXPath)
709 : {
710 761 : m_aosXPath.push_back(osXPath);
711 761 : }
712 :
713 569 : void SetFixedValue(const CPLString &osFixedValue)
714 : {
715 569 : m_osFixedValue = osFixedValue;
716 569 : }
717 :
718 910 : void SetDefaultValue(const CPLString &osDefaultValue)
719 : {
720 910 : m_osDefaultValue = osDefaultValue;
721 910 : }
722 :
723 21276 : void SetCategory(Category eCategory)
724 : {
725 21276 : m_eCategory = eCategory;
726 21276 : }
727 :
728 317422 : void SetMinOccurs(int nMinOccurs)
729 : {
730 317422 : m_nMinOccurs = nMinOccurs;
731 317422 : }
732 :
733 215750 : void SetMaxOccurs(int nMaxOccurs)
734 : {
735 215750 : m_nMaxOccurs = nMaxOccurs;
736 215750 : }
737 :
738 818 : void SetRepetitionOnSequence(bool b)
739 : {
740 818 : m_bRepetitionOnSequence = b;
741 818 : }
742 :
743 275 : void SetIncludeThisEltInBlob(bool b)
744 : {
745 275 : m_bIncludeThisEltInBlob = b;
746 275 : }
747 :
748 8939 : void SetAbstractElementXPath(const CPLString &osName)
749 : {
750 8939 : m_osAbstractElementXPath = osName;
751 8939 : }
752 :
753 20708 : void SetRelatedClassXPath(const CPLString &osName)
754 : {
755 20708 : m_osRelatedClassXPath = osName;
756 20708 : }
757 :
758 16 : void SetJunctionLayer(const CPLString &osName)
759 : {
760 16 : m_osJunctionLayer = osName;
761 16 : }
762 :
763 22 : void SetIgnored()
764 : {
765 22 : m_bIgnored = true;
766 22 : }
767 :
768 194369 : void SetDocumentation(const CPLString &osDoc)
769 : {
770 194369 : m_osDoc = osDoc;
771 194369 : }
772 :
773 149468 : void SetMayAppearOutOfOrder(bool b)
774 : {
775 149468 : m_bMayAppearOutOfOrder = b;
776 149468 : }
777 :
778 : static CPLString
779 22 : MakePKIDFieldXPathFromXLinkHrefXPath(const CPLString &osBaseXPath)
780 : {
781 44 : return "{" + osBaseXPath + "}_pkid";
782 : }
783 :
784 25 : static CPLString MakeXLinkRawContentFieldXPathFromXLinkHrefXPath(
785 : const CPLString &osBaseXPath)
786 : {
787 50 : return "{" + osBaseXPath + "}_rawcontent";
788 : }
789 :
790 : static CPLString
791 40 : MakeXLinkDerivedFieldXPathFromXLinkHrefXPath(const CPLString &osBaseXPath,
792 : const CPLString &osName)
793 : {
794 80 : return "{" + osBaseXPath + "}_derived_" + osName;
795 : }
796 :
797 1683030 : const CPLString &GetName() const
798 : {
799 1683030 : return m_osName;
800 : }
801 :
802 6479110 : const CPLString &GetXPath() const
803 : {
804 6479110 : return m_osXPath;
805 : }
806 :
807 188628 : const std::vector<CPLString> &GetAlternateXPaths() const
808 : {
809 188628 : return m_aosXPath;
810 : }
811 :
812 662614 : GMLASFieldType GetType() const
813 : {
814 662614 : return m_eType;
815 : }
816 :
817 371 : OGRwkbGeometryType GetGeomType() const
818 : {
819 371 : return m_eGeomType;
820 : }
821 :
822 364973 : const CPLString &GetTypeName() const
823 : {
824 364973 : return m_osTypeName;
825 : }
826 :
827 174207 : int GetWidth() const
828 : {
829 174207 : return m_nWidth;
830 : }
831 :
832 220414 : bool IsNotNullable() const
833 : {
834 220414 : return m_bNotNullable;
835 : }
836 :
837 340968 : bool IsArray() const
838 : {
839 340968 : return m_bArray;
840 : }
841 :
842 223891 : bool IsList() const
843 : {
844 223891 : return m_bList;
845 : }
846 :
847 1989860 : const CPLString &GetFixedValue() const
848 : {
849 1989860 : return m_osFixedValue;
850 : }
851 :
852 1846150 : const CPLString &GetDefaultValue() const
853 : {
854 1846150 : return m_osDefaultValue;
855 : }
856 :
857 2353300 : Category GetCategory() const
858 : {
859 2353300 : return m_eCategory;
860 : }
861 :
862 378164 : int GetMinOccurs() const
863 : {
864 378164 : return m_nMinOccurs;
865 : }
866 :
867 902551 : int GetMaxOccurs() const
868 : {
869 902551 : return m_nMaxOccurs;
870 : }
871 :
872 14524 : bool GetRepetitionOnSequence() const
873 : {
874 14524 : return m_bRepetitionOnSequence;
875 : }
876 :
877 5788 : bool GetIncludeThisEltInBlob() const
878 : {
879 5788 : return m_bIncludeThisEltInBlob;
880 : }
881 :
882 13505 : const CPLString &GetAbstractElementXPath() const
883 : {
884 13505 : return m_osAbstractElementXPath;
885 : }
886 :
887 24 : const CPLString &GetJunctionLayer() const
888 : {
889 24 : return m_osJunctionLayer;
890 : }
891 :
892 229789 : const CPLString &GetRelatedClassXPath() const
893 : {
894 229789 : return m_osRelatedClassXPath;
895 : }
896 :
897 188189 : bool IsIgnored() const
898 : {
899 188189 : return m_bIgnored;
900 : }
901 :
902 330201 : const CPLString &GetDocumentation() const
903 : {
904 330201 : return m_osDoc;
905 : }
906 :
907 4 : bool MayAppearOutOfOrder() const
908 : {
909 4 : return m_bMayAppearOutOfOrder;
910 : }
911 :
912 : static GMLASFieldType GetTypeFromString(const CPLString &osType);
913 : };
914 :
915 : /************************************************************************/
916 : /* GMLASFeatureClass */
917 : /************************************************************************/
918 :
919 : class GMLASFeatureClass
920 : {
921 : /** User facing name */
922 : CPLString m_osName{};
923 :
924 : /** XPath to the main element of the feature class */
925 : CPLString m_osXPath{};
926 :
927 : /** List of fields */
928 : std::vector<GMLASField> m_aoFields{};
929 :
930 : /** Child nested classes */
931 : std::vector<GMLASFeatureClass> m_aoNestedClasses{};
932 :
933 : /** Whether this layer corresponds to a (multiple instantiated) xs:group
934 : or a repeated sequence */
935 : bool m_bIsRepeatedSequence = false;
936 :
937 : /** Whether this is a repeated group. Should be set together with
938 : * m_bIsRepeatedSequence */
939 : bool m_bIsGroup = false;
940 :
941 : /** Only used for junction tables. The XPath to the parent table */
942 : CPLString m_osParentXPath{};
943 :
944 : /** Only used for junction tables. The XPath to the child table */
945 : CPLString m_osChildXPath{};
946 :
947 : /** Whether this corresponds to a top-level XSD element in the schema */
948 : bool m_bIsTopLevelElt = false;
949 :
950 : /** Documentation from schema */
951 : CPLString m_osDoc{};
952 :
953 : public:
954 42692 : GMLASFeatureClass() = default;
955 :
956 : void SetName(const CPLString &osName);
957 : void SetXPath(const CPLString &osXPath);
958 : void AddField(const GMLASField &oField);
959 : void PrependFields(const std::vector<GMLASField> &aoFields);
960 : void AppendFields(const std::vector<GMLASField> &aoFields);
961 : void AddNestedClass(const GMLASFeatureClass &oNestedClass);
962 :
963 4132 : void SetIsRepeatedSequence(bool bIsRepeatedSequence)
964 : {
965 4132 : m_bIsRepeatedSequence = bIsRepeatedSequence;
966 4132 : }
967 :
968 211 : void SetIsGroup(bool bIsGroup)
969 : {
970 211 : m_bIsGroup = bIsGroup;
971 211 : }
972 :
973 8939 : void SetParentXPath(const CPLString &osXPath)
974 : {
975 8939 : m_osParentXPath = osXPath;
976 8939 : }
977 :
978 8939 : void SetChildXPath(const CPLString &osXPath)
979 : {
980 8939 : m_osChildXPath = osXPath;
981 8939 : }
982 :
983 3139 : void SetIsTopLevelElt(bool bIsTopLevelElt)
984 : {
985 3139 : m_bIsTopLevelElt = bIsTopLevelElt;
986 3139 : }
987 :
988 30613 : void SetDocumentation(const CPLString &osDoc)
989 : {
990 30613 : m_osDoc = osDoc;
991 30613 : }
992 :
993 85874 : const CPLString &GetName() const
994 : {
995 85874 : return m_osName;
996 : }
997 :
998 9260040 : const CPLString &GetXPath() const
999 : {
1000 9260040 : return m_osXPath;
1001 : }
1002 :
1003 327814 : const std::vector<GMLASField> &GetFields() const
1004 : {
1005 327814 : return m_aoFields;
1006 : }
1007 :
1008 62830 : std::vector<GMLASField> &GetFields()
1009 : {
1010 62830 : return m_aoFields;
1011 : }
1012 :
1013 15417 : const std::vector<GMLASFeatureClass> &GetNestedClasses() const
1014 : {
1015 15417 : return m_aoNestedClasses;
1016 : }
1017 :
1018 48442 : std::vector<GMLASFeatureClass> &GetNestedClasses()
1019 : {
1020 48442 : return m_aoNestedClasses;
1021 : }
1022 :
1023 8506710 : bool IsRepeatedSequence() const
1024 : {
1025 8506710 : return m_bIsRepeatedSequence;
1026 : }
1027 :
1028 4223360 : bool IsGroup() const
1029 : {
1030 4223360 : return m_bIsGroup;
1031 : }
1032 :
1033 70405 : const CPLString &GetParentXPath() const
1034 : {
1035 70405 : return m_osParentXPath;
1036 : }
1037 :
1038 : const CPLString &GetChildXPath() const
1039 : {
1040 : return m_osChildXPath;
1041 : }
1042 :
1043 6478 : bool IsTopLevelElt() const
1044 : {
1045 6478 : return m_bIsTopLevelElt;
1046 : }
1047 :
1048 11614 : const CPLString &GetDocumentation() const
1049 : {
1050 11614 : return m_osDoc;
1051 : }
1052 : };
1053 :
1054 : /************************************************************************/
1055 : /* GMLASSchemaAnalyzer */
1056 : /************************************************************************/
1057 :
1058 : class GMLASSchemaAnalyzer
1059 : {
1060 : GMLASXPathMatcher &m_oIgnoredXPathMatcher;
1061 :
1062 : GMLASXPathMatcher &m_oChildrenElementsConstraintsXPathMatcher;
1063 :
1064 : GMLASXPathMatcher &m_oForcedFlattenedXPathMatcher;
1065 :
1066 : GMLASXPathMatcher &m_oDisabledFlattenedXPathMatcher;
1067 :
1068 : std::map<CPLString, std::vector<CPLString>>
1069 : m_oMapChildrenElementsConstraints{};
1070 :
1071 : /** Whether repeated strings, integers, reals should be in corresponding
1072 : OGR array types. */
1073 : bool m_bUseArrays = true;
1074 :
1075 : /** Whether OGR field null state should be used. */
1076 : bool m_bUseNullState = false;
1077 :
1078 : /** Whether, when dealing with schemas that import the
1079 : GML namespace, and that at least one of them has
1080 : elements that derive from gml:_Feature or
1081 : gml:AbstractFeatureonly, only such elements should be
1082 : instantiated as OGR layers, during the first pass that
1083 : iterates over top level elements of the imported
1084 : schemas. */
1085 : bool m_bInstantiateGMLFeaturesOnly = true;
1086 :
1087 : /** Vector of feature classes */
1088 : std::vector<GMLASFeatureClass> m_aoClasses{};
1089 :
1090 : /** Map from a namespace URI to the corresponding prefix */
1091 : std::map<CPLString, CPLString> m_oMapURIToPrefix{};
1092 :
1093 : /** Map element XPath to its XSElementDeclaration* */
1094 : std::map<CPLString, XSElementDeclaration *> m_oMapXPathToEltDecl{};
1095 :
1096 : typedef std::map<XSElementDeclaration *,
1097 : std::vector<XSElementDeclaration *>>
1098 : tMapParentEltToChildElt;
1099 : /** Map from a base/parent element to a vector of derived/children
1100 : elements that are substitutionGroup of it. The map only
1101 : addresses the direct derived types, and not the 2nd level or more
1102 : derived ones. For that recursion in the map must be used.*/
1103 : tMapParentEltToChildElt m_oMapParentEltToChildElt{};
1104 :
1105 : /** Map from a XSModelGroup* object to the name of its group definition. */
1106 : std::map<XSModelGroup *, XSModelGroupDefinition *> m_oMapModelGroupToMGD{};
1107 :
1108 : /** Map from (non namespace prefixed) element names to the number of
1109 : elements that share the same namespace (in different namespaces) */
1110 : std::map<CPLString, int> m_oMapEltNamesToInstanceCount{};
1111 :
1112 : /** Set of elements that match a OGR layer */
1113 : std::set<XSElementDeclaration *> m_oSetEltsForTopClass{};
1114 :
1115 : /** Set of elements that are simple enough to be inlined whenever they
1116 : are referenced with cardinality 1. The use case if base:identifier
1117 : used by Inspire schemas. */
1118 : std::set<XSElementDeclaration *> m_oSetSimpleEnoughElts{};
1119 :
1120 : /** Maximum length of layer and field identifiers*/
1121 : int m_nIdentifierMaxLength = 0;
1122 :
1123 : /** Whether case insensitive comparison should be used for identifier
1124 : * equality testing */
1125 : bool m_bCaseInsensitiveIdentifier = CASE_INSENSITIVE_IDENTIFIER_DEFAULT;
1126 :
1127 : /** Whether to launder identifiers like postgresql does */
1128 : bool m_bPGIdentifierLaundering = PG_IDENTIFIER_LAUNDERING_DEFAULT;
1129 :
1130 : /* Maximum number of fields in an element considered for flattening. */
1131 : int m_nMaximumFieldsForFlattening = MAXIMUM_FIELDS_FLATTENING_DEFAULT;
1132 :
1133 : /** GML version found: 2.1.1, 3.1.1 or 3.2.1 or empty*/
1134 : CPLString m_osGMLVersionFound{};
1135 :
1136 : /** Set of schemas opened */
1137 : std::set<CPLString> m_oSetSchemaURLs{};
1138 :
1139 : /** Map from namespace URI to namespace prefix coming from the
1140 : * examination of xmlns:foo=bar attributes of the top element of the
1141 : * GML document */
1142 : std::map<CPLString, CPLString> m_oMapDocNSURIToPrefix{};
1143 :
1144 : bool m_bAlwaysGenerateOGRId = ALWAYS_GENERATE_OGR_ID_DEFAULT;
1145 :
1146 : static bool IsSame(const XSModelGroup *poModelGroup1,
1147 : const XSModelGroup *poModelGroup2);
1148 : XSModelGroupDefinition *
1149 : GetGroupDefinition(const XSModelGroup *poModelGroup);
1150 : bool SetFieldFromAttribute(GMLASField &oField, XSAttributeUse *poAttr,
1151 : const CPLString &osXPathPrefix,
1152 : const CPLString &osNamePrefix = CPLString());
1153 : void GetConcreteImplementationTypes(
1154 : XSElementDeclaration *poParentElt,
1155 : std::vector<XSElementDeclaration *> &apoImplEltList);
1156 : std::vector<XSElementDeclaration *>
1157 : GetConstraintChildrenElements(const CPLString &osFullXPath);
1158 : bool FindElementsWithMustBeToLevel(
1159 : const CPLString &osParentXPath, XSModelGroup *poModelGroup,
1160 : int nRecursionCounter,
1161 : std::set<XSElementDeclaration *> &oSetVisitedEltDecl,
1162 : std::set<XSModelGroup *> &oSetVisitedModelGroups,
1163 : std::vector<XSElementDeclaration *> &oVectorEltsForTopClass,
1164 : std::set<CPLString> &aoSetXPathEltsForTopClass, XSModel *poModel,
1165 : bool &bSimpleEnoughOut, int &nCountSubEltsOut);
1166 : static void BuildMapCountOccurrencesOfSameName(
1167 : XSModelGroup *poModelGroup,
1168 : std::map<CPLString, int> &oMapCountOccurrencesOfSameName);
1169 : bool ExploreModelGroup(
1170 : XSModelGroup *psMainModelGroup, XSAttributeUseList *poMainAttrList,
1171 : GMLASFeatureClass &oClass, int nRecursionCounter,
1172 : std::set<XSModelGroup *> &oSetVisitedModelGroups, XSModel *poModel,
1173 : const std::map<CPLString, int> &oMapCountOccurrencesOfSameName);
1174 : void SetFieldTypeAndWidthFromDefinition(XSSimpleTypeDefinition *poST,
1175 : GMLASField &oField);
1176 : CPLString GetPrefix(const CPLString &osNamespaceURI);
1177 : CPLString MakeXPath(const CPLString &osNamespace, const CPLString &osName);
1178 : bool LaunderFieldNames(GMLASFeatureClass &oClass);
1179 : void LaunderClassNames();
1180 :
1181 : XSElementDeclaration *
1182 : GetTopElementDeclarationFromXPath(const CPLString &osXPath,
1183 : XSModel *poModel);
1184 :
1185 : bool InstantiateClassFromEltDeclaration(XSElementDeclaration *poEltDecl,
1186 : XSModel *poModel, bool &bError);
1187 : void CreateNonNestedRelationship(
1188 : XSElementDeclaration *poElt,
1189 : std::vector<XSElementDeclaration *> &apoSubEltList,
1190 : GMLASFeatureClass &oClass, int nMaxOccurs, bool bEltNameWillNeedPrefix,
1191 : bool bForceJunctionTable, bool bCaseOfConstraintChildren);
1192 :
1193 : bool IsGMLNamespace(const CPLString &osURI);
1194 :
1195 : bool DerivesFromGMLFeature(XSElementDeclaration *poEltDecl);
1196 :
1197 : bool IsIgnoredXPath(const CPLString &osXPath);
1198 :
1199 : static void
1200 : CollectClassesReferences(GMLASFeatureClass &oClass,
1201 : std::vector<GMLASFeatureClass *> &aoClasses);
1202 :
1203 : CPL_DISALLOW_COPY_ASSIGN(GMLASSchemaAnalyzer)
1204 :
1205 : public:
1206 : GMLASSchemaAnalyzer(
1207 : GMLASXPathMatcher &oIgnoredXPathMatcher,
1208 : GMLASXPathMatcher &oChildrenElementsConstraintsXPathMatcher,
1209 : const std::map<CPLString, std::vector<CPLString>>
1210 : &oMapChildrenElementsConstraints,
1211 : GMLASXPathMatcher &oForcedFlattenedXPathMatcher,
1212 : GMLASXPathMatcher &oDisabledFlattenedXPathMatcher);
1213 :
1214 188 : void SetUseArrays(bool b)
1215 : {
1216 188 : m_bUseArrays = b;
1217 188 : }
1218 :
1219 188 : void SetUseNullState(bool b)
1220 : {
1221 188 : m_bUseNullState = b;
1222 188 : }
1223 :
1224 188 : void SetInstantiateGMLFeaturesOnly(bool b)
1225 : {
1226 188 : m_bInstantiateGMLFeaturesOnly = b;
1227 188 : }
1228 :
1229 188 : void SetIdentifierMaxLength(int nLength)
1230 : {
1231 188 : m_nIdentifierMaxLength = nLength;
1232 188 : }
1233 :
1234 188 : void SetCaseInsensitiveIdentifier(bool b)
1235 : {
1236 188 : m_bCaseInsensitiveIdentifier = b;
1237 188 : }
1238 :
1239 188 : void SetPGIdentifierLaundering(bool b)
1240 : {
1241 188 : m_bPGIdentifierLaundering = b;
1242 188 : }
1243 :
1244 188 : void SetMaximumFieldsForFlattening(int n)
1245 : {
1246 188 : m_nMaximumFieldsForFlattening = n;
1247 188 : }
1248 :
1249 153 : void SetMapDocNSURIToPrefix(const std::map<CPLString, CPLString> &oMap)
1250 : {
1251 153 : m_oMapDocNSURIToPrefix = oMap;
1252 153 : }
1253 :
1254 188 : void SetAlwaysGenerateOGRId(bool b)
1255 : {
1256 188 : m_bAlwaysGenerateOGRId = b;
1257 188 : }
1258 :
1259 : bool Analyze(GMLASXSDCache &oCache, const CPLString &osBaseDirname,
1260 : std::vector<PairURIFilename> &aoXSDs, bool bSchemaFullChecking,
1261 : bool bHandleMultipleImports);
1262 :
1263 173 : const std::vector<GMLASFeatureClass> &GetClasses() const
1264 : {
1265 173 : return m_aoClasses;
1266 : }
1267 :
1268 173 : const std::map<CPLString, CPLString> &GetMapURIToPrefix() const
1269 : {
1270 173 : return m_oMapURIToPrefix;
1271 : }
1272 :
1273 173 : const CPLString &GetGMLVersionFound() const
1274 : {
1275 173 : return m_osGMLVersionFound;
1276 : }
1277 :
1278 173 : const std::set<CPLString> &GetSchemaURLS() const
1279 : {
1280 173 : return m_oSetSchemaURLs;
1281 : }
1282 :
1283 13535 : static CPLString BuildJunctionTableXPath(const CPLString &osEltXPath,
1284 : const CPLString &osSubEltXPath)
1285 : {
1286 27070 : return osEltXPath + "|" + osSubEltXPath;
1287 : }
1288 : };
1289 :
1290 : /************************************************************************/
1291 : /* OGRGMLASDataSource */
1292 : /************************************************************************/
1293 :
1294 : class OGRGMLASLayer;
1295 : class GMLASReader;
1296 :
1297 : class OGRGMLASDataSource final : public GDALDataset
1298 : {
1299 : struct XercesInitializer
1300 : {
1301 : XercesInitializer();
1302 : ~XercesInitializer();
1303 : };
1304 :
1305 : // MUST be first member, to get destroyed last after we have cleaned up
1306 : // all other Xerces dependent objects.
1307 : XercesInitializer m_oXercesInitializer{};
1308 :
1309 : std::vector<std::unique_ptr<OGRGMLASLayer>> m_apoLayers{};
1310 : std::map<CPLString, CPLString> m_oMapURIToPrefix{};
1311 : CPLString m_osGMLFilename{};
1312 : std::unique_ptr<OGRLayer> m_poFieldsMetadataLayer{};
1313 : std::unique_ptr<OGRLayer> m_poLayersMetadataLayer{};
1314 : std::unique_ptr<OGRLayer> m_poRelationshipsLayer{};
1315 : std::unique_ptr<OGRLayer> m_poOtherMetadataLayer{};
1316 : std::vector<OGRLayer *> m_apoRequestedMetadataLayers{};
1317 : std::shared_ptr<VSIVirtualHandle> m_fpGML{};
1318 : std::shared_ptr<VSIVirtualHandle> m_fpGMLParser{};
1319 : bool m_bLayerInitFinished = false;
1320 : bool m_bSchemaFullChecking = false;
1321 : bool m_bHandleMultipleImports = false;
1322 : bool m_bValidate = false;
1323 : bool m_bRemoveUnusedLayers = false;
1324 : bool m_bRemoveUnusedFields = false;
1325 : bool m_bFirstPassDone = false;
1326 : /** Map from a SRS name to a boolean indicating if its coordinate
1327 : order is inverted. */
1328 : std::map<CPLString, bool> m_oMapSRSNameToInvertedAxis{};
1329 :
1330 : /** Map from geometry field definition to its expected SRSName */
1331 : std::map<OGRGeomFieldDefn *, CPLString> m_oMapGeomFieldDefnToSRSName{};
1332 :
1333 : /* map the ID attribute to its belonging layer, e.g foo.1 -> layer Foo */
1334 : std::map<CPLString, OGRGMLASLayer *> m_oMapElementIdToLayer{};
1335 :
1336 : /* map the ID attribute to the feature PKID (when different from itself) */
1337 : std::map<CPLString, CPLString> m_oMapElementIdToPKID{};
1338 :
1339 : std::vector<PairURIFilename> m_aoXSDsManuallyPassed{};
1340 :
1341 : /** Default value for srsDimension attribute. */
1342 : int m_nDefaultSrsDimension = 0;
1343 :
1344 : GMLASConfiguration m_oConf{};
1345 :
1346 : /** Schema cache */
1347 : GMLASXSDCache m_oCache{};
1348 :
1349 : GMLASXPathMatcher m_oIgnoredXPathMatcher{};
1350 :
1351 : GMLASXPathMatcher m_oChildrenElementsConstraintsXPathMatcher{};
1352 :
1353 : GMLASXPathMatcher m_oForcedFlattenedXPathMatcher{};
1354 :
1355 : GMLASXPathMatcher m_oDisabledFlattenedXPathMatcher{};
1356 :
1357 : GMLASSwapCoordinatesEnum m_eSwapCoordinates = GMLAS_SWAP_AUTO;
1358 :
1359 : /** Base unique identifier */
1360 : CPLString m_osHash{};
1361 :
1362 : vsi_l_offset m_nFileSize = 0;
1363 :
1364 : std::unique_ptr<GMLASReader> m_poReader{};
1365 :
1366 : bool m_bEndOfReaderLayers = false;
1367 :
1368 : int m_nCurMetadataLayerIdx = -1;
1369 :
1370 : GMLASXLinkResolver m_oXLinkResolver{};
1371 :
1372 : CPLString m_osGMLVersionFound{};
1373 :
1374 : bool m_bFoundSWE = false;
1375 :
1376 : // Pointers are also included in m_apoLayers
1377 : std::vector<OGRGMLASLayer *> m_apoSWEDataArrayLayersRef{};
1378 :
1379 : // Path to gmlasconf.xml. It is a /vsimem temporary file if
1380 : // m_bUnlinkConfigFileAfterUse is set.
1381 : std::string m_osConfigFile{};
1382 :
1383 : // Whether m_osConfigFile should be removed at closing.
1384 : bool m_bUnlinkConfigFileAfterUse = false;
1385 :
1386 : void TranslateClasses(OGRGMLASLayer *poParentLayer,
1387 : const GMLASFeatureClass &oFC);
1388 :
1389 : bool RunFirstPassIfNeeded(GMLASReader *poReader,
1390 : GDALProgressFunc pfnProgress,
1391 : void *pProgressData);
1392 :
1393 : void FillOtherMetadataLayer(GDALOpenInfo *poOpenInfo,
1394 : const CPLString &osConfigFile,
1395 : const std::vector<PairURIFilename> &aoXSDs,
1396 : const std::set<CPLString> &oSetSchemaURLs);
1397 :
1398 : static std::vector<PairURIFilename>
1399 : BuildXSDVector(const CPLString &osXSDFilenames);
1400 :
1401 : void InitReaderWithFirstPassElements(GMLASReader *poReader);
1402 :
1403 : public:
1404 : OGRGMLASDataSource();
1405 :
1406 : ~OGRGMLASDataSource();
1407 :
1408 : virtual int GetLayerCount() override;
1409 : virtual OGRLayer *GetLayer(int) override;
1410 : virtual OGRLayer *GetLayerByName(const char *pszName) override;
1411 :
1412 : virtual void ResetReading() override;
1413 : virtual OGRFeature *GetNextFeature(OGRLayer **ppoBelongingLayer,
1414 : double *pdfProgressPct,
1415 : GDALProgressFunc pfnProgress,
1416 : void *pProgressData) override;
1417 : virtual int TestCapability(const char *) override;
1418 :
1419 : bool Open(GDALOpenInfo *poOpenInfo);
1420 :
1421 1384 : std::vector<std::unique_ptr<OGRGMLASLayer>> &GetLayers()
1422 : {
1423 1384 : return m_apoLayers;
1424 : }
1425 :
1426 1384 : const std::map<CPLString, CPLString> &GetMapURIToPrefix() const
1427 : {
1428 1384 : return m_oMapURIToPrefix;
1429 : }
1430 :
1431 1467 : const CPLString &GetGMLFilename() const
1432 : {
1433 1467 : return m_osGMLFilename;
1434 : }
1435 :
1436 : const CPLString &GetGMLVersionFound() const
1437 : {
1438 : return m_osGMLVersionFound;
1439 : }
1440 :
1441 15446 : OGRLayer *GetFieldsMetadataLayer()
1442 : {
1443 15446 : return m_poFieldsMetadataLayer.get();
1444 : }
1445 :
1446 15417 : OGRLayer *GetLayersMetadataLayer()
1447 : {
1448 15417 : return m_poLayersMetadataLayer.get();
1449 : }
1450 :
1451 15420 : OGRLayer *GetRelationshipsLayer()
1452 : {
1453 15420 : return m_poRelationshipsLayer.get();
1454 : }
1455 :
1456 : OGRGMLASLayer *GetLayerByXPath(const CPLString &osXPath);
1457 :
1458 : GMLASReader *CreateReader(std::shared_ptr<VSIVirtualHandle> &fpGML,
1459 : GDALProgressFunc pfnProgress = nullptr,
1460 : void *pProgressData = nullptr);
1461 :
1462 1276 : GMLASXSDCache &GetCache()
1463 : {
1464 1276 : return m_oCache;
1465 : }
1466 :
1467 : void PushUnusedGMLFilePointer(std::shared_ptr<VSIVirtualHandle> &fpGML);
1468 : std::shared_ptr<VSIVirtualHandle> PopUnusedGMLFilePointer();
1469 :
1470 43835 : bool IsLayerInitFinished() const
1471 : {
1472 43835 : return m_bLayerInitFinished;
1473 : }
1474 :
1475 1276 : GMLASSwapCoordinatesEnum GetSwapCoordinates() const
1476 : {
1477 1276 : return m_eSwapCoordinates;
1478 : }
1479 :
1480 1276 : const std::map<CPLString, bool> &GetMapIgnoredXPathToWarn() const
1481 : {
1482 1276 : return m_oConf.m_oMapIgnoredXPathToWarn;
1483 : }
1484 :
1485 1276 : const GMLASXPathMatcher &GetIgnoredXPathMatcher() const
1486 : {
1487 1276 : return m_oIgnoredXPathMatcher;
1488 : }
1489 :
1490 553 : const GMLASConfiguration &GetConf() const
1491 : {
1492 553 : return m_oConf;
1493 : }
1494 :
1495 : const std::vector<PairURIFilename> &GetXSDsManuallyPassed() const
1496 : {
1497 : return m_aoXSDsManuallyPassed;
1498 : }
1499 : };
1500 :
1501 : /************************************************************************/
1502 : /* OGRGMLASLayer */
1503 : /************************************************************************/
1504 :
1505 : class OGRGMLASLayer final : public OGRLayer
1506 : {
1507 : friend class OGRGMLASDataSource;
1508 :
1509 : OGRGMLASDataSource *m_poDS = nullptr;
1510 : GMLASFeatureClass m_oFC{};
1511 : bool m_bLayerDefnFinalized = false;
1512 : int m_nMaxFieldIndex = 0;
1513 : OGRFeatureDefn *m_poFeatureDefn = nullptr;
1514 :
1515 : /** Map from XPath to corresponding field index in OGR layer
1516 : definition */
1517 : std::map<CPLString, int> m_oMapFieldXPathToOGRFieldIdx{};
1518 :
1519 : /** Map from XPath to corresponding geometry field index in OGR layer
1520 : definition */
1521 : std::map<CPLString, int> m_oMapFieldXPathToOGRGeomFieldIdx{};
1522 :
1523 : /** Map from a OGR field index to the corresponding field index in
1524 : m_oFC.GetFields() */
1525 : std::map<int, int> m_oMapOGRFieldIdxtoFCFieldIdx{};
1526 : std::map<int, int> m_oMapOGRGeomFieldIdxtoFCFieldIdx{};
1527 :
1528 : /** Map from XPath to corresponding field index in m_oFC.GetFields() */
1529 : std::map<CPLString, int> m_oMapFieldXPathToFCFieldIdx{};
1530 :
1531 : bool m_bEOF = false;
1532 : std::unique_ptr<GMLASReader> m_poReader{};
1533 : std::shared_ptr<VSIVirtualHandle> m_fpGML{};
1534 : /** OGR field index of the ID field */
1535 : int m_nIDFieldIdx = -1;
1536 : /** Whether the ID field is generated, or comes from the XML content */
1537 : bool m_bIDFieldIsGenerated = false;
1538 : /** Pointer to parent layer */
1539 : OGRGMLASLayer *m_poParentLayer = nullptr;
1540 : /** OGR field index of the field that points to the parent ID */
1541 : int m_nParentIDFieldIdx = -1;
1542 :
1543 : std::map<CPLString, CPLString> m_oMapSWEFieldToOGRFieldName{};
1544 :
1545 : OGRFeature *GetNextRawFeature();
1546 :
1547 : bool InitReader();
1548 :
1549 14208 : void SetLayerDefnFinalized(bool bVal)
1550 : {
1551 14208 : m_bLayerDefnFinalized = bVal;
1552 14208 : }
1553 :
1554 : CPLString LaunderFieldName(const CPLString &osFieldName);
1555 :
1556 : CPLString GetXPathFromOGRFieldIndex(int nIdx) const;
1557 :
1558 : CPL_DISALLOW_COPY_ASSIGN(OGRGMLASLayer)
1559 :
1560 : public:
1561 : OGRGMLASLayer(OGRGMLASDataSource *poDS, const GMLASFeatureClass &oFC,
1562 : OGRGMLASLayer *poParentLayer, bool bAlwaysGenerateOGRPKId);
1563 : explicit OGRGMLASLayer(const char *pszLayerName);
1564 : virtual ~OGRGMLASLayer();
1565 :
1566 311087 : virtual const char *GetName() override
1567 : {
1568 311087 : return GetDescription();
1569 : }
1570 :
1571 : virtual OGRFeatureDefn *GetLayerDefn() override;
1572 : virtual void ResetReading() override;
1573 : virtual OGRFeature *GetNextFeature() override;
1574 :
1575 919 : virtual int TestCapability(const char *) override
1576 : {
1577 919 : return FALSE;
1578 : }
1579 :
1580 3 : void SetDataSource(OGRGMLASDataSource *poDS)
1581 : {
1582 3 : m_poDS = poDS;
1583 3 : }
1584 :
1585 : void PostInit(bool bIncludeGeometryXML);
1586 : void
1587 : ProcessDataRecordCreateFields(CPLXMLNode *psDataRecord,
1588 : const std::vector<OGRFeature *> &apoFeatures,
1589 : OGRLayer *poFieldsMetadataLayer);
1590 : void ProcessDataRecordFillFeature(CPLXMLNode *psDataRecord,
1591 : OGRFeature *poFeature);
1592 : void
1593 : ProcessDataRecordOfDataArrayCreateFields(OGRGMLASLayer *poParentLayer,
1594 : CPLXMLNode *psDataRecord,
1595 : OGRLayer *poFieldsMetadataLayer);
1596 : void CreateCompoundFoldedMappings();
1597 :
1598 22004500 : const GMLASFeatureClass &GetFeatureClass() const
1599 : {
1600 22004500 : return m_oFC;
1601 : }
1602 :
1603 : int GetOGRFieldIndexFromXPath(const CPLString &osXPath) const;
1604 : int GetOGRGeomFieldIndexFromXPath(const CPLString &osXPath) const;
1605 :
1606 190220 : int GetIDFieldIdx() const
1607 : {
1608 190220 : return m_nIDFieldIdx;
1609 : }
1610 :
1611 52459 : bool IsGeneratedIDField() const
1612 : {
1613 52459 : return m_bIDFieldIsGenerated;
1614 : }
1615 :
1616 109441 : OGRGMLASLayer *GetParent()
1617 : {
1618 109441 : return m_poParentLayer;
1619 : }
1620 :
1621 90567 : int GetParentIDFieldIdx() const
1622 : {
1623 90567 : return m_nParentIDFieldIdx;
1624 : }
1625 :
1626 : int GetFCFieldIndexFromOGRFieldIdx(int iOGRFieldIdx) const;
1627 : int GetFCFieldIndexFromOGRGeomFieldIdx(int iOGRGeomFieldIdx) const;
1628 : int GetFCFieldIndexFromXPath(const CPLString &osXPath) const;
1629 :
1630 : bool EvaluateFilter(OGRFeature *poFeature);
1631 :
1632 : bool RemoveField(int nIdx);
1633 : void InsertNewField(int nInsertPos, const OGRFieldDefn &oFieldDefn,
1634 : const CPLString &osXPath);
1635 :
1636 : CPLString
1637 : GetXPathOfFieldLinkForAttrToOtherLayer(const CPLString &osFieldName,
1638 : const CPLString &osTargetLayerXPath);
1639 : CPLString
1640 : CreateLinkForAttrToOtherLayer(const CPLString &osFieldName,
1641 : const CPLString &osTargetLayerXPath);
1642 : };
1643 :
1644 : /************************************************************************/
1645 : /* GMLASReader */
1646 : /************************************************************************/
1647 :
1648 : class GMLASReader final : public DefaultHandler
1649 : {
1650 : /** Schema cache */
1651 : GMLASXSDCache &m_oCache;
1652 :
1653 : /** Object to tell if a XPath must be ignored */
1654 : const GMLASXPathMatcher &m_oIgnoredXPathMatcher;
1655 :
1656 : /** XLink resolver */
1657 : GMLASXLinkResolver &m_oXLinkResolver;
1658 :
1659 : /** Whether we should stop parsing */
1660 : bool m_bParsingError = false;
1661 :
1662 : /** Xerces reader object */
1663 : std::unique_ptr<SAX2XMLReader> m_poSAXReader{};
1664 :
1665 : /** Token for Xerces */
1666 : XMLPScanToken m_oToFill{};
1667 :
1668 : /** File descriptor */
1669 : std::shared_ptr<VSIVirtualHandle> m_fp{};
1670 :
1671 : /** Input source */
1672 : std::unique_ptr<GMLASInputSource> m_GMLInputSource{};
1673 :
1674 : /** Whether we are at the first iteration */
1675 : bool m_bFirstIteration = true;
1676 :
1677 : /** Whether we have reached end of file (or an error) */
1678 : bool m_bEOF = false;
1679 :
1680 : /** Whether GetNextFeature() has been user interrupted (progress cbk) */
1681 : bool m_bInterrupted = false;
1682 :
1683 : /** Error handler (for Xerces reader) */
1684 : GMLASErrorHandler m_oErrorHandler{};
1685 :
1686 : /** Map URI namespaces to their prefix */
1687 : std::map<CPLString, CPLString> m_oMapURIToPrefix{};
1688 :
1689 : /** List of OGR layers */
1690 : std::vector<std::unique_ptr<OGRGMLASLayer>> *m_apoLayers = nullptr;
1691 :
1692 : /** Vector of features ready for consumption */
1693 : std::list<std::pair<std::unique_ptr<OGRFeature>, OGRGMLASLayer *>>
1694 : m_aoFeaturesReady{};
1695 :
1696 : /** OGR field index of the current field */
1697 : int m_nCurFieldIdx = -1;
1698 :
1699 : /** OGR geometry field index of the current field */
1700 : int m_nCurGeomFieldIdx = -1;
1701 :
1702 : /** XML nested level of current field */
1703 : int m_nCurFieldLevel = 0;
1704 :
1705 : /** Whether we should store all content of the current field as XML */
1706 : bool m_bIsXMLBlob = false;
1707 : bool m_bIsXMLBlobIncludeUpper = false;
1708 :
1709 : /** Content of the current field */
1710 : CPLString m_osTextContent{};
1711 :
1712 : /** For list field types, list of content */
1713 : CPLStringList m_osTextContentList{};
1714 : /** Estimated memory footprint of m_osTextContentList */
1715 : size_t m_nTextContentListEstimatedSize = 0;
1716 :
1717 : /** Which layer is of interest for the reader, or NULL for all */
1718 : OGRGMLASLayer *m_poLayerOfInterest = nullptr;
1719 :
1720 : /** Stack of length of split XPath components */
1721 : std::vector<size_t> m_anStackXPathLength{};
1722 :
1723 : /** Current absolute XPath */
1724 : CPLString m_osCurXPath{};
1725 :
1726 : /** Current XPath, relative to top-level feature */
1727 : CPLString m_osCurSubXPath{};
1728 :
1729 : /** Current XML nesting level */
1730 : int m_nLevel = 0;
1731 :
1732 : /** Whether we are in a gml:boundedBy element at level 1 */
1733 : bool m_bInGMLBoundedByLevel1 = false;
1734 :
1735 : /** Default value for srsDimension attribute. */
1736 : int m_nDefaultSrsDimension = 0;
1737 :
1738 : /** Map layer to global FID */
1739 : std::map<OGRLayer *, int> m_oMapGlobalCounter{};
1740 :
1741 : /** Parsing context */
1742 : struct Context
1743 : {
1744 : /** XML nesting level */
1745 : int m_nLevel = 0;
1746 :
1747 : /** Current feature */
1748 : OGRFeature *m_poFeature = nullptr;
1749 :
1750 : /** Layer of m_poFeature */
1751 : OGRGMLASLayer *m_poLayer = nullptr;
1752 :
1753 : /** Current layer in a repeated group */
1754 : OGRGMLASLayer *m_poGroupLayer = nullptr;
1755 :
1756 : /** Nesting level of m_poCurGroupLayer */
1757 : int m_nGroupLayerLevel = -1;
1758 :
1759 : /** Index of the last processed OGR field in m_poCurGroupLayer */
1760 : int m_nLastFieldIdxGroupLayer = -1;
1761 :
1762 : /** Map layer to local FID */
1763 : std::map<OGRLayer *, int> m_oMapCounter{};
1764 :
1765 : /** Current XPath, relative to (current) top-level feature */
1766 : CPLString m_osCurSubXPath{};
1767 :
1768 : void Dump() const;
1769 : };
1770 :
1771 : /** Current context */
1772 : Context m_oCurCtxt{};
1773 :
1774 : /** Stack of saved contexts */
1775 : std::vector<Context> m_aoStackContext{};
1776 :
1777 : /** Context used in m_apsXMLNodeStack */
1778 : struct NodeLastChild
1779 : {
1780 : /** Current node */
1781 : CPLXMLNode *psNode = nullptr;
1782 :
1783 : /** Last child of psNode (for fast append operations) */
1784 : CPLXMLNode *psLastChild = nullptr;
1785 : };
1786 :
1787 : /** Stack of contexts to build XML tree of GML Geometry */
1788 : std::vector<NodeLastChild> m_apsXMLNodeStack{};
1789 :
1790 : /** Counter used to prevent XML billion laugh attacks */
1791 : int m_nEntityCounter = 0;
1792 :
1793 : /** Maximum allowed number of XML nesting level */
1794 : int m_nMaxLevel = 100;
1795 :
1796 : /** Maximum allowed size of XML content in byte */
1797 : size_t m_nMaxContentSize = 512000000;
1798 :
1799 : /** Map from a SRS name to a boolean indicating if its coordinate
1800 : order is inverted. */
1801 : std::map<CPLString, bool> m_oMapSRSNameToInvertedAxis{};
1802 :
1803 : /** Set of geometry fields with unknown SRS */
1804 : std::set<OGRGeomFieldDefn *> m_oSetGeomFieldsWithUnknownSRS{};
1805 :
1806 : /** Map from geometry field definition to its expected SRSName.
1807 : This is used to know if reprojection must be done */
1808 : std::map<OGRGeomFieldDefn *, CPLString> m_oMapGeomFieldDefnToSRSName{};
1809 :
1810 : /** Whether this parsing involves schema validation */
1811 : bool m_bValidate = false;
1812 :
1813 : /** Entity resolver used during schema validation */
1814 : std::unique_ptr<GMLASBaseEntityResolver> m_poEntityResolver{};
1815 :
1816 : /** First level from which warnings about ignored XPath should be
1817 : silent. */
1818 : int m_nLevelSilentIgnoredXPath = -1;
1819 :
1820 : /** Whether a warning should be emitted when an element or attribute is
1821 : found in the document parsed, but ignored because of the ignored
1822 : XPath defined. */
1823 : std::map<CPLString, bool> m_oMapIgnoredXPathToWarn{};
1824 :
1825 : /** Policy to decide when to invert coordinates */
1826 : GMLASSwapCoordinatesEnum m_eSwapCoordinates = GMLAS_SWAP_AUTO;
1827 :
1828 : /** Initial pass to guess SRS, etc... */
1829 : bool m_bInitialPass = false;
1830 :
1831 : /** Whether to process swe:DataArray in a special way */
1832 : bool m_bProcessSWEDataArray = false;
1833 :
1834 : /** Whether to process swe:DataArray in a special way */
1835 : bool m_bProcessSWEDataRecord = false;
1836 :
1837 : /** Depth level of the swe:DataArray element */
1838 : int m_nSWEDataArrayLevel = -1;
1839 :
1840 : /** Field name to which the DataArray belongs to */
1841 : CPLString m_osSWEDataArrayParentField{};
1842 :
1843 : /** Depth level of the swe:DataRecord element */
1844 : int m_nSWEDataRecordLevel = -1;
1845 :
1846 : OGRLayer *m_poFieldsMetadataLayer = nullptr;
1847 : OGRLayer *m_poLayersMetadataLayer = nullptr;
1848 : OGRLayer *m_poRelationshipsLayer = nullptr;
1849 :
1850 : /** Base unique identifier */
1851 : CPLString m_osHash{};
1852 :
1853 : vsi_l_offset m_nFileSize = 0;
1854 :
1855 : bool m_bWarnUnexpected = false;
1856 :
1857 : /** Map from layer to a map of field XPath to a set of matching
1858 : URL specific resolution rule index */
1859 : std::map<OGRGMLASLayer *, std::map<CPLString, std::set<int>>>
1860 : m_oMapXLinkFields{};
1861 :
1862 : /** Variables that could be local but more efficient to have same
1863 : persistent, so as to save many memory allocations/deallocations */
1864 : CPLString m_osLocalname{};
1865 : CPLString m_osNSUri{};
1866 : CPLString m_osNSPrefix{};
1867 : CPLString m_osXPath{};
1868 : CPLString m_osLayerXPath{};
1869 : CPLString m_osAttrNSUri{};
1870 : CPLString m_osAttrNSPrefix{};
1871 : CPLString m_osAttrLocalName{};
1872 : CPLString m_osAttrXPath{};
1873 : CPLString m_osAttrValue{};
1874 : CPLString m_osText{};
1875 :
1876 : std::vector<OGRGMLASLayer *> m_apoSWEDataArrayLayersRef{};
1877 : std::vector<std::unique_ptr<OGRGMLASLayer>> m_apoSWEDataArrayLayersOwned{};
1878 :
1879 : int m_nSWEDataArrayLayerIdx = 0;
1880 :
1881 : /* Set of 3 maps used for xlink:href="#xxxx" internal links resolution */
1882 : /* 1) map the ID attribute to its belonging layer, e.g foo.1 -> layer Foo */
1883 : std::map<CPLString, OGRGMLASLayer *> m_oMapElementIdToLayer{};
1884 : /* 2) map the ID attribute to the feature PKID (when different from itself)
1885 : */
1886 : std::map<CPLString, CPLString> m_oMapElementIdToPKID{};
1887 : /* 3) map each (layer, field_xpath) to the list of ID it refers to */
1888 : /* e.g (layer Bar, field_xpath) -> [foo.1, foo.2] */
1889 : std::map<std::pair<OGRGMLASLayer *, CPLString>, std::vector<CPLString>>
1890 : m_oMapFieldXPathToLinkValue{};
1891 :
1892 : void SetField(OGRFeature *poFeature, OGRGMLASLayer *poLayer, int nAttrIdx,
1893 : const CPLString &osAttrValue);
1894 :
1895 : void CreateNewFeature(const CPLString &osLocalname);
1896 :
1897 : void PushFeatureReady(std::unique_ptr<OGRFeature> &&,
1898 : OGRGMLASLayer *poLayer);
1899 :
1900 : void PushContext(const Context &oContext);
1901 : void PopContext();
1902 :
1903 : void BuildXMLBlobStartElement(const CPLString &osXPath,
1904 : const Attributes &attrs);
1905 :
1906 : OGRGMLASLayer *GetLayerByXPath(const CPLString &osXPath);
1907 :
1908 : void AttachAsLastChild(CPLXMLNode *psNode);
1909 :
1910 : void ProcessSWEDataArray(CPLXMLNode *psRoot);
1911 : void ProcessSWEDataRecord(CPLXMLNode *psRoot);
1912 : void ProcessGeometry(CPLXMLNode *psRoot);
1913 :
1914 : void ProcessAttributes(const Attributes &attrs);
1915 : void ProcessXLinkHref(int nAttrIdx, const CPLString &osAttrXPath,
1916 : const CPLString &osAttrValue);
1917 : void
1918 : ExploreXMLDoc(const CPLString &osAttrXPath,
1919 : const GMLASXLinkResolutionConf::URLSpecificResolution &oRule,
1920 : CPLXMLNode *psNode, const CPLString &osParentXPath,
1921 : const GMLASXPathMatcher &oMatcher,
1922 : const std::map<CPLString, size_t> &oMapFieldXPathToIdx);
1923 :
1924 : void CreateFieldsForURLSpecificRules();
1925 : void CreateFieldsForURLSpecificRule(
1926 : OGRGMLASLayer *poLayer, int nFieldIdx, const CPLString &osFieldXPath,
1927 : int &nInsertFieldIdx,
1928 : const GMLASXLinkResolutionConf::URLSpecificResolution &oRule);
1929 :
1930 321225 : bool FillTextContent() const
1931 : {
1932 321225 : return !m_bInitialPass && m_nCurFieldIdx >= 0;
1933 : }
1934 :
1935 : void ProcessInternalXLinkFirstPass(
1936 : bool bRemoveUnusedFields,
1937 : std::map<OGRGMLASLayer *, std::set<CPLString>> &oMapUnusedFields);
1938 :
1939 : CPL_DISALLOW_COPY_ASSIGN(GMLASReader)
1940 :
1941 : public:
1942 : GMLASReader(GMLASXSDCache &oCache,
1943 : const GMLASXPathMatcher &oIgnoredXPathMatcher,
1944 : GMLASXLinkResolver &oXLinkResolver);
1945 : ~GMLASReader();
1946 :
1947 : bool Init(const char *pszFilename,
1948 : const std::shared_ptr<VSIVirtualHandle> &fp,
1949 : const std::map<CPLString, CPLString> &oMapURIToPrefix,
1950 : std::vector<std::unique_ptr<OGRGMLASLayer>> &apoLayers,
1951 : bool bValidate, const std::vector<PairURIFilename> &aoXSDs,
1952 : bool bSchemaFullChecking, bool bHandleMultipleImports);
1953 :
1954 : void SetLayerOfInterest(OGRGMLASLayer *poLayer);
1955 :
1956 1384 : void SetMapIgnoredXPathToWarn(const std::map<CPLString, bool> &oMap)
1957 : {
1958 1384 : m_oMapIgnoredXPathToWarn = oMap;
1959 1384 : }
1960 :
1961 1276 : void SetSwapCoordinates(GMLASSwapCoordinatesEnum eVal)
1962 : {
1963 1276 : m_eSwapCoordinates = eVal;
1964 1276 : }
1965 :
1966 39 : const std::shared_ptr<VSIVirtualHandle> &GetFP() const
1967 : {
1968 39 : return m_fp;
1969 : }
1970 :
1971 108 : const std::map<CPLString, bool> &GetMapSRSNameToInvertedAxis() const
1972 : {
1973 108 : return m_oMapSRSNameToInvertedAxis;
1974 : }
1975 :
1976 1263 : void SetMapSRSNameToInvertedAxis(const std::map<CPLString, bool> &oMap)
1977 : {
1978 1263 : m_oMapSRSNameToInvertedAxis = oMap;
1979 1263 : }
1980 :
1981 : const std::map<OGRGeomFieldDefn *, CPLString> &
1982 108 : GetMapGeomFieldDefnToSRSName() const
1983 : {
1984 108 : return m_oMapGeomFieldDefnToSRSName;
1985 : }
1986 :
1987 1263 : void SetMapGeomFieldDefnToSRSName(
1988 : const std::map<OGRGeomFieldDefn *, CPLString> &oMap)
1989 : {
1990 1263 : m_oMapGeomFieldDefnToSRSName = oMap;
1991 1263 : }
1992 :
1993 108 : const std::map<CPLString, OGRGMLASLayer *> &GetMapElementIdToLayer() const
1994 : {
1995 108 : return m_oMapElementIdToLayer;
1996 : }
1997 :
1998 : void
1999 1263 : SetMapElementIdToLayer(const std::map<CPLString, OGRGMLASLayer *> &oMap)
2000 : {
2001 1263 : m_oMapElementIdToLayer = oMap;
2002 1263 : }
2003 :
2004 108 : const std::map<CPLString, CPLString> &GetMapElementIdToPKID() const
2005 : {
2006 108 : return m_oMapElementIdToPKID;
2007 : }
2008 :
2009 1263 : void SetMapElementIdToPKID(const std::map<CPLString, CPLString> &oMap)
2010 : {
2011 1263 : m_oMapElementIdToPKID = oMap;
2012 1263 : }
2013 :
2014 108 : int GetDefaultSrsDimension() const
2015 : {
2016 108 : return m_nDefaultSrsDimension;
2017 : }
2018 :
2019 1263 : void SetDefaultSrsDimension(int nDim)
2020 : {
2021 1263 : m_nDefaultSrsDimension = nDim;
2022 1263 : }
2023 :
2024 1384 : void SetHash(const CPLString &osHash)
2025 : {
2026 1384 : m_osHash = osHash;
2027 1384 : }
2028 :
2029 1384 : void SetFileSize(vsi_l_offset nFileSize)
2030 : {
2031 1384 : m_nFileSize = nFileSize;
2032 1384 : }
2033 :
2034 : OGRFeature *GetNextFeature(OGRGMLASLayer **ppoBelongingLayer = nullptr,
2035 : GDALProgressFunc pfnProgress = nullptr,
2036 : void *pProgressData = nullptr);
2037 :
2038 : virtual void startElement(const XMLCh *const uri,
2039 : const XMLCh *const localname,
2040 : const XMLCh *const qname,
2041 : const Attributes &attrs) override;
2042 : virtual void endElement(const XMLCh *const uri,
2043 : const XMLCh *const localname,
2044 : const XMLCh *const qname) override;
2045 :
2046 : virtual void characters(const XMLCh *const chars,
2047 : const XMLSize_t length) override;
2048 :
2049 : void startEntity(const XMLCh *const name) override;
2050 :
2051 : bool RunFirstPass(GDALProgressFunc pfnProgress, void *pProgressData,
2052 : bool bRemoveUnusedLayers, bool bRemoveUnusedFields,
2053 : bool bProcessSWEDataArray,
2054 : OGRLayer *poFieldsMetadataLayer,
2055 : OGRLayer *poLayersMetadataLayer,
2056 : OGRLayer *poRelationshipsLayer,
2057 : std::set<CPLString> &aoSetRemovedLayerNames);
2058 :
2059 : static bool LoadXSDInParser(SAX2XMLReader *poParser, GMLASXSDCache &oCache,
2060 : GMLASBaseEntityResolver &oXSDEntityResolver,
2061 : const CPLString &osBaseDirname,
2062 : const CPLString &osXSDFilename,
2063 : Grammar **ppoGrammar, bool bSchemaFullChecking,
2064 : bool bHandleMultipleImports);
2065 :
2066 : void SetSWEDataArrayLayersRef(const std::vector<OGRGMLASLayer *> &ar);
2067 :
2068 1371 : void SetProcessDataRecord(bool b)
2069 : {
2070 1371 : m_bProcessSWEDataRecord = b;
2071 1371 : }
2072 :
2073 108 : std::vector<std::unique_ptr<OGRGMLASLayer>> StealSWEDataArrayLayersOwned()
2074 : {
2075 108 : return std::move(m_apoSWEDataArrayLayersOwned);
2076 : }
2077 : };
2078 :
2079 : CPLString OGRGMLASTruncateIdentifier(const CPLString &osName,
2080 : int nIdentMaxLength);
2081 :
2082 : CPLString OGRGMLASAddSerialNumber(const CPLString &osNameIn, int iOccurrence,
2083 : size_t nOccurrences, int nIdentMaxLength);
2084 :
2085 : #endif // OGR_GMLAS_INCLUDED
|