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 : #include "ogr_gmlas.h"
16 : #include "ogr_p.h"
17 : #include "ogrlibjsonutils.h"
18 : #include "cpl_time.h"
19 :
20 : #include <algorithm>
21 : #include <cmath>
22 :
23 : namespace GMLAS
24 : {
25 :
26 : /************************************************************************/
27 : /* GMLASWriter */
28 : /************************************************************************/
29 :
30 : typedef std::pair<CPLString, CPLString> PairNSElement;
31 :
32 : typedef std::vector<PairNSElement> XPathComponents;
33 :
34 : typedef std::pair<CPLString, CPLString> PairLayerNameColName;
35 :
36 : class LayerDescription
37 : {
38 : public:
39 : CPLString osName{};
40 : CPLString osXPath{};
41 : CPLString osPKIDName{};
42 : CPLString osParentPKIDName{};
43 : bool bIsSelected = false;
44 : bool bIsTopLevel = false;
45 : bool bIsJunction = false;
46 : // map a field sequential number to a field
47 : std::map<int, GMLASField> oMapIdxToField{};
48 : // map a field xpath to its sequential number
49 : std::map<CPLString, int> oMapFieldXPathToIdx{};
50 : std::map<CPLString, int> oMapFieldNameToOGRIdx{};
51 : std::vector<PairLayerNameColName> aoReferencingLayers{};
52 :
53 : // NOTE: this doesn't scale to arbitrarily large datasets
54 : std::set<GIntBig> aoSetReferencedFIDs{};
55 :
56 117 : LayerDescription() = default;
57 :
58 852 : int GetOGRIdxFromFieldName(const CPLString &osFieldName) const
59 : {
60 852 : const auto oIter = oMapFieldNameToOGRIdx.find(osFieldName);
61 852 : if (oIter == oMapFieldNameToOGRIdx.end())
62 0 : return -1;
63 852 : return oIter->second;
64 : }
65 : };
66 :
67 : class GMLASWriter
68 : {
69 : GMLASConfiguration m_oConf{};
70 : CPLString m_osFilename{};
71 : CPLString m_osGMLVersion{};
72 : CPLString m_osSRSNameFormat{};
73 : #ifdef _WIN32
74 : CPLString m_osEOL = "\r\n";
75 : #else
76 : CPLString m_osEOL = "\n";
77 : #endif
78 : GDALDataset *m_poSrcDS;
79 : CPLStringList m_aosOptions{};
80 : VSIVirtualHandleUniquePtr m_fpXML{};
81 : std::unique_ptr<OGRGMLASDataSource> m_poTmpDS{};
82 : OGRLayer *m_poLayersMDLayer = nullptr;
83 : OGRLayer *m_poFieldsMDLayer = nullptr;
84 : OGRLayer *m_poLayerRelationshipsLayer = nullptr;
85 : std::vector<LayerDescription> m_aoLayerDesc{};
86 : std::map<CPLString, int> m_oMapLayerNameToIdx{};
87 : std::map<CPLString, int> m_oMapXPathToIdx{};
88 : std::map<CPLString, OGRLayer *> m_oMapLayerNameToLayer{};
89 : std::map<CPLString, XPathComponents> m_oMapXPathToComponents{};
90 : std::map<const OGRSpatialReference *, bool> m_oMapSRSToCoordSwap{};
91 :
92 : CPLString m_osTargetNameSpace = szOGRGMLAS_URI;
93 : CPLString m_osTargetNameSpacePrefix = szOGRGMLAS_PREFIX;
94 :
95 : CPLString m_osIndentation = std::string(INDENT_SIZE_DEFAULT, ' ');
96 : int m_nIndentLevel = 0;
97 :
98 674 : void IncIndent()
99 : {
100 674 : ++m_nIndentLevel;
101 674 : }
102 :
103 674 : void DecIndent()
104 : {
105 674 : --m_nIndentLevel;
106 674 : }
107 :
108 : void PrintIndent(VSILFILE *fp);
109 :
110 : void PrintLine(VSILFILE *fp, const char *fmt, ...)
111 : CPL_PRINT_FUNC_FORMAT(3, 4);
112 :
113 : bool WriteXSD(const CPLString &osXSDFilenameIn,
114 : const std::vector<PairURIFilename> &aoXSDs);
115 : bool WriteXMLHeader(bool bWFS2FeatureCollection, GIntBig nTotalFeatures,
116 : bool bGenerateXSD, const CPLString &osXSDFilenameIn,
117 : const std::vector<PairURIFilename> &aoXSDs,
118 : const std::map<CPLString, CPLString> &oMapURIToPrefix);
119 : bool CollectLayers();
120 : bool CollectFields();
121 : bool CollectRelationships();
122 : void ComputeTopLevelFIDs();
123 : bool WriteLayer(bool bWFS2FeatureCollection, const LayerDescription &oDesc,
124 : GIntBig &nFeaturesWritten, GIntBig nTotalTopLevelFeatures,
125 : GDALProgressFunc pfnProgress, void *pProgressData);
126 :
127 : bool WriteFeature(OGRFeature *poFeature, const LayerDescription &oLayerDesc,
128 : const std::set<CPLString> &oSetLayersInIteration,
129 : const XPathComponents &aoInitialComponents,
130 : const XPathComponents &aoPrefixComponents, int nRecLevel);
131 :
132 : void WriteClosingTags(size_t nCommonLength,
133 : const XPathComponents &aoCurComponents,
134 : const XPathComponents &aoNewComponents,
135 : bool bCurIsRegularField, bool bNewIsRegularField);
136 :
137 : void WriteClosingAndStartingTags(const XPathComponents &aoCurComponents,
138 : const XPathComponents &aoNewComponents,
139 : bool bCurIsRegularField);
140 :
141 : void PrintMultipleValuesSeparator(const GMLASField &oField,
142 : const XPathComponents &aoFieldComponents);
143 :
144 : OGRLayer *
145 : GetFilteredLayer(OGRLayer *poSrcLayer, const CPLString &osFilter,
146 : const std::set<CPLString> &oSetLayersInIteration);
147 : void ReleaseFilteredLayer(OGRLayer *poSrcLayer, OGRLayer *poIterLayer);
148 :
149 : bool WriteFieldRegular(OGRFeature *poFeature, const GMLASField &oField,
150 : const LayerDescription &oLayerDesc,
151 : /*XPathComponents& aoLayerComponents,*/
152 : XPathComponents &aoCurComponents,
153 : const XPathComponents &aoPrefixComponents,
154 : /*const std::set<CPLString>& oSetLayersInIteration,*/
155 : bool &bAtLeastOneFieldWritten,
156 : bool &bCurIsRegularField);
157 :
158 : bool WriteFieldNoLink(OGRFeature *poFeature, const GMLASField &oField,
159 : const LayerDescription &oLayerDesc,
160 : XPathComponents &aoLayerComponents,
161 : XPathComponents &aoCurComponents,
162 : const XPathComponents &aoPrefixComponents,
163 : const std::set<CPLString> &oSetLayersInIteration,
164 : int nRecLevel, bool &bAtLeastOneFieldWritten,
165 : bool &bCurIsRegularField);
166 :
167 : bool WriteFieldWithLink(OGRFeature *poFeature, const GMLASField &oField,
168 : const LayerDescription &oLayerDesc,
169 : XPathComponents &aoLayerComponents,
170 : XPathComponents &aoCurComponents,
171 : const XPathComponents &aoPrefixComponents,
172 : const std::set<CPLString> &oSetLayersInIteration,
173 : int nRecLevel, bool &bAtLeastOneFieldWritten,
174 : bool &bCurIsRegularField);
175 :
176 : bool WriteFieldJunctionTable(
177 : OGRFeature *poFeature, const GMLASField &oField,
178 : const LayerDescription &oLayerDesc, XPathComponents &aoLayerComponents,
179 : XPathComponents &aoCurComponents,
180 : const XPathComponents &aoPrefixComponents,
181 : const std::set<CPLString> &oSetLayersInIteration, int nRecLevel,
182 : bool &bAtLeastOneFieldWritten, bool &bCurIsRegularField);
183 :
184 : void Close();
185 :
186 : OGRLayer *GetLayerByName(const CPLString &osName);
187 :
188 : const XPathComponents &SplitXPath(const CPLString &osXPath);
189 :
190 : bool GetCoordSwap(const OGRSpatialReference *poSRS);
191 :
192 : CPL_DISALLOW_COPY_ASSIGN(GMLASWriter)
193 :
194 : public:
195 : GMLASWriter(const char *pszFilename, GDALDataset *poSrcDS,
196 : CSLConstList papszOptions);
197 :
198 : bool Write(GDALProgressFunc pfnProgress, void *pProgressData);
199 : };
200 :
201 : /************************************************************************/
202 : /* GMLASWriter() */
203 : /************************************************************************/
204 :
205 21 : GMLASWriter::GMLASWriter(const char *pszFilename, GDALDataset *poSrcDS,
206 21 : CSLConstList papszOptions)
207 21 : : m_osFilename(pszFilename), m_poSrcDS(poSrcDS), m_aosOptions(papszOptions)
208 : {
209 21 : }
210 :
211 : /************************************************************************/
212 : /* Close() */
213 : /************************************************************************/
214 :
215 10 : void GMLASWriter::Close()
216 : {
217 10 : m_fpXML.reset();
218 10 : m_poTmpDS.reset();
219 10 : }
220 :
221 : /************************************************************************/
222 : /* Write() */
223 : /************************************************************************/
224 :
225 21 : bool GMLASWriter::Write(GDALProgressFunc pfnProgress, void *pProgressData)
226 : {
227 22 : if (m_poSrcDS->GetLayerCount() == 0 &&
228 1 : m_poSrcDS->GetLayerByName(szOGR_OTHER_METADATA) == nullptr)
229 : {
230 1 : CPLError(CE_Failure, CPLE_AppDefined, "Source dataset has no layers");
231 1 : return false;
232 : }
233 :
234 : // Load configuration file
235 : CPLString osConfigFile =
236 40 : m_aosOptions.FetchNameValueDef(szCONFIG_FILE_OPTION, "");
237 20 : bool bUnlinkAfterUse = false;
238 20 : if (osConfigFile.empty())
239 : {
240 19 : osConfigFile = GMLASConfiguration::GetDefaultConfFile(bUnlinkAfterUse);
241 : }
242 20 : if (osConfigFile.empty())
243 : {
244 0 : CPLError(CE_Warning, CPLE_AppDefined,
245 : "No configuration file found. Using hard-coded defaults");
246 0 : m_oConf.Finalize();
247 : }
248 : else
249 : {
250 20 : const bool bOK = m_oConf.Load(osConfigFile);
251 20 : if (bUnlinkAfterUse)
252 0 : VSIUnlink(osConfigFile.c_str());
253 20 : if (!bOK)
254 : {
255 1 : CPLError(CE_Failure, CPLE_AppDefined,
256 : "Loading of configuration failed");
257 1 : return false;
258 : }
259 : }
260 :
261 : CPLString osXSDFilenames =
262 38 : m_aosOptions.FetchNameValueDef(szINPUT_XSD_OPTION, "");
263 38 : std::vector<PairURIFilename> aoXSDs;
264 38 : std::map<CPLString, CPLString> oMapURIToPrefix;
265 38 : CPLString osGMLVersion;
266 :
267 19 : if (!osXSDFilenames.empty())
268 : {
269 : // Create a fake GMLAS dataset from the XSD= value
270 2 : m_poTmpDS = std::make_unique<OGRGMLASDataSource>();
271 2 : GDALOpenInfo oOpenInfo(szGMLAS_PREFIX, GA_ReadOnly);
272 2 : oOpenInfo.papszOpenOptions = CSLSetNameValue(
273 : oOpenInfo.papszOpenOptions, szXSD_OPTION, osXSDFilenames);
274 2 : bool bRet = m_poTmpDS->Open(&oOpenInfo);
275 2 : CSLDestroy(oOpenInfo.papszOpenOptions);
276 2 : oOpenInfo.papszOpenOptions = nullptr;
277 2 : if (!bRet)
278 : {
279 1 : return false;
280 : }
281 : }
282 :
283 18 : GDALDataset *poQueryDS = m_poTmpDS ? m_poTmpDS.get() : m_poSrcDS;
284 :
285 : // No explicit XSD creation option, then we assume that the source
286 : // dataset contains all the metadata layers we need
287 : OGRLayer *poOtherMetadataLayer =
288 18 : poQueryDS->GetLayerByName(szOGR_OTHER_METADATA);
289 18 : if (poOtherMetadataLayer == nullptr)
290 : {
291 1 : CPLError(CE_Failure, CPLE_AppDefined,
292 : "Cannot establish schema since no %s creation option "
293 : "specified and no %s found in source "
294 : "dataset. One of them must be defined.",
295 : szINPUT_XSD_OPTION, szOGR_OTHER_METADATA);
296 1 : return false;
297 : }
298 :
299 17 : m_poLayersMDLayer = poQueryDS->GetLayerByName(szOGR_LAYERS_METADATA);
300 17 : m_poFieldsMDLayer = poQueryDS->GetLayerByName(szOGR_FIELDS_METADATA);
301 17 : m_poLayerRelationshipsLayer =
302 17 : poQueryDS->GetLayerByName(szOGR_LAYER_RELATIONSHIPS);
303 17 : if (m_poLayersMDLayer == nullptr)
304 : {
305 1 : CPLError(CE_Failure, CPLE_AppDefined, "%s not found",
306 : szOGR_LAYERS_METADATA);
307 1 : return false;
308 : }
309 16 : if (m_poFieldsMDLayer == nullptr)
310 : {
311 1 : CPLError(CE_Failure, CPLE_AppDefined, "%s not found",
312 : szOGR_FIELDS_METADATA);
313 1 : return false;
314 : }
315 15 : if (m_poLayerRelationshipsLayer == nullptr)
316 : {
317 1 : CPLError(CE_Failure, CPLE_AppDefined, "%s not found",
318 : szOGR_LAYER_RELATIONSHIPS);
319 1 : return false;
320 : }
321 :
322 28 : std::map<int, CPLString> oMapToUri;
323 28 : std::map<int, CPLString> oMapToLocation;
324 28 : std::map<int, CPLString> oMapToPrefix;
325 129 : for (auto &&poFeature : *poOtherMetadataLayer)
326 : {
327 115 : const char *pszKey = poFeature->GetFieldAsString(szKEY);
328 115 : int i = 0;
329 115 : if (sscanf(pszKey, szNAMESPACE_URI_FMT, &i) == 1 && i > 0)
330 : {
331 26 : oMapToUri[i] = poFeature->GetFieldAsString(szVALUE);
332 : }
333 89 : else if (sscanf(pszKey, szNAMESPACE_LOCATION_FMT, &i) == 1 && i > 0)
334 : {
335 13 : oMapToLocation[i] = poFeature->GetFieldAsString(szVALUE);
336 : }
337 76 : else if (sscanf(pszKey, szNAMESPACE_PREFIX_FMT, &i) == 1 && i > 0)
338 : {
339 26 : oMapToPrefix[i] = poFeature->GetFieldAsString(szVALUE);
340 : }
341 50 : else if (EQUAL(pszKey, szGML_VERSION))
342 : {
343 0 : osGMLVersion = poFeature->GetFieldAsString(szVALUE);
344 : }
345 : }
346 14 : poOtherMetadataLayer->ResetReading();
347 :
348 40 : for (int i = 1; i <= static_cast<int>(oMapToUri.size()); ++i)
349 : {
350 26 : if (oMapToUri.find(i) != oMapToUri.end())
351 : {
352 26 : const CPLString &osURI(oMapToUri[i]);
353 26 : aoXSDs.push_back(PairURIFilename(osURI, oMapToLocation[i]));
354 26 : if (oMapToPrefix.find(i) != oMapToPrefix.end())
355 : {
356 26 : oMapURIToPrefix[osURI] = oMapToPrefix[i];
357 : }
358 : }
359 : }
360 :
361 14 : if (!CollectLayers())
362 1 : return false;
363 :
364 13 : if (!CollectFields())
365 0 : return false;
366 :
367 13 : if (!CollectRelationships())
368 0 : return false;
369 :
370 13 : const char *pszLayers = m_aosOptions.FetchNameValue(szLAYERS_OPTION);
371 13 : if (pszLayers)
372 : {
373 6 : for (const auto &oLayerIter : m_oMapLayerNameToIdx)
374 : {
375 3 : LayerDescription &oDesc = m_aoLayerDesc[oLayerIter.second];
376 3 : oDesc.bIsSelected = false;
377 : }
378 :
379 3 : char **papszLayers = CSLTokenizeString2(pszLayers, ",", 0);
380 5 : for (char **papszIter = papszLayers; *papszIter != nullptr; ++papszIter)
381 : {
382 3 : if (EQUAL(*papszIter, "{SPATIAL_LAYERS}"))
383 : {
384 2 : for (const auto &oLayerIter : m_oMapLayerNameToIdx)
385 : {
386 1 : LayerDescription &oDesc = m_aoLayerDesc[oLayerIter.second];
387 1 : if (oDesc.bIsTopLevel)
388 : {
389 1 : bool bIsGeometric = false;
390 2 : for (const auto &oFieldIter : oDesc.oMapIdxToField)
391 : {
392 2 : if (oFieldIter.second.GetType() ==
393 : GMLAS_FT_GEOMETRY)
394 : {
395 1 : bIsGeometric = true;
396 1 : break;
397 : }
398 : }
399 1 : oDesc.bIsSelected = bIsGeometric;
400 : }
401 : }
402 : }
403 : else
404 : {
405 2 : const auto oLayerIter = m_oMapLayerNameToIdx.find(*papszIter);
406 2 : if (oLayerIter == m_oMapLayerNameToIdx.end())
407 : {
408 1 : CPLError(CE_Warning, CPLE_AppDefined,
409 : "Layer %s specified in LAYERS option "
410 : "does not exist",
411 : *papszIter);
412 1 : CSLDestroy(papszLayers);
413 1 : return false;
414 : }
415 : else
416 : {
417 1 : LayerDescription &oDesc = m_aoLayerDesc[oLayerIter->second];
418 1 : oDesc.bIsSelected = true;
419 : }
420 : }
421 : }
422 2 : CSLDestroy(papszLayers);
423 : }
424 : else
425 : {
426 10 : ComputeTopLevelFIDs();
427 : }
428 :
429 12 : const bool bWFS2FeatureCollection = EQUAL(
430 : m_aosOptions.FetchNameValueDef(szWRAPPING_OPTION, m_oConf.m_osWrapping),
431 : szWFS2_FEATURECOLLECTION);
432 :
433 12 : if (pfnProgress == GDALDummyProgress)
434 12 : pfnProgress = nullptr;
435 : // Compute total number of top level features
436 12 : GIntBig nTotalTopLevelFeatures = -1;
437 12 : if (pfnProgress != nullptr || bWFS2FeatureCollection)
438 : {
439 4 : nTotalTopLevelFeatures = 0;
440 8 : for (const auto &oLayerIter : m_oMapLayerNameToIdx)
441 : {
442 4 : const LayerDescription &oDesc = m_aoLayerDesc[oLayerIter.second];
443 4 : OGRLayer *poSrcLayer = m_poSrcDS->GetLayerByName(oDesc.osName);
444 4 : if (oDesc.bIsSelected && poSrcLayer != nullptr)
445 : {
446 4 : nTotalTopLevelFeatures += poSrcLayer->GetFeatureCount(true);
447 4 : nTotalTopLevelFeatures -=
448 4 : static_cast<GIntBig>(oDesc.aoSetReferencedFIDs.size());
449 : }
450 : }
451 4 : CPLDebug("GMLAS", CPL_FRMT_GIB " top level features to be written",
452 : nTotalTopLevelFeatures);
453 : }
454 :
455 : // Now read options related to writing
456 : int nIndentSize =
457 : std::min(INDENT_SIZE_MAX,
458 : std::max(INDENT_SIZE_MIN,
459 12 : atoi(m_aosOptions.FetchNameValueDef(
460 : szINDENT_SIZE_OPTION,
461 12 : CPLSPrintf("%d", m_oConf.m_nIndentSize)))));
462 12 : m_osIndentation.assign(nIndentSize, ' ');
463 :
464 36 : if (oMapURIToPrefix.find(szGML32_URI) != oMapURIToPrefix.end() ||
465 : // Used by tests
466 24 : oMapURIToPrefix.find("http://fake_gml32") != oMapURIToPrefix.end())
467 : {
468 8 : m_osGMLVersion = "3.2.1";
469 : }
470 : else
471 : {
472 4 : m_osGMLVersion = osGMLVersion;
473 4 : CPL_IGNORE_RET_VAL(osGMLVersion);
474 : }
475 :
476 : m_osSRSNameFormat = m_aosOptions.FetchNameValueDef(
477 12 : szSRSNAME_FORMAT_OPTION, m_oConf.m_osSRSNameFormat);
478 :
479 : CPLString osLineFormat = m_aosOptions.FetchNameValueDef(
480 24 : szLINEFORMAT_OPTION, m_oConf.m_osLineFormat);
481 12 : if (!osLineFormat.empty())
482 : {
483 12 : if (EQUAL(osLineFormat, szCRLF))
484 0 : m_osEOL = "\r\n";
485 12 : else if (EQUAL(osLineFormat, szLF))
486 0 : m_osEOL = "\n";
487 : }
488 :
489 : CPLString osOutXSDFilename =
490 24 : m_aosOptions.FetchNameValueDef(szOUTPUT_XSD_FILENAME_OPTION, "");
491 : const bool bGenerateXSD =
492 20 : !bWFS2FeatureCollection &&
493 20 : (m_osFilename != "/vsistdout/" || !osOutXSDFilename.empty()) &&
494 8 : m_aosOptions.FetchBool(szGENERATE_XSD_OPTION, true);
495 :
496 : // Write .xsd
497 12 : if (bWFS2FeatureCollection)
498 4 : VSIUnlink(CPLResetExtensionSafe(m_osFilename, "xsd").c_str());
499 8 : else if (bGenerateXSD && !WriteXSD(osOutXSDFilename, aoXSDs))
500 1 : return false;
501 :
502 : // Write .xml header
503 11 : if (!WriteXMLHeader(bWFS2FeatureCollection, nTotalTopLevelFeatures,
504 : bGenerateXSD, osOutXSDFilename, aoXSDs,
505 : oMapURIToPrefix))
506 1 : return false;
507 :
508 : // Iterate over layers
509 10 : GIntBig nFeaturesWritten = 0;
510 10 : bool bRet = true;
511 124 : for (const auto &oLayerIter : m_oMapLayerNameToIdx)
512 : {
513 114 : if (m_aoLayerDesc[oLayerIter.second].bIsSelected)
514 : {
515 : bRet =
516 30 : WriteLayer(bWFS2FeatureCollection,
517 30 : m_aoLayerDesc[oLayerIter.second], nFeaturesWritten,
518 : nTotalTopLevelFeatures, pfnProgress, pProgressData);
519 30 : if (!bRet)
520 0 : break;
521 : }
522 : }
523 10 : CPLDebug("GMLAS", CPL_FRMT_GIB " top level features written",
524 : nFeaturesWritten);
525 :
526 : // Epilogue of .xml file
527 10 : if (bWFS2FeatureCollection)
528 : {
529 3 : PrintLine(m_fpXML.get(), "</%s:%s>", szWFS_PREFIX,
530 : szFEATURE_COLLECTION);
531 : }
532 : else
533 : {
534 7 : PrintLine(m_fpXML.get(), "</%s:%s>", m_osTargetNameSpacePrefix.c_str(),
535 : szFEATURE_COLLECTION);
536 : }
537 :
538 10 : Close();
539 10 : return bRet;
540 : }
541 :
542 : /************************************************************************/
543 : /* GetLayerByName() */
544 : /************************************************************************/
545 :
546 : // Mostly equivalent to m_poSrcDS->GetLayerByName(), except that we use
547 : // a map to cache instead of linear search.
548 1077 : OGRLayer *GMLASWriter::GetLayerByName(const CPLString &osName)
549 : {
550 1077 : const auto oIter = m_oMapLayerNameToLayer.find(osName);
551 1077 : if (oIter == m_oMapLayerNameToLayer.end())
552 : {
553 117 : OGRLayer *poLayer = m_poSrcDS->GetLayerByName(osName);
554 117 : m_oMapLayerNameToLayer[osName] = poLayer;
555 117 : return poLayer;
556 : }
557 960 : return oIter->second;
558 : }
559 :
560 : /************************************************************************/
561 : /* XMLEscape() */
562 : /************************************************************************/
563 :
564 399 : static CPLString XMLEscape(const CPLString &osStr)
565 : {
566 399 : char *pszEscaped = CPLEscapeString(osStr, -1, CPLES_XML);
567 399 : CPLString osRet(pszEscaped);
568 399 : CPLFree(pszEscaped);
569 399 : return osRet;
570 : }
571 :
572 : /************************************************************************/
573 : /* WriteXSD() */
574 : /************************************************************************/
575 :
576 8 : bool GMLASWriter::WriteXSD(const CPLString &osXSDFilenameIn,
577 : const std::vector<PairURIFilename> &aoXSDs)
578 : {
579 : const CPLString osXSDFilename(
580 8 : !osXSDFilenameIn.empty()
581 : ? osXSDFilenameIn
582 16 : : CPLString(CPLResetExtensionSafe(m_osFilename, "xsd")));
583 8 : VSILFILE *fpXSD = VSIFOpenL(osXSDFilename, "wb");
584 8 : if (fpXSD == nullptr)
585 : {
586 1 : CPLError(CE_Failure, CPLE_AppDefined, "Cannot create %s",
587 : osXSDFilename.c_str());
588 1 : return false;
589 : }
590 :
591 7 : PrintLine(fpXSD, "<?xml version=\"1.0\" encoding=\"utf-8\" ?>");
592 7 : PrintLine(fpXSD, "<xs:schema ");
593 7 : PrintLine(fpXSD, " targetNamespace=\"%s\"",
594 14 : XMLEscape(m_osTargetNameSpace).c_str());
595 7 : PrintLine(fpXSD, " xmlns:%s=\"%s\"", m_osTargetNameSpacePrefix.c_str(),
596 14 : XMLEscape(m_osTargetNameSpace).c_str());
597 7 : PrintLine(fpXSD, " xmlns:xs=\"%s\"", szXS_URI);
598 7 : PrintLine(fpXSD, " elementFormDefault=\"qualified\" version=\"1.0\" >");
599 :
600 : // Those imports are not really needed, since the schemaLocation are
601 : // already specified in the .xml file, but that helps validating the
602 : // document with libxml2/xmllint since it can only accept one single main
603 : // schema.
604 21 : for (size_t i = 0; i < aoXSDs.size(); ++i)
605 : {
606 14 : if (!aoXSDs[i].second.empty())
607 : {
608 7 : if (!aoXSDs[i].first.empty())
609 : {
610 14 : PrintLine(fpXSD,
611 : "<xs:import namespace=\"%s\" schemaLocation=\"%s\"/>",
612 14 : XMLEscape(aoXSDs[i].first).c_str(),
613 14 : XMLEscape(aoXSDs[i].second).c_str());
614 : }
615 : else
616 : {
617 0 : PrintLine(fpXSD, "<xs:import schemaLocation=\"%s\"/>",
618 0 : XMLEscape(aoXSDs[i].second).c_str());
619 : }
620 : }
621 : }
622 :
623 7 : PrintLine(fpXSD,
624 : "<xs:element name=\"%s\" "
625 : "type=\"%s:%sType\"/>",
626 : szFEATURE_COLLECTION, m_osTargetNameSpacePrefix.c_str(),
627 : szFEATURE_COLLECTION);
628 :
629 7 : PrintLine(fpXSD, "<xs:complexType name=\"%sType\">", szFEATURE_COLLECTION);
630 7 : PrintLine(fpXSD, " <xs:sequence>");
631 7 : PrintLine(fpXSD,
632 : " <xs:element name=\"%s\" "
633 : "minOccurs=\"0\" maxOccurs=\"unbounded\">",
634 : szFEATURE_MEMBER);
635 7 : PrintLine(fpXSD, " <xs:complexType>");
636 7 : PrintLine(fpXSD, " <xs:sequence>");
637 7 : PrintLine(fpXSD, " <xs:any/>");
638 7 : PrintLine(fpXSD, " </xs:sequence>");
639 7 : PrintLine(fpXSD, " </xs:complexType>");
640 7 : PrintLine(fpXSD, " </xs:element>");
641 7 : PrintLine(fpXSD, " </xs:sequence>");
642 7 : PrintLine(fpXSD, "</xs:complexType>");
643 7 : PrintLine(fpXSD, "</xs:schema>");
644 :
645 7 : VSIFCloseL(fpXSD);
646 :
647 7 : return true;
648 : }
649 :
650 : /************************************************************************/
651 : /* WriteXMLHeader() */
652 : /************************************************************************/
653 :
654 11 : bool GMLASWriter::WriteXMLHeader(
655 : bool bWFS2FeatureCollection, GIntBig nTotalFeatures, bool bGenerateXSD,
656 : const CPLString &osXSDFilenameIn,
657 : const std::vector<PairURIFilename> &aoXSDs,
658 : const std::map<CPLString, CPLString> &oMapURIToPrefix)
659 : {
660 11 : m_fpXML.reset(VSIFOpenL(m_osFilename, "wb"));
661 11 : if (m_fpXML == nullptr)
662 : {
663 1 : CPLError(CE_Failure, CPLE_AppDefined, "Cannot create %s",
664 : m_osFilename.c_str());
665 1 : return false;
666 : }
667 :
668 : // Delete potentially existing .gfs file
669 10 : VSIUnlink(CPLResetExtensionSafe(m_osFilename, "gfs").c_str());
670 :
671 20 : std::map<CPLString, CPLString> aoWrittenPrefixes;
672 10 : aoWrittenPrefixes[szXSI_PREFIX] = szXSI_URI;
673 :
674 10 : PrintLine(m_fpXML.get(), "<?xml version=\"1.0\" encoding=\"utf-8\" ?>");
675 10 : if (bWFS2FeatureCollection)
676 : {
677 3 : PrintLine(m_fpXML.get(), "<%s:%s", szWFS_PREFIX, szFEATURE_COLLECTION);
678 :
679 : const CPLString osTimestamp(m_aosOptions.FetchNameValueDef(
680 3 : szTIMESTAMP_OPTION, m_oConf.m_osTimestamp));
681 3 : if (osTimestamp.empty())
682 : {
683 : struct tm sTime;
684 2 : CPLUnixTimeToYMDHMS(time(nullptr), &sTime);
685 2 : PrintLine(m_fpXML.get(),
686 : " timeStamp=\"%04d-%02d-%02dT%02d:%02d:%02dZ\"",
687 2 : sTime.tm_year + 1900, sTime.tm_mon + 1, sTime.tm_mday,
688 : sTime.tm_hour, sTime.tm_min, sTime.tm_sec);
689 : }
690 : else
691 : {
692 1 : PrintLine(m_fpXML.get(), " timeStamp=\"%s\"",
693 : osTimestamp.c_str());
694 : }
695 3 : PrintLine(m_fpXML.get(), " numberMatched=\"unknown\"");
696 3 : PrintLine(m_fpXML.get(), " numberReturned=\"" CPL_FRMT_GIB "\"",
697 : nTotalFeatures);
698 3 : PrintLine(m_fpXML.get(), " xmlns:%s=\"%s\"", szWFS_PREFIX,
699 : szWFS20_URI);
700 3 : aoWrittenPrefixes[szWFS_PREFIX] = szWFS20_URI;
701 : }
702 : else
703 : {
704 7 : PrintLine(m_fpXML.get(), "<%s:%s", m_osTargetNameSpacePrefix.c_str(),
705 : szFEATURE_COLLECTION);
706 7 : PrintLine(m_fpXML.get(), " xmlns:%s=\"%s\"",
707 : m_osTargetNameSpacePrefix.c_str(),
708 14 : XMLEscape(m_osTargetNameSpace).c_str());
709 : }
710 10 : PrintLine(m_fpXML.get(), " xmlns:%s=\"%s\"", szXSI_PREFIX, szXSI_URI);
711 :
712 20 : CPLString osSchemaURI;
713 10 : if (bWFS2FeatureCollection)
714 : {
715 : const CPLString osWFS20SchemaLocation(m_aosOptions.FetchNameValueDef(
716 6 : szWFS20_SCHEMALOCATION_OPTION, m_oConf.m_osWFS20SchemaLocation));
717 3 : osSchemaURI += szWFS20_URI;
718 3 : osSchemaURI += " ";
719 3 : osSchemaURI += osWFS20SchemaLocation;
720 : }
721 7 : else if (bGenerateXSD || !osXSDFilenameIn.empty())
722 : {
723 : const CPLString osXSDFilename(
724 7 : !osXSDFilenameIn.empty()
725 : ? osXSDFilenameIn
726 : : CPLString(CPLGetFilename(
727 14 : CPLResetExtensionSafe(m_osFilename, "xsd").c_str())));
728 7 : osSchemaURI += m_osTargetNameSpace;
729 7 : osSchemaURI += " ";
730 7 : osSchemaURI += osXSDFilename;
731 : }
732 :
733 30 : for (size_t i = 0; i < aoXSDs.size(); ++i)
734 : {
735 20 : const CPLString &osURI(aoXSDs[i].first);
736 20 : const CPLString &osLocation(aoXSDs[i].second);
737 :
738 20 : CPLString osPrefix;
739 20 : if (!osURI.empty())
740 : {
741 20 : const auto oIter = oMapURIToPrefix.find(osURI);
742 20 : if (oIter != oMapURIToPrefix.end())
743 : {
744 20 : osPrefix = oIter->second;
745 : }
746 : }
747 20 : if (!osPrefix.empty())
748 : {
749 20 : const auto &oIter = aoWrittenPrefixes.find(osPrefix);
750 20 : if (oIter != aoWrittenPrefixes.end())
751 : {
752 0 : if (oIter->second != osURI)
753 : {
754 0 : CPLDebug("GMLAS",
755 : "Namespace prefix %s already defined as URI %s "
756 : "but now redefefined as %s. Skipped",
757 0 : osPrefix.c_str(), oIter->second.c_str(),
758 : osURI.c_str());
759 : }
760 0 : continue;
761 : }
762 20 : aoWrittenPrefixes[osPrefix] = osURI;
763 : }
764 :
765 20 : if (osURI.empty())
766 : {
767 0 : if (!osLocation.empty())
768 : {
769 0 : PrintLine(m_fpXML.get(), " xsi:%s=\"%s\"",
770 : szNO_NAMESPACE_SCHEMA_LOCATION,
771 0 : XMLEscape(osLocation).c_str());
772 : }
773 : }
774 : else
775 : {
776 20 : if (osPrefix.empty())
777 : {
778 0 : osPrefix = CPLSPrintf("ns%d", static_cast<int>(i));
779 : }
780 :
781 20 : PrintLine(m_fpXML.get(), " xmlns:%s=\"%s\"", osPrefix.c_str(),
782 40 : XMLEscape(osURI).c_str());
783 :
784 20 : if (!osLocation.empty())
785 : {
786 10 : if (!osSchemaURI.empty())
787 10 : osSchemaURI += " ";
788 10 : osSchemaURI += osURI;
789 10 : osSchemaURI += " ";
790 10 : osSchemaURI += osLocation;
791 : }
792 : }
793 : }
794 :
795 10 : if (!osSchemaURI.empty())
796 : {
797 10 : PrintLine(m_fpXML.get(), " xsi:%s=\"%s\" >", szSCHEMA_LOCATION,
798 20 : XMLEscape(osSchemaURI).c_str());
799 : }
800 :
801 : // Write optional user comment
802 : CPLString osComment(
803 10 : m_aosOptions.FetchNameValueDef(szCOMMENT_OPTION, m_oConf.m_osComment));
804 10 : if (!osComment.empty())
805 : {
806 : while (true)
807 : {
808 3 : const size_t nSizeBefore = osComment.size();
809 3 : osComment.replaceAll("--", "- -");
810 3 : if (nSizeBefore == osComment.size())
811 1 : break;
812 2 : }
813 1 : PrintLine(m_fpXML.get(), "<!-- %s -->", osComment.c_str());
814 : }
815 :
816 10 : return true;
817 : }
818 :
819 : /************************************************************************/
820 : /* CollectLayers() */
821 : /************************************************************************/
822 :
823 14 : bool GMLASWriter::CollectLayers()
824 : {
825 14 : OGRFeatureDefn *poFDefn = m_poLayersMDLayer->GetLayerDefn();
826 14 : const char *const apszFields[] = {szLAYER_NAME, szLAYER_XPATH,
827 : szLAYER_CATEGORY, szLAYER_PKID_NAME,
828 : szLAYER_PARENT_PKID_NAME};
829 79 : for (size_t i = 0; i < CPL_ARRAYSIZE(apszFields); ++i)
830 : {
831 66 : if (poFDefn->GetFieldIndex(apszFields[i]) < 0)
832 : {
833 2 : CPLError(CE_Failure, CPLE_AppDefined,
834 1 : "Cannot find field %s in %s layer", apszFields[i],
835 1 : m_poLayersMDLayer->GetName());
836 1 : return false;
837 : }
838 : }
839 :
840 13 : m_poLayersMDLayer->SetAttributeFilter(nullptr);
841 13 : m_poLayersMDLayer->ResetReading();
842 130 : for (auto &&poFeature : *m_poLayersMDLayer)
843 : {
844 117 : LayerDescription desc;
845 117 : desc.osName = poFeature->GetFieldAsString(szLAYER_NAME);
846 117 : desc.osXPath = poFeature->GetFieldAsString(szLAYER_XPATH);
847 117 : desc.osPKIDName = poFeature->GetFieldAsString(szLAYER_PKID_NAME);
848 : desc.osParentPKIDName =
849 117 : poFeature->GetFieldAsString(szLAYER_PARENT_PKID_NAME);
850 117 : desc.bIsTopLevel = EQUAL(poFeature->GetFieldAsString(szLAYER_CATEGORY),
851 : szTOP_LEVEL_ELEMENT);
852 117 : desc.bIsSelected = desc.bIsTopLevel;
853 117 : desc.bIsJunction = EQUAL(poFeature->GetFieldAsString(szLAYER_CATEGORY),
854 : szJUNCTION_TABLE);
855 :
856 117 : OGRLayer *poLyr = GetLayerByName(desc.osName);
857 117 : if (poLyr)
858 : {
859 117 : if (!desc.osPKIDName.empty())
860 101 : desc.oMapFieldNameToOGRIdx[desc.osPKIDName] =
861 101 : poLyr->GetLayerDefn()->GetFieldIndex(desc.osPKIDName);
862 117 : if (!desc.osParentPKIDName.empty())
863 64 : desc.oMapFieldNameToOGRIdx[desc.osParentPKIDName] =
864 64 : poLyr->GetLayerDefn()->GetFieldIndex(desc.osParentPKIDName);
865 : }
866 :
867 117 : m_aoLayerDesc.push_back(desc);
868 117 : if (m_oMapLayerNameToIdx.find(desc.osName) !=
869 234 : m_oMapLayerNameToIdx.end())
870 : {
871 0 : CPLError(CE_Failure, CPLE_AppDefined,
872 : "Several layers with same %s = %s", szLAYER_NAME,
873 : desc.osName.c_str());
874 0 : return false;
875 : }
876 218 : if (!desc.bIsJunction &&
877 218 : m_oMapXPathToIdx.find(desc.osXPath) != m_oMapXPathToIdx.end())
878 : {
879 0 : CPLError(CE_Failure, CPLE_AppDefined,
880 : "Several layers with same %s = %s", szLAYER_XPATH,
881 : desc.osXPath.c_str());
882 0 : return false;
883 : }
884 117 : const int nIdx = static_cast<int>(m_aoLayerDesc.size() - 1);
885 117 : m_oMapLayerNameToIdx[desc.osName] = nIdx;
886 117 : if (!desc.bIsJunction)
887 101 : m_oMapXPathToIdx[desc.osXPath] = nIdx;
888 : }
889 13 : m_poLayersMDLayer->ResetReading();
890 :
891 13 : return true;
892 : }
893 :
894 : /************************************************************************/
895 : /* CollectFields() */
896 : /************************************************************************/
897 :
898 13 : bool GMLASWriter::CollectFields()
899 : {
900 13 : OGRFeatureDefn *poFDefn = m_poFieldsMDLayer->GetLayerDefn();
901 13 : const char *const apszFields[] = {
902 : szLAYER_NAME, szFIELD_INDEX,
903 : szFIELD_NAME, szFIELD_TYPE,
904 : szFIELD_XPATH, szFIELD_CATEGORY,
905 : szFIELD_RELATED_LAYER, szFIELD_JUNCTION_LAYER,
906 : szFIELD_IS_LIST, szFIELD_MIN_OCCURS,
907 : szFIELD_MAX_OCCURS, szFIELD_REPETITION_ON_SEQUENCE,
908 : szFIELD_DEFAULT_VALUE};
909 182 : for (size_t i = 0; i < CPL_ARRAYSIZE(apszFields); ++i)
910 : {
911 169 : if (poFDefn->GetFieldIndex(apszFields[i]) < 0)
912 : {
913 0 : CPLError(CE_Failure, CPLE_AppDefined,
914 0 : "Cannot find field %s in %s layer", apszFields[i],
915 0 : m_poFieldsMDLayer->GetName());
916 0 : return false;
917 : }
918 : }
919 :
920 13 : m_poFieldsMDLayer->SetAttributeFilter(
921 26 : (CPLString(szFIELD_CATEGORY) + " != '" + szSWE_FIELD + "'").c_str());
922 13 : m_poFieldsMDLayer->ResetReading();
923 677 : for (auto &&poFeature : m_poFieldsMDLayer)
924 : {
925 664 : GMLASField oField;
926 :
927 664 : oField.SetName(poFeature->GetFieldAsString(szFIELD_NAME));
928 :
929 664 : CPLString osLayerName(poFeature->GetFieldAsString(szLAYER_NAME));
930 664 : const auto &oIterToIdx = m_oMapLayerNameToIdx.find(osLayerName);
931 664 : if (oIterToIdx == m_oMapLayerNameToIdx.end())
932 : {
933 : // Shouldn't happen for well behaved metadata
934 0 : CPLError(CE_Warning, CPLE_AppDefined,
935 : "Cannot find in %s layer %s, "
936 : "referenced in %s by field %s",
937 : szOGR_LAYERS_METADATA, osLayerName.c_str(),
938 0 : szOGR_FIELDS_METADATA, oField.GetName().c_str());
939 0 : continue;
940 : }
941 664 : if (m_aoLayerDesc[oIterToIdx->second].bIsJunction)
942 : {
943 0 : continue;
944 : }
945 :
946 664 : CPLString osXPath(poFeature->GetFieldAsString(szFIELD_XPATH));
947 664 : oField.SetXPath(osXPath);
948 :
949 664 : CPLString osType(poFeature->GetFieldAsString(szFIELD_TYPE));
950 664 : if (!osType.empty())
951 : {
952 584 : if (osType == szFAKEXS_JSON_DICT)
953 20 : oField.SetType(GMLAS_FT_STRING, osType);
954 564 : else if (osType == szFAKEXS_GEOMETRY)
955 : {
956 135 : oField.SetType(GMLAS_FT_GEOMETRY, osType);
957 : // Hack for geometry field that have a xpath like
958 : // foo/bar/gml:Point,foo/bar/gml:LineString,...
959 135 : size_t nPos = osXPath.find("/gml:Point,");
960 135 : if (nPos != std::string::npos)
961 0 : osXPath.resize(nPos);
962 135 : oField.SetXPath(osXPath);
963 : }
964 : else
965 429 : oField.SetType(GMLASField::GetTypeFromString(osType), osType);
966 : }
967 :
968 664 : CPLString osCategory(poFeature->GetFieldAsString(szFIELD_CATEGORY));
969 664 : if (osCategory == szREGULAR)
970 : {
971 568 : oField.SetCategory(GMLASField::REGULAR);
972 : }
973 96 : else if (osCategory == szPATH_TO_CHILD_ELEMENT_NO_LINK)
974 : {
975 56 : oField.SetCategory(GMLASField::PATH_TO_CHILD_ELEMENT_NO_LINK);
976 : }
977 40 : else if (osCategory == szPATH_TO_CHILD_ELEMENT_WITH_LINK)
978 : {
979 16 : oField.SetCategory(GMLASField::PATH_TO_CHILD_ELEMENT_WITH_LINK);
980 : }
981 24 : else if (osCategory == szPATH_TO_CHILD_ELEMENT_WITH_JUNCTION_TABLE)
982 : {
983 16 : oField.SetCategory(
984 : GMLASField::PATH_TO_CHILD_ELEMENT_WITH_JUNCTION_TABLE);
985 :
986 : CPLString osJunctionLayer(
987 16 : poFeature->GetFieldAsString(szFIELD_JUNCTION_LAYER));
988 16 : if (osJunctionLayer.empty())
989 : {
990 : // Shouldn't happen for well behaved metadata
991 0 : CPLError(CE_Warning, CPLE_AppDefined,
992 : "Missing value for %s for field (%s,%s)",
993 : szFIELD_JUNCTION_LAYER, osLayerName.c_str(),
994 0 : oField.GetName().c_str());
995 0 : continue;
996 : }
997 16 : oField.SetJunctionLayer(osJunctionLayer);
998 : }
999 8 : else if (osCategory == szGROUP)
1000 : {
1001 8 : oField.SetCategory(GMLASField::GROUP);
1002 : }
1003 : else
1004 : {
1005 : // Shouldn't happen for well behaved metadata
1006 0 : CPLError(CE_Warning, CPLE_AppDefined,
1007 : "Unknown category = %s for field (%s,%s)",
1008 : osCategory.c_str(), osLayerName.c_str(),
1009 0 : oField.GetName().c_str());
1010 0 : continue;
1011 : }
1012 :
1013 : CPLString osRelatedLayer(
1014 664 : poFeature->GetFieldAsString(szFIELD_RELATED_LAYER));
1015 760 : if (!osRelatedLayer.empty() &&
1016 96 : m_oMapLayerNameToIdx.find(osRelatedLayer) !=
1017 760 : m_oMapLayerNameToIdx.end())
1018 : {
1019 96 : oField.SetRelatedClassXPath(
1020 96 : m_aoLayerDesc[m_oMapLayerNameToIdx[osRelatedLayer]].osXPath);
1021 : }
1022 :
1023 664 : oField.SetList(
1024 664 : CPL_TO_BOOL(poFeature->GetFieldAsInteger(szFIELD_IS_LIST)));
1025 :
1026 664 : oField.SetMinOccurs(poFeature->GetFieldAsInteger(szFIELD_MIN_OCCURS));
1027 664 : oField.SetMaxOccurs(poFeature->GetFieldAsInteger(szFIELD_MAX_OCCURS));
1028 664 : oField.SetRepetitionOnSequence(CPL_TO_BOOL(
1029 : poFeature->GetFieldAsInteger(szFIELD_REPETITION_ON_SEQUENCE)));
1030 664 : oField.SetDefaultValue(
1031 : poFeature->GetFieldAsString(szFIELD_DEFAULT_VALUE));
1032 :
1033 664 : const int nIdx = poFeature->GetFieldAsInteger(szFIELD_INDEX);
1034 :
1035 664 : const int nLayerIdx = m_oMapLayerNameToIdx[osLayerName];
1036 664 : LayerDescription &oLayerDesc = m_aoLayerDesc[nLayerIdx];
1037 664 : if (oLayerDesc.oMapIdxToField.find(nIdx) !=
1038 1328 : oLayerDesc.oMapIdxToField.end())
1039 : {
1040 : // Shouldn't happen for well behaved metadata
1041 0 : CPLError(CE_Failure, CPLE_AppDefined,
1042 : "Field %s of %s has the same index as field %s",
1043 0 : oField.GetName().c_str(), osLayerName.c_str(),
1044 0 : oLayerDesc.oMapIdxToField[nIdx].GetName().c_str());
1045 0 : return false;
1046 : }
1047 664 : oLayerDesc.oMapIdxToField[nIdx] = oField;
1048 :
1049 664 : if (!oField.GetXPath().empty())
1050 : {
1051 656 : if (oLayerDesc.oMapFieldXPathToIdx.find(oField.GetXPath()) !=
1052 1312 : oLayerDesc.oMapFieldXPathToIdx.end())
1053 : {
1054 : // Shouldn't happen for well behaved metadata
1055 0 : CPLError(
1056 : CE_Failure, CPLE_AppDefined,
1057 : "Field %s of %s has the same XPath as field %s",
1058 0 : oField.GetName().c_str(), osLayerName.c_str(),
1059 : oLayerDesc
1060 : .oMapIdxToField
1061 0 : [oLayerDesc.oMapFieldXPathToIdx[oField.GetXPath()]]
1062 0 : .GetName()
1063 : .c_str());
1064 0 : return false;
1065 : }
1066 656 : oLayerDesc.oMapFieldXPathToIdx[oField.GetXPath()] = nIdx;
1067 : }
1068 :
1069 664 : OGRLayer *poLyr = GetLayerByName(osLayerName);
1070 664 : if (poLyr)
1071 : {
1072 664 : oLayerDesc.oMapFieldNameToOGRIdx[oField.GetName()] =
1073 664 : poLyr->GetLayerDefn()->GetFieldIndex(oField.GetName());
1074 664 : if (oField.GetType() == GMLAS_FT_GEOMETRY)
1075 : {
1076 135 : oLayerDesc.oMapFieldNameToOGRIdx[oField.GetName() + "_xml"] =
1077 270 : poLyr->GetLayerDefn()->GetFieldIndex(
1078 270 : (oField.GetName() + "_xml").c_str());
1079 : }
1080 : }
1081 : }
1082 13 : m_poFieldsMDLayer->ResetReading();
1083 :
1084 13 : return true;
1085 : }
1086 :
1087 : /************************************************************************/
1088 : /* CollectRelationships() */
1089 : /************************************************************************/
1090 :
1091 13 : bool GMLASWriter::CollectRelationships()
1092 : {
1093 13 : OGRFeatureDefn *poFDefn = m_poLayerRelationshipsLayer->GetLayerDefn();
1094 13 : const char *const apszFields[] = {szPARENT_LAYER, szCHILD_LAYER,
1095 : szPARENT_ELEMENT_NAME};
1096 52 : for (size_t i = 0; i < CPL_ARRAYSIZE(apszFields); ++i)
1097 : {
1098 39 : if (poFDefn->GetFieldIndex(apszFields[i]) < 0)
1099 : {
1100 0 : CPLError(CE_Failure, CPLE_AppDefined,
1101 0 : "Cannot find field %s in %s layer", apszFields[i],
1102 0 : m_poLayerRelationshipsLayer->GetName());
1103 0 : return false;
1104 : }
1105 : }
1106 :
1107 13 : m_poLayerRelationshipsLayer->SetAttributeFilter(nullptr);
1108 13 : m_poLayerRelationshipsLayer->ResetReading();
1109 :
1110 109 : for (auto &&poFeature : m_poLayerRelationshipsLayer)
1111 : {
1112 : const CPLString osParentLayer(
1113 96 : poFeature->GetFieldAsString(szPARENT_LAYER));
1114 96 : if (m_oMapLayerNameToIdx.find(osParentLayer) ==
1115 192 : m_oMapLayerNameToIdx.end())
1116 : {
1117 : // Shouldn't happen for well behaved metadata
1118 0 : CPLError(CE_Warning, CPLE_AppDefined,
1119 : "Cannot find in %s layer %s, referenced in %s",
1120 : szOGR_LAYERS_METADATA, osParentLayer.c_str(),
1121 : szOGR_LAYER_RELATIONSHIPS);
1122 0 : continue;
1123 : }
1124 :
1125 : const CPLString osChildLayer(
1126 96 : poFeature->GetFieldAsString(szCHILD_LAYER));
1127 96 : if (m_oMapLayerNameToIdx.find(osChildLayer) ==
1128 192 : m_oMapLayerNameToIdx.end())
1129 : {
1130 : // Shouldn't happen for well behaved metadata
1131 0 : CPLError(CE_Warning, CPLE_AppDefined,
1132 : "Cannot find in %s layer %s, referenced in %s",
1133 : szOGR_LAYERS_METADATA, osChildLayer.c_str(),
1134 : szOGR_LAYER_RELATIONSHIPS);
1135 0 : continue;
1136 : }
1137 :
1138 96 : const int nChildLayerIdx = m_oMapLayerNameToIdx[osChildLayer];
1139 96 : if (m_aoLayerDesc[nChildLayerIdx].bIsTopLevel)
1140 : {
1141 : const CPLString osReferencingField(
1142 24 : poFeature->GetFieldAsString(szPARENT_ELEMENT_NAME));
1143 :
1144 48 : m_aoLayerDesc[nChildLayerIdx].aoReferencingLayers.push_back(
1145 48 : PairLayerNameColName(osParentLayer, osReferencingField));
1146 : }
1147 : }
1148 13 : m_poLayerRelationshipsLayer->ResetReading();
1149 :
1150 13 : return true;
1151 : }
1152 :
1153 : /************************************************************************/
1154 : /* ComputeTopLevelFIDs() */
1155 : /* */
1156 : /* Find which features of top-level layers are referenced by other */
1157 : /* features, in which case we don't need to emit them in their layer */
1158 : /************************************************************************/
1159 :
1160 10 : void GMLASWriter::ComputeTopLevelFIDs()
1161 : {
1162 124 : for (size_t i = 0; i < m_aoLayerDesc.size(); ++i)
1163 : {
1164 114 : LayerDescription &oDesc = m_aoLayerDesc[i];
1165 114 : OGRLayer *poLayer = GetLayerByName(oDesc.osName);
1166 144 : if (oDesc.bIsTopLevel && poLayer != nullptr &&
1167 30 : !oDesc.aoReferencingLayers.empty())
1168 : {
1169 44 : for (size_t j = 0; j < oDesc.aoReferencingLayers.size(); ++j)
1170 : {
1171 48 : CPLString osSQL;
1172 48 : CPLString osFID("FID");
1173 48 : if (poLayer->GetFIDColumn() &&
1174 24 : !EQUAL(poLayer->GetFIDColumn(), ""))
1175 : {
1176 24 : osFID = poLayer->GetFIDColumn();
1177 : }
1178 :
1179 : // Determine if the referencing field points to a junction
1180 : // table
1181 : const auto oIter = m_oMapLayerNameToIdx.find(
1182 24 : oDesc.aoReferencingLayers[j].first);
1183 24 : if (oIter != m_oMapLayerNameToIdx.end())
1184 : {
1185 : const LayerDescription &oReferencingLayerDesc =
1186 24 : m_aoLayerDesc[oIter->second];
1187 1212 : for (const auto &oIterField :
1188 2448 : oReferencingLayerDesc.oMapIdxToField)
1189 : {
1190 1236 : const GMLASField &oField = oIterField.second;
1191 1236 : if (oField.GetName() ==
1192 1236 : oDesc.aoReferencingLayers[j].second)
1193 : {
1194 24 : if (oField.GetCategory() ==
1195 : GMLASField::
1196 : PATH_TO_CHILD_ELEMENT_WITH_JUNCTION_TABLE)
1197 : {
1198 : osSQL.Printf(
1199 : "SELECT s.\"%s\" AS ogr_main_fid "
1200 : "FROM \"%s\" s "
1201 : "JOIN \"%s\" j ON j.%s = s.\"%s\"",
1202 : osFID.c_str(), oDesc.osName.c_str(),
1203 8 : oField.GetJunctionLayer().c_str(),
1204 16 : szCHILD_PKID, oDesc.osPKIDName.c_str());
1205 : }
1206 24 : break;
1207 : }
1208 : }
1209 : }
1210 :
1211 : // Otherwise we can use the referencing (layer_name,
1212 : // field_name) tuple directly.
1213 24 : if (osSQL.empty())
1214 : {
1215 : osSQL.Printf("SELECT s.\"%s\" AS ogr_main_fid "
1216 : "FROM \"%s\" s "
1217 : "JOIN \"%s\" m ON m.\"%s\" = s.\"%s\"",
1218 : osFID.c_str(), oDesc.osName.c_str(),
1219 16 : oDesc.aoReferencingLayers[j].first.c_str(),
1220 16 : oDesc.aoReferencingLayers[j].second.c_str(),
1221 48 : oDesc.osPKIDName.c_str());
1222 : }
1223 :
1224 24 : CPLDebug("GMLAS", "Executing %s", osSQL.c_str());
1225 : OGRLayer *poSQLLyr =
1226 24 : m_poSrcDS->ExecuteSQL(osSQL, nullptr, nullptr);
1227 24 : if (poSQLLyr)
1228 : {
1229 68 : for (auto &&poFeature : *poSQLLyr)
1230 : {
1231 44 : const GIntBig nFID = poFeature->GetFieldAsInteger64(0);
1232 44 : oDesc.aoSetReferencedFIDs.insert(nFID);
1233 : }
1234 24 : m_poSrcDS->ReleaseResultSet(poSQLLyr);
1235 : }
1236 : }
1237 : }
1238 : }
1239 10 : }
1240 :
1241 : /************************************************************************/
1242 : /* SplitXPath() */
1243 : /************************************************************************/
1244 :
1245 : // Decompose a XPath ns1:foo1/@ns2:foo2/... in its components
1246 : // [ (ns1,foo1), (ns2,@foo2), ... ]
1247 600 : static XPathComponents SplitXPathInternal(const CPLString &osXPath)
1248 : {
1249 600 : char **papszTokens = CSLTokenizeString2(osXPath, "/", 0);
1250 600 : XPathComponents aoComponents;
1251 1946 : for (int i = 0; papszTokens[i] != nullptr; ++i)
1252 : {
1253 1346 : const bool bAttr = (papszTokens[i][0] == '@');
1254 : char **papszNSElt =
1255 1346 : CSLTokenizeString2(papszTokens[i] + (bAttr ? 1 : 0), ":", 0);
1256 1346 : if (papszNSElt[0] != nullptr && papszNSElt[1] != nullptr &&
1257 1238 : papszNSElt[2] == nullptr)
1258 : {
1259 1238 : CPLString osVal(papszNSElt[1]);
1260 1238 : size_t nPos = osVal.find(szEXTRA_SUFFIX);
1261 1238 : if (nPos != std::string::npos)
1262 4 : osVal.resize(nPos);
1263 1238 : aoComponents.push_back(PairNSElement(
1264 2476 : papszNSElt[0], (bAttr ? CPLString("@") : CPLString()) + osVal));
1265 : }
1266 108 : else if (papszNSElt[0] != nullptr && papszNSElt[1] == nullptr)
1267 : {
1268 108 : CPLString osVal(papszNSElt[0]);
1269 108 : size_t nPos = osVal.find(szEXTRA_SUFFIX);
1270 108 : if (nPos != std::string::npos)
1271 0 : osVal.resize(nPos);
1272 108 : aoComponents.push_back(PairNSElement(
1273 216 : "", (bAttr ? CPLString("@") : CPLString()) + osVal));
1274 : }
1275 1346 : CSLDestroy(papszNSElt);
1276 : }
1277 600 : CSLDestroy(papszTokens);
1278 600 : return aoComponents;
1279 : }
1280 :
1281 988 : const XPathComponents &GMLASWriter::SplitXPath(const CPLString &osXPath)
1282 : {
1283 988 : const auto oIter = m_oMapXPathToComponents.find(osXPath);
1284 988 : if (oIter != m_oMapXPathToComponents.end())
1285 388 : return oIter->second;
1286 :
1287 600 : m_oMapXPathToComponents[osXPath] = SplitXPathInternal(osXPath);
1288 600 : return m_oMapXPathToComponents[osXPath];
1289 : }
1290 :
1291 : /************************************************************************/
1292 : /* IsAttr() */
1293 : /************************************************************************/
1294 :
1295 3138 : static bool IsAttr(const PairNSElement &pair)
1296 : {
1297 3138 : return !pair.second.empty() && pair.second[0] == '@';
1298 : }
1299 :
1300 : /************************************************************************/
1301 : /* MakeXPath() */
1302 : /************************************************************************/
1303 :
1304 1462 : static CPLString MakeXPath(const PairNSElement &pair)
1305 : {
1306 1462 : if (pair.first.empty())
1307 : {
1308 64 : if (IsAttr(pair))
1309 128 : return pair.second.substr(1);
1310 : else
1311 0 : return pair.second;
1312 : }
1313 1398 : else if (IsAttr(pair))
1314 36 : return pair.first + ":" + pair.second.substr(1);
1315 : else
1316 2760 : return pair.first + ":" + pair.second;
1317 : }
1318 :
1319 : /************************************************************************/
1320 : /* WriteLayer() */
1321 : /************************************************************************/
1322 :
1323 30 : bool GMLASWriter::WriteLayer(bool bWFS2FeatureCollection,
1324 : const LayerDescription &oDesc,
1325 : GIntBig &nFeaturesWritten,
1326 : GIntBig nTotalTopLevelFeatures,
1327 : GDALProgressFunc pfnProgress, void *pProgressData)
1328 : {
1329 30 : OGRLayer *poSrcLayer = GetLayerByName(oDesc.osName);
1330 30 : if (poSrcLayer == nullptr)
1331 0 : return true;
1332 :
1333 30 : poSrcLayer->ResetReading();
1334 30 : IncIndent();
1335 30 : std::set<CPLString> oSetLayersInIteration;
1336 30 : oSetLayersInIteration.insert(oDesc.osName);
1337 30 : bool bRet = true;
1338 63 : for (auto &&poFeature : *poSrcLayer)
1339 : {
1340 33 : if (oDesc.aoSetReferencedFIDs.find(poFeature->GetFID()) ==
1341 66 : oDesc.aoSetReferencedFIDs.end())
1342 : {
1343 10 : PrintIndent(m_fpXML.get());
1344 10 : if (bWFS2FeatureCollection)
1345 : {
1346 3 : PrintLine(m_fpXML.get(), "<%s:%s>", szWFS_PREFIX, szMEMBER);
1347 : }
1348 : else
1349 : {
1350 7 : PrintLine(m_fpXML.get(), "<%s:%s>",
1351 : m_osTargetNameSpacePrefix.c_str(), szFEATURE_MEMBER);
1352 : }
1353 :
1354 10 : bRet = WriteFeature(poFeature.get(), oDesc, oSetLayersInIteration,
1355 20 : XPathComponents(), XPathComponents(), 0);
1356 :
1357 10 : PrintIndent(m_fpXML.get());
1358 10 : if (bWFS2FeatureCollection)
1359 : {
1360 3 : PrintLine(m_fpXML.get(), "</%s:%s>", szWFS_PREFIX, szMEMBER);
1361 : }
1362 : else
1363 : {
1364 7 : PrintLine(m_fpXML.get(), "</%s:%s>",
1365 : m_osTargetNameSpacePrefix.c_str(), szFEATURE_MEMBER);
1366 : }
1367 :
1368 10 : if (bRet)
1369 : {
1370 10 : nFeaturesWritten++;
1371 10 : const double dfPct = static_cast<double>(nFeaturesWritten) /
1372 : nTotalTopLevelFeatures;
1373 10 : if (pfnProgress && !pfnProgress(dfPct, "", pProgressData))
1374 : {
1375 0 : bRet = false;
1376 : }
1377 : }
1378 10 : if (!bRet)
1379 0 : break;
1380 : }
1381 : }
1382 30 : poSrcLayer->ResetReading();
1383 30 : DecIndent();
1384 :
1385 30 : return bRet;
1386 : }
1387 :
1388 : /************************************************************************/
1389 : /* FindCommonPrefixLength() */
1390 : /************************************************************************/
1391 :
1392 1070 : static size_t FindCommonPrefixLength(const XPathComponents &a,
1393 : const XPathComponents &b)
1394 : {
1395 1070 : size_t i = 0;
1396 2588 : for (; i < a.size() && i < b.size(); ++i)
1397 : {
1398 1980 : if (a[i].first != b[i].first || a[i].second != b[i].second)
1399 462 : break;
1400 : }
1401 1070 : return i;
1402 : }
1403 :
1404 : /************************************************************************/
1405 : /* WriteClosingTags() */
1406 : /************************************************************************/
1407 :
1408 906 : void GMLASWriter::WriteClosingTags(size_t nCommonLength,
1409 : const XPathComponents &aoCurComponents,
1410 : const XPathComponents &aoNewComponents,
1411 : bool bCurIsRegularField,
1412 : bool bNewIsRegularField)
1413 : {
1414 906 : if (nCommonLength < aoCurComponents.size())
1415 : {
1416 608 : bool bFieldIsAnotherAttrOfCurElt = false;
1417 608 : size_t i = aoCurComponents.size() - 1;
1418 :
1419 608 : bool bMustIndent = !bCurIsRegularField;
1420 :
1421 608 : if (IsAttr(aoCurComponents.back()))
1422 : {
1423 160 : if (nCommonLength + 1 == aoCurComponents.size() &&
1424 160 : nCommonLength + 1 == aoNewComponents.size() &&
1425 46 : IsAttr(aoNewComponents.back()))
1426 : {
1427 24 : bFieldIsAnotherAttrOfCurElt = true;
1428 : }
1429 : else
1430 : {
1431 : /*
1432 : a/@b cur
1433 : a new
1434 : ==> <a b="">foo</a>
1435 :
1436 : a/@b cur
1437 : a/c new
1438 : ==> <a b="">
1439 : <c/>
1440 : </a>
1441 :
1442 : a/@b cur
1443 : c new
1444 : ==> <a b=""/>
1445 : <c/>
1446 :
1447 : */
1448 66 : if ((nCommonLength == 0 ||
1449 132 : nCommonLength + 2 <= aoCurComponents.size()) &&
1450 : i >= 2)
1451 : {
1452 20 : PrintLine(m_fpXML.get(), " />");
1453 20 : i -= 2;
1454 20 : DecIndent();
1455 20 : bMustIndent = true;
1456 : }
1457 : else
1458 : {
1459 46 : VSIFPrintfL(m_fpXML.get(), ">");
1460 46 : CPLAssert(i > 0);
1461 46 : i--;
1462 : // Print a new line except in the <elt attr="foo">bar</elt>
1463 : // situation
1464 116 : if (!(nCommonLength + 1 == aoCurComponents.size() &&
1465 46 : nCommonLength == aoNewComponents.size() &&
1466 24 : bNewIsRegularField))
1467 : {
1468 30 : PrintLine(m_fpXML.get(), "%s", "");
1469 : }
1470 : }
1471 : }
1472 : }
1473 :
1474 608 : if (!bFieldIsAnotherAttrOfCurElt)
1475 : {
1476 1178 : for (; i >= nCommonLength; --i)
1477 : {
1478 624 : if (bMustIndent)
1479 : {
1480 150 : PrintIndent(m_fpXML.get());
1481 : }
1482 624 : bMustIndent = true;
1483 624 : PrintLine(m_fpXML.get(), "</%s>",
1484 1248 : MakeXPath(aoCurComponents[i]).c_str());
1485 624 : DecIndent();
1486 624 : if (i == 0)
1487 30 : break;
1488 : }
1489 : }
1490 : }
1491 906 : }
1492 :
1493 : /************************************************************************/
1494 : /* WriteClosingAndStartingTags() */
1495 : /************************************************************************/
1496 :
1497 120 : void GMLASWriter::WriteClosingAndStartingTags(
1498 : const XPathComponents &aoCurComponents,
1499 : const XPathComponents &aoNewComponents, bool bCurIsRegularField)
1500 : {
1501 :
1502 : const size_t nCommonLength =
1503 120 : FindCommonPrefixLength(aoCurComponents, aoNewComponents);
1504 :
1505 120 : WriteClosingTags(nCommonLength, aoCurComponents, aoNewComponents,
1506 : bCurIsRegularField, false);
1507 180 : for (size_t i = nCommonLength; i < aoNewComponents.size(); ++i)
1508 : {
1509 60 : IncIndent();
1510 60 : PrintIndent(m_fpXML.get());
1511 60 : PrintLine(m_fpXML.get(), "<%s>", MakeXPath(aoNewComponents[i]).c_str());
1512 : }
1513 120 : }
1514 :
1515 : /************************************************************************/
1516 : /* WriteFeature() */
1517 : /************************************************************************/
1518 :
1519 222 : bool GMLASWriter::WriteFeature(OGRFeature *poFeature,
1520 : const LayerDescription &oLayerDesc,
1521 : const std::set<CPLString> &oSetLayersInIteration,
1522 : const XPathComponents &aoInitialComponents,
1523 : const XPathComponents &aoPrefixComponents,
1524 : int nRecLevel)
1525 : {
1526 222 : if (nRecLevel == 100)
1527 : {
1528 0 : CPLError(CE_Failure, CPLE_NotSupported,
1529 : "WriteFeature() called with 100 levels of recursion");
1530 0 : return false;
1531 : }
1532 :
1533 444 : XPathComponents aoCurComponents(aoInitialComponents);
1534 444 : XPathComponents aoLayerComponents;
1535 222 : bool bAtLeastOneFieldWritten = false;
1536 222 : bool bCurIsRegularField = false;
1537 1086 : for (const auto &oIter : oLayerDesc.oMapIdxToField)
1538 : {
1539 864 : const GMLASField &oField = oIter.second;
1540 864 : const GMLASField::Category eCategory(oField.GetCategory());
1541 864 : if (eCategory == GMLASField::REGULAR)
1542 : {
1543 728 : WriteFieldRegular(poFeature, oField, oLayerDesc,
1544 : /*aoLayerComponents, */
1545 : aoCurComponents, aoPrefixComponents,
1546 : /*oSetLayersInIteration,*/
1547 : bAtLeastOneFieldWritten, bCurIsRegularField);
1548 : }
1549 136 : else if (eCategory == GMLASField::PATH_TO_CHILD_ELEMENT_NO_LINK ||
1550 : eCategory == GMLASField::GROUP)
1551 : {
1552 84 : if (!WriteFieldNoLink(
1553 : poFeature, oField, oLayerDesc, aoLayerComponents,
1554 : aoCurComponents, aoPrefixComponents, oSetLayersInIteration,
1555 : nRecLevel, bAtLeastOneFieldWritten, bCurIsRegularField))
1556 : {
1557 0 : return false;
1558 : }
1559 : }
1560 52 : else if (eCategory == GMLASField::PATH_TO_CHILD_ELEMENT_WITH_LINK)
1561 : {
1562 36 : if (!WriteFieldWithLink(
1563 : poFeature, oField, oLayerDesc, aoLayerComponents,
1564 : aoCurComponents, aoPrefixComponents, oSetLayersInIteration,
1565 : nRecLevel, bAtLeastOneFieldWritten, bCurIsRegularField))
1566 : {
1567 0 : return false;
1568 : }
1569 : }
1570 16 : else if (eCategory ==
1571 : GMLASField::PATH_TO_CHILD_ELEMENT_WITH_JUNCTION_TABLE)
1572 : {
1573 16 : if (!WriteFieldJunctionTable(
1574 : poFeature, oField, oLayerDesc, aoLayerComponents,
1575 : aoCurComponents, aoPrefixComponents, oSetLayersInIteration,
1576 : nRecLevel, bAtLeastOneFieldWritten, bCurIsRegularField))
1577 : {
1578 0 : return false;
1579 : }
1580 : }
1581 : }
1582 :
1583 222 : if (!bAtLeastOneFieldWritten && aoInitialComponents.empty() &&
1584 0 : !oLayerDesc.osXPath.empty())
1585 : {
1586 0 : aoLayerComponents = SplitXPath(oLayerDesc.osXPath);
1587 0 : const CPLString osLayerElt(MakeXPath(aoLayerComponents.back()));
1588 0 : PrintIndent(m_fpXML.get());
1589 0 : VSIFPrintfL(m_fpXML.get(), "%s", m_osIndentation.c_str());
1590 0 : PrintLine(m_fpXML.get(), "<%s />", osLayerElt.c_str());
1591 : }
1592 : else
1593 : {
1594 : const size_t nCommonLength =
1595 222 : FindCommonPrefixLength(aoCurComponents, aoInitialComponents);
1596 222 : WriteClosingTags(nCommonLength, aoCurComponents, aoInitialComponents,
1597 : bCurIsRegularField, false);
1598 : }
1599 :
1600 222 : return true;
1601 : }
1602 :
1603 : /************************************************************************/
1604 : /* PrintMultipleValuesSeparator() */
1605 : /************************************************************************/
1606 :
1607 100 : void GMLASWriter::PrintMultipleValuesSeparator(
1608 : const GMLASField &oField, const XPathComponents &aoFieldComponents)
1609 : {
1610 100 : if (oField.IsList())
1611 : {
1612 44 : VSIFPrintfL(m_fpXML.get(), " ");
1613 : }
1614 : else
1615 : {
1616 56 : PrintLine(m_fpXML.get(), "</%s>",
1617 112 : MakeXPath(aoFieldComponents.back()).c_str());
1618 56 : PrintIndent(m_fpXML.get());
1619 56 : VSIFPrintfL(m_fpXML.get(), "<%s>",
1620 112 : MakeXPath(aoFieldComponents.back()).c_str());
1621 : }
1622 100 : }
1623 :
1624 : /************************************************************************/
1625 : /* PrintXMLDouble() */
1626 : /************************************************************************/
1627 :
1628 40 : static void PrintXMLDouble(VSILFILE *fp, double dfVal)
1629 : {
1630 40 : if (std::isinf(dfVal))
1631 : {
1632 0 : if (dfVal > 0)
1633 0 : VSIFPrintfL(fp, "INF");
1634 : else
1635 0 : VSIFPrintfL(fp, "-INF");
1636 : }
1637 40 : else if (std::isnan(dfVal))
1638 0 : VSIFPrintfL(fp, "NaN");
1639 : else
1640 40 : VSIFPrintfL(fp, "%.16g", dfVal);
1641 40 : }
1642 :
1643 : /************************************************************************/
1644 : /* AreGeomsEqualAxisOrderInsensitive() */
1645 : /************************************************************************/
1646 :
1647 13 : static bool AreGeomsEqualAxisOrderInsensitive(const OGRGeometry *poGeomRef,
1648 : OGRGeometry *poGeomModifiable)
1649 : {
1650 13 : if (poGeomRef->Equals(poGeomModifiable))
1651 11 : return true;
1652 2 : poGeomModifiable->swapXY();
1653 2 : return CPL_TO_BOOL(poGeomRef->Equals(poGeomModifiable));
1654 : }
1655 :
1656 : /************************************************************************/
1657 : /* GetCoordSwap() */
1658 : /************************************************************************/
1659 :
1660 0 : bool GMLASWriter::GetCoordSwap(const OGRSpatialReference *poSRS)
1661 : {
1662 0 : const auto oIter = m_oMapSRSToCoordSwap.find(poSRS);
1663 0 : if (oIter != m_oMapSRSToCoordSwap.end())
1664 0 : return oIter->second;
1665 :
1666 0 : bool bCoordSwap = false;
1667 0 : if (m_osSRSNameFormat != "SHORT")
1668 : {
1669 0 : const auto &map = poSRS->GetDataAxisToSRSAxisMapping();
1670 0 : if (map.size() >= 2 && map[0] == 2 && map[1] == 1)
1671 : {
1672 0 : bCoordSwap = true;
1673 : }
1674 : }
1675 0 : m_oMapSRSToCoordSwap[poSRS] = bCoordSwap;
1676 0 : return bCoordSwap;
1677 : }
1678 :
1679 : /************************************************************************/
1680 : /* WriteFieldRegular() */
1681 : /************************************************************************/
1682 :
1683 728 : bool GMLASWriter::WriteFieldRegular(
1684 : OGRFeature *poFeature, const GMLASField &oField,
1685 : const LayerDescription &oLayerDesc,
1686 : /*XPathComponents& aoLayerComponents,*/
1687 : XPathComponents &aoCurComponents, const XPathComponents &aoPrefixComponents,
1688 : /*const std::set<CPLString>& oSetLayersInIteration,*/
1689 : bool &bAtLeastOneFieldWritten, bool &bCurIsRegularField)
1690 : {
1691 728 : const bool bIsGeometryField = oField.GetTypeName() == szFAKEXS_GEOMETRY;
1692 : const int nFieldIdx =
1693 818 : bIsGeometryField ?
1694 : // Some drivers may not store the geometry field name,
1695 : // so for a feature with a single geometry, use it
1696 90 : (poFeature->GetGeomFieldCount() == 1
1697 90 : ? 0
1698 90 : : poFeature->GetGeomFieldIndex(oField.GetName()))
1699 638 : : oLayerDesc.GetOGRIdxFromFieldName(oField.GetName());
1700 1456 : XPathComponents aoFieldComponents = SplitXPath(oField.GetXPath());
1701 728 : aoFieldComponents.insert(aoFieldComponents.begin(),
1702 : aoPrefixComponents.begin(),
1703 1456 : aoPrefixComponents.end());
1704 :
1705 : // For extension/* case
1706 728 : if (!aoFieldComponents.empty() && aoFieldComponents.back().second == "*")
1707 : {
1708 8 : aoFieldComponents.pop_back();
1709 : }
1710 :
1711 : const size_t nCommonLength =
1712 728 : FindCommonPrefixLength(aoCurComponents, aoFieldComponents);
1713 :
1714 : const bool bEmptyContent =
1715 1456 : nFieldIdx < 0 ||
1716 90 : ((bIsGeometryField && !poFeature->GetGeomFieldRef(nFieldIdx)) ||
1717 716 : (!bIsGeometryField && !poFeature->IsFieldSetAndNotNull(nFieldIdx)));
1718 : const bool bIsNull =
1719 728 : m_oConf.m_bUseNullState && (!bIsGeometryField && nFieldIdx >= 0 &&
1720 0 : poFeature->IsFieldNull(nFieldIdx));
1721 728 : bool bMustBeEmittedEvenIfEmpty = oField.GetMinOccurs() > 0 || bIsNull;
1722 728 : if (!m_oConf.m_bUseNullState && oField.GetMinOccurs() == 0 &&
1723 268 : bEmptyContent && nCommonLength + 1 == aoCurComponents.size() &&
1724 116 : IsAttr(aoCurComponents.back()) &&
1725 1464 : nCommonLength == aoFieldComponents.size() &&
1726 732 : oLayerDesc.oMapFieldXPathToIdx.find(oField.GetXPath() + "/" +
1727 736 : szAT_XSI_NIL) ==
1728 732 : oLayerDesc.oMapFieldXPathToIdx.end())
1729 : {
1730 : // This is quite tricky to determine if a <foo bar="baz"/> node is
1731 : // valid or if we must add a xsi:nil="true" to make it valid
1732 : // For now assume that a string can be empty
1733 0 : if (oField.GetType() != GMLAS_FT_STRING)
1734 0 : bMustBeEmittedEvenIfEmpty = true;
1735 : }
1736 :
1737 728 : if (bEmptyContent && !bMustBeEmittedEvenIfEmpty)
1738 160 : return true;
1739 :
1740 : // Do not emit optional attributes at default/fixed value
1741 808 : if (!aoFieldComponents.empty() && oField.GetMinOccurs() == 0 &&
1742 240 : IsAttr(aoFieldComponents.back()))
1743 : {
1744 86 : const CPLString &osDefaultVal(!oField.GetDefaultValue().empty()
1745 86 : ? oField.GetDefaultValue()
1746 78 : : oField.GetFixedValue());
1747 86 : if (!osDefaultVal.empty())
1748 : {
1749 8 : if (oField.GetType() == GMLAS_FT_BOOLEAN)
1750 : {
1751 0 : const int nVal = poFeature->GetFieldAsInteger(nFieldIdx);
1752 0 : if (osDefaultVal == "false" && nVal == 0)
1753 0 : return true;
1754 0 : if (osDefaultVal == "true" && nVal == 1)
1755 0 : return true;
1756 : }
1757 8 : else if (osDefaultVal == poFeature->GetFieldAsString(nFieldIdx))
1758 : {
1759 4 : return true;
1760 : }
1761 : }
1762 : }
1763 :
1764 564 : bAtLeastOneFieldWritten = true;
1765 :
1766 4 : if (bEmptyContent && nCommonLength + 1 == aoCurComponents.size() &&
1767 568 : IsAttr(aoCurComponents.back()) &&
1768 0 : nCommonLength == aoFieldComponents.size())
1769 : {
1770 : // Particular case for <a foo="bar" xsi:nil="true"/>
1771 0 : VSIFPrintfL(m_fpXML.get(), " xsi:nil=\"true\">");
1772 0 : aoCurComponents = aoFieldComponents;
1773 0 : bCurIsRegularField = true;
1774 0 : return true;
1775 : }
1776 : else
1777 : {
1778 : // Emit closing tags
1779 564 : WriteClosingTags(nCommonLength, aoCurComponents, aoFieldComponents,
1780 564 : bCurIsRegularField, true);
1781 : }
1782 :
1783 : // Emit opening tags and attribute names
1784 : // We may do a 0 iteration in case of returning from an attribute
1785 : // to its element
1786 564 : bool bWriteEltContent = true;
1787 1238 : for (size_t i = nCommonLength; i < aoFieldComponents.size(); ++i)
1788 : {
1789 674 : if (i + 1 == aoFieldComponents.size() && IsAttr(aoFieldComponents[i]))
1790 : {
1791 90 : if (aoFieldComponents[i].second != szAT_ANY_ATTR)
1792 : {
1793 82 : VSIFPrintfL(m_fpXML.get(),
1794 164 : " %s=", MakeXPath(aoFieldComponents[i]).c_str());
1795 82 : bWriteEltContent = false;
1796 : }
1797 : }
1798 : else
1799 : {
1800 584 : if (i > nCommonLength)
1801 60 : PrintLine(m_fpXML.get(), "%s", "");
1802 584 : IncIndent();
1803 584 : PrintIndent(m_fpXML.get());
1804 :
1805 710 : if (i + 2 == aoFieldComponents.size() &&
1806 126 : IsAttr(aoFieldComponents[i + 1]))
1807 : {
1808 : // Are we an element that is going to have an
1809 : // attribute ?
1810 66 : VSIFPrintfL(m_fpXML.get(), "<%s",
1811 132 : MakeXPath(aoFieldComponents[i]).c_str());
1812 : }
1813 : else
1814 : {
1815 : // Are we a regular element ?
1816 518 : if (bEmptyContent)
1817 : {
1818 4 : VSIFPrintfL(m_fpXML.get(), "<%s xsi:nil=\"true\">",
1819 8 : MakeXPath(aoFieldComponents[i]).c_str());
1820 : }
1821 : else
1822 : {
1823 514 : VSIFPrintfL(m_fpXML.get(), "<%s>",
1824 1028 : MakeXPath(aoFieldComponents[i]).c_str());
1825 : }
1826 : }
1827 : }
1828 : }
1829 :
1830 : // Write content
1831 564 : if (!bWriteEltContent)
1832 82 : VSIFPrintfL(m_fpXML.get(), "\"");
1833 :
1834 564 : if (!bEmptyContent && oField.GetTypeName() == szFAKEXS_JSON_DICT)
1835 : {
1836 8 : json_object *poObj = nullptr;
1837 8 : if (OGRJSonParse(poFeature->GetFieldAsString(nFieldIdx), &poObj))
1838 : {
1839 8 : if (json_type_object == json_object_get_type(poObj))
1840 : {
1841 : json_object_iter it;
1842 8 : it.key = nullptr;
1843 8 : it.val = nullptr;
1844 8 : it.entry = nullptr;
1845 16 : json_object_object_foreachC(poObj, it)
1846 : {
1847 16 : if (it.val != nullptr &&
1848 8 : json_object_get_type(it.val) == json_type_string)
1849 : {
1850 8 : VSIFPrintfL(
1851 : m_fpXML.get(), " %s=\"%s\"", it.key,
1852 16 : XMLEscape(json_object_get_string(it.val)).c_str());
1853 : }
1854 : }
1855 : }
1856 8 : json_object_put(poObj);
1857 : }
1858 : }
1859 556 : else if (!bEmptyContent && bIsGeometryField)
1860 : {
1861 78 : bool bWriteOGRGeom = true;
1862 78 : OGRGeometry *poGeom = poFeature->GetGeomFieldRef(nFieldIdx);
1863 :
1864 : // In case the original GML string was saved, fetch it and compare it
1865 : // to the current OGR geometry. If they match (in a axis order
1866 : // insensitive way), then use the original GML string
1867 : const int nFieldXMLIdx =
1868 78 : oLayerDesc.GetOGRIdxFromFieldName(oField.GetName() + "_xml");
1869 78 : if (nFieldXMLIdx >= 0 && poFeature->IsFieldSetAndNotNull(nFieldXMLIdx))
1870 : {
1871 13 : if (poFeature->GetFieldDefnRef(nFieldXMLIdx)->GetType() ==
1872 : OFTStringList)
1873 : {
1874 1 : if (wkbFlatten(poGeom->getGeometryType()) ==
1875 : wkbGeometryCollection)
1876 : {
1877 2 : OGRGeometryCollection oGC;
1878 : char **papszValues =
1879 1 : poFeature->GetFieldAsStringList(nFieldXMLIdx);
1880 4 : for (int j = 0;
1881 4 : papszValues != nullptr && papszValues[j] != nullptr;
1882 : ++j)
1883 : {
1884 3 : OGRGeometry *poPart = OGRGeometry::FromHandle(
1885 3 : OGR_G_CreateFromGML(papszValues[j]));
1886 3 : if (poPart)
1887 3 : oGC.addGeometryDirectly(poPart);
1888 : }
1889 1 : if (AreGeomsEqualAxisOrderInsensitive(poGeom, &oGC))
1890 : {
1891 4 : for (int j = 0; papszValues != nullptr &&
1892 4 : papszValues[j] != nullptr;
1893 : ++j)
1894 : {
1895 3 : if (j > 0)
1896 2 : PrintMultipleValuesSeparator(oField,
1897 : aoFieldComponents);
1898 3 : VSIFPrintfL(m_fpXML.get(), "%s", papszValues[j]);
1899 : }
1900 1 : bWriteOGRGeom = false;
1901 : }
1902 : }
1903 : }
1904 : else
1905 : {
1906 12 : const char *pszXML = poFeature->GetFieldAsString(nFieldXMLIdx);
1907 : auto poOrigGeom = std::unique_ptr<OGRGeometry>(
1908 24 : OGRGeometry::FromHandle(OGR_G_CreateFromGML(pszXML)));
1909 :
1910 12 : if (poOrigGeom != nullptr)
1911 : {
1912 12 : if (AreGeomsEqualAxisOrderInsensitive(poGeom,
1913 : poOrigGeom.get()))
1914 : {
1915 12 : VSIFPrintfL(m_fpXML.get(), "%s", pszXML);
1916 12 : bWriteOGRGeom = false;
1917 : }
1918 : }
1919 : }
1920 : }
1921 :
1922 78 : if (bWriteOGRGeom)
1923 : {
1924 130 : CPLString osExtraElt;
1925 65 : bool bGMLSurface311 = false;
1926 65 : bool bGMLCurve311 = false;
1927 65 : bool bGMLPoint311 = false;
1928 65 : if (m_osGMLVersion == "3.1.1" &&
1929 65 : MakeXPath(aoFieldComponents.back()) == "gml:Surface")
1930 : {
1931 0 : bGMLSurface311 = true;
1932 : }
1933 65 : else if (m_osGMLVersion == "3.1.1" &&
1934 65 : MakeXPath(aoFieldComponents.back()) == "gml:Curve")
1935 : {
1936 0 : bGMLCurve311 = true;
1937 : }
1938 65 : else if (m_osGMLVersion == "3.1.1" &&
1939 65 : MakeXPath(aoFieldComponents.back()) == "gml:Point")
1940 : {
1941 0 : bGMLPoint311 = true;
1942 : }
1943 :
1944 : const double dfGMLVersion =
1945 65 : m_osGMLVersion.empty() ? 3.2 : CPLAtof(m_osGMLVersion);
1946 65 : char **papszOptions = CSLSetNameValue(
1947 : nullptr, "FORMAT",
1948 65 : (dfGMLVersion >= 2.0 && dfGMLVersion < 3.0) ? "GML2"
1949 65 : : (dfGMLVersion >= 3.0 && dfGMLVersion < 3.2) ? "GML3"
1950 : : "GML32");
1951 65 : papszOptions = CSLSetNameValue(papszOptions, "SRSNAME_FORMAT",
1952 : m_osSRSNameFormat);
1953 :
1954 65 : if (dfGMLVersion < 3.0)
1955 : {
1956 0 : bool bSwap = false;
1957 : const OGRSpatialReference *poSRS =
1958 0 : poGeom->getSpatialReference();
1959 0 : if (poSRS != nullptr && GetCoordSwap(poSRS))
1960 0 : bSwap = true;
1961 0 : papszOptions = CSLSetNameValue(papszOptions, "COORD_SWAP",
1962 : bSwap ? "TRUE" : "FALSE");
1963 : }
1964 :
1965 70 : if (oField.GetMaxOccurs() > 1 &&
1966 5 : wkbFlatten(poGeom->getGeometryType()) == wkbGeometryCollection)
1967 : {
1968 5 : OGRGeometryCollection *poGC = poGeom->toGeometryCollection();
1969 20 : for (int j = 0; j < poGC->getNumGeometries(); ++j)
1970 : {
1971 15 : if (dfGMLVersion >= 3.2)
1972 : {
1973 : CPLString osGMLID =
1974 15 : poFeature->GetFieldAsString(oLayerDesc.osPKIDName);
1975 15 : osGMLID += CPLSPrintf(".geom%d.%d", nFieldIdx, j);
1976 : papszOptions =
1977 15 : CSLSetNameValue(papszOptions, "GMLID", osGMLID);
1978 : }
1979 15 : if (j > 0)
1980 10 : PrintMultipleValuesSeparator(oField, aoFieldComponents);
1981 15 : char *pszGML = OGR_G_ExportToGMLEx(
1982 : OGRGeometry::ToHandle(poGC->getGeometryRef(j)),
1983 : papszOptions);
1984 15 : if (pszGML)
1985 15 : VSIFPrintfL(m_fpXML.get(), "%s", pszGML);
1986 15 : CPLFree(pszGML);
1987 : }
1988 : }
1989 : else
1990 : {
1991 60 : if (dfGMLVersion >= 3.2)
1992 : {
1993 : CPLString osGMLID =
1994 60 : poFeature->GetFieldAsString(oLayerDesc.osPKIDName);
1995 60 : osGMLID += CPLSPrintf(".geom%d", nFieldIdx);
1996 : papszOptions =
1997 60 : CSLSetNameValue(papszOptions, "GMLID", osGMLID);
1998 : }
1999 60 : char *pszGML = OGR_G_ExportToGMLEx(
2000 : OGRGeometry::ToHandle(poGeom), papszOptions);
2001 60 : if (pszGML)
2002 : {
2003 60 : if (bGMLSurface311 && STARTS_WITH(pszGML, "<gml:Polygon>"))
2004 : {
2005 0 : char *pszEnd = strstr(pszGML, "</gml:Polygon>");
2006 0 : if (pszEnd)
2007 : {
2008 0 : *pszEnd = '\0';
2009 0 : VSIFPrintfL(m_fpXML.get(),
2010 : "<gml:patches><gml:PolygonPatch>%s"
2011 : "</gml:PolygonPatch></gml:patches>",
2012 : pszGML + strlen("<gml:Polygon>"));
2013 0 : }
2014 : }
2015 60 : else if (bGMLCurve311 &&
2016 0 : STARTS_WITH(pszGML, "<gml:LineString>"))
2017 : {
2018 0 : char *pszEnd = strstr(pszGML, "</gml:LineString>");
2019 0 : if (pszEnd)
2020 : {
2021 0 : *pszEnd = '\0';
2022 0 : VSIFPrintfL(
2023 : m_fpXML.get(),
2024 : "<gml:segments><gml:LineStringSegment>%s"
2025 : "</gml:LineStringSegment></gml:segments>",
2026 : pszGML + strlen("<gml:LineString>"));
2027 0 : }
2028 : }
2029 60 : else if (bGMLPoint311 && STARTS_WITH(pszGML, "<gml:Point>"))
2030 : {
2031 0 : char *pszEnd = strstr(pszGML, "</gml:Point>");
2032 0 : if (pszEnd)
2033 : {
2034 0 : *pszEnd = '\0';
2035 0 : VSIFPrintfL(m_fpXML.get(), "%s",
2036 : pszGML + strlen("<gml:Point>"));
2037 0 : }
2038 : }
2039 : else
2040 : {
2041 60 : VSIFPrintfL(m_fpXML.get(), "%s", pszGML);
2042 : }
2043 : }
2044 60 : CPLFree(pszGML);
2045 : }
2046 65 : CSLDestroy(papszOptions);
2047 78 : }
2048 : }
2049 478 : else if (!bEmptyContent && oField.GetTypeName() == szXS_ANY_TYPE)
2050 : {
2051 48 : CPLString osXML(poFeature->GetFieldAsString(nFieldIdx));
2052 : // Check that the content is valid XML
2053 72 : CPLString osValidatingXML("<X>" + osXML + "</X>");
2054 24 : CPLXMLNode *psNode = CPLParseXMLString(osValidatingXML);
2055 24 : if (psNode != nullptr)
2056 : {
2057 24 : VSIFPrintfL(m_fpXML.get(), "%s", osXML.c_str());
2058 24 : CPLDestroyXMLNode(psNode);
2059 : }
2060 : else
2061 : {
2062 : // Otherwise consider it as text and escape
2063 0 : VSIFPrintfL(m_fpXML.get(), "%s", XMLEscape(osXML).c_str());
2064 : }
2065 : }
2066 454 : else if (!bEmptyContent)
2067 : {
2068 : const OGRFieldType eOGRType(
2069 450 : poFeature->GetFieldDefnRef(nFieldIdx)->GetType());
2070 450 : switch (oField.GetType())
2071 : {
2072 20 : case GMLAS_FT_BOOLEAN:
2073 : {
2074 20 : if ((oField.GetMaxOccurs() > 1 || oField.IsList()) &&
2075 : eOGRType == OFTIntegerList)
2076 : {
2077 8 : int nCount = 0;
2078 : const int *panValues =
2079 8 : poFeature->GetFieldAsIntegerList(nFieldIdx, &nCount);
2080 24 : for (int j = 0; j < nCount; ++j)
2081 : {
2082 16 : if (j > 0)
2083 8 : PrintMultipleValuesSeparator(oField,
2084 : aoFieldComponents);
2085 16 : VSIFPrintfL(m_fpXML.get(),
2086 16 : panValues[j] ? "true" : "false");
2087 : }
2088 : }
2089 : else
2090 : {
2091 12 : VSIFPrintfL(m_fpXML.get(),
2092 12 : poFeature->GetFieldAsInteger(nFieldIdx)
2093 : ? "true"
2094 : : "false");
2095 : }
2096 20 : break;
2097 : }
2098 :
2099 76 : case GMLAS_FT_DATETIME:
2100 : case GMLAS_FT_DATE:
2101 : case GMLAS_FT_TIME:
2102 : {
2103 76 : if (eOGRType == OFTDateTime || eOGRType == OFTDate ||
2104 : eOGRType == OFTTime)
2105 : {
2106 : char *pszFormatted =
2107 76 : OGRGetXMLDateTime(poFeature->GetRawFieldRef(nFieldIdx));
2108 76 : char *pszT = strchr(pszFormatted, 'T');
2109 76 : if (oField.GetType() == GMLAS_FT_TIME && pszT != nullptr)
2110 : {
2111 4 : VSIFPrintfL(m_fpXML.get(), "%s", pszT + 1);
2112 : }
2113 : else
2114 : {
2115 72 : if (oField.GetType() == GMLAS_FT_DATE)
2116 : {
2117 4 : if (pszT)
2118 4 : *pszT = '\0';
2119 : }
2120 72 : VSIFPrintfL(m_fpXML.get(), "%s", pszFormatted);
2121 : }
2122 76 : VSIFree(pszFormatted);
2123 : }
2124 : else
2125 : {
2126 0 : CPLError(CE_Warning, CPLE_AppDefined,
2127 : "Invalid content for field %s of type %s: %s",
2128 0 : oField.GetName().c_str(),
2129 0 : oField.GetTypeName().c_str(),
2130 : poFeature->GetFieldAsString(nFieldIdx));
2131 : }
2132 76 : break;
2133 : }
2134 :
2135 4 : case GMLAS_FT_BASE64BINARY:
2136 : {
2137 4 : if (eOGRType == OFTBinary)
2138 : {
2139 4 : int nCount = 0;
2140 : GByte *pabyContent =
2141 4 : poFeature->GetFieldAsBinary(nFieldIdx, &nCount);
2142 4 : char *pszBase64 = CPLBase64Encode(nCount, pabyContent);
2143 4 : VSIFPrintfL(m_fpXML.get(), "%s", pszBase64);
2144 4 : CPLFree(pszBase64);
2145 : }
2146 : else
2147 : {
2148 0 : CPLError(CE_Warning, CPLE_AppDefined,
2149 : "Invalid content for field %s of type %s: %s",
2150 0 : oField.GetName().c_str(),
2151 0 : oField.GetTypeName().c_str(),
2152 : poFeature->GetFieldAsString(nFieldIdx));
2153 : }
2154 4 : break;
2155 : }
2156 :
2157 4 : case GMLAS_FT_HEXBINARY:
2158 : {
2159 4 : if (eOGRType == OFTBinary)
2160 : {
2161 4 : int nCount = 0;
2162 : GByte *pabyContent =
2163 4 : poFeature->GetFieldAsBinary(nFieldIdx, &nCount);
2164 40 : for (int i = 0; i < nCount; ++i)
2165 36 : VSIFPrintfL(m_fpXML.get(), "%02X", pabyContent[i]);
2166 : }
2167 : else
2168 : {
2169 0 : CPLError(CE_Warning, CPLE_AppDefined,
2170 : "Invalid content for field %s of type %s: %s",
2171 0 : oField.GetName().c_str(),
2172 0 : oField.GetTypeName().c_str(),
2173 : poFeature->GetFieldAsString(nFieldIdx));
2174 : }
2175 4 : break;
2176 : }
2177 :
2178 346 : default:
2179 : {
2180 382 : if ((oField.GetMaxOccurs() > 1 || oField.IsList()) &&
2181 36 : (eOGRType == OFTStringList || eOGRType == OFTRealList ||
2182 8 : eOGRType == OFTIntegerList ||
2183 : eOGRType == OFTInteger64List))
2184 : {
2185 64 : if (eOGRType == OFTStringList)
2186 : {
2187 : char **papszValues =
2188 28 : poFeature->GetFieldAsStringList(nFieldIdx);
2189 84 : for (int j = 0; papszValues != nullptr &&
2190 84 : papszValues[j] != nullptr;
2191 : ++j)
2192 : {
2193 56 : if (j > 0)
2194 28 : PrintMultipleValuesSeparator(oField,
2195 : aoFieldComponents);
2196 56 : VSIFPrintfL(m_fpXML.get(), "%s",
2197 112 : XMLEscape(papszValues[j]).c_str());
2198 : }
2199 : }
2200 36 : else if (eOGRType == OFTRealList)
2201 : {
2202 12 : int nCount = 0;
2203 : const double *padfValues =
2204 12 : poFeature->GetFieldAsDoubleList(nFieldIdx, &nCount);
2205 40 : for (int j = 0; j < nCount; ++j)
2206 : {
2207 28 : if (j > 0)
2208 16 : PrintMultipleValuesSeparator(oField,
2209 : aoFieldComponents);
2210 28 : PrintXMLDouble(m_fpXML.get(), padfValues[j]);
2211 : }
2212 : }
2213 24 : else if (eOGRType == OFTIntegerList)
2214 : {
2215 16 : int nCount = 0;
2216 16 : const int *panValues = poFeature->GetFieldAsIntegerList(
2217 : nFieldIdx, &nCount);
2218 56 : for (int j = 0; j < nCount; ++j)
2219 : {
2220 40 : if (j > 0)
2221 24 : PrintMultipleValuesSeparator(oField,
2222 : aoFieldComponents);
2223 40 : VSIFPrintfL(m_fpXML.get(), "%d", panValues[j]);
2224 : }
2225 : }
2226 8 : else if (eOGRType == OFTInteger64List)
2227 : {
2228 8 : int nCount = 0;
2229 : const GIntBig *panValues =
2230 8 : poFeature->GetFieldAsInteger64List(nFieldIdx,
2231 : &nCount);
2232 28 : for (int j = 0; j < nCount; ++j)
2233 : {
2234 20 : if (j > 0)
2235 12 : PrintMultipleValuesSeparator(oField,
2236 : aoFieldComponents);
2237 20 : VSIFPrintfL(m_fpXML.get(), CPL_FRMT_GIB,
2238 20 : panValues[j]);
2239 : }
2240 : }
2241 : }
2242 282 : else if (eOGRType == OFTReal)
2243 : {
2244 12 : PrintXMLDouble(m_fpXML.get(),
2245 : poFeature->GetFieldAsDouble(nFieldIdx));
2246 : }
2247 : else
2248 : {
2249 270 : VSIFPrintfL(
2250 : m_fpXML.get(), "%s",
2251 540 : XMLEscape(poFeature->GetFieldAsString(nFieldIdx))
2252 : .c_str());
2253 : }
2254 346 : break;
2255 : }
2256 : }
2257 : }
2258 :
2259 564 : if (!bWriteEltContent)
2260 82 : VSIFPrintfL(m_fpXML.get(), "\"");
2261 :
2262 564 : aoCurComponents = std::move(aoFieldComponents);
2263 564 : bCurIsRegularField = true;
2264 :
2265 564 : return true;
2266 : }
2267 :
2268 : /************************************************************************/
2269 : /* WriteFieldNoLink() */
2270 : /************************************************************************/
2271 :
2272 84 : bool GMLASWriter::WriteFieldNoLink(
2273 : OGRFeature *poFeature, const GMLASField &oField,
2274 : const LayerDescription &oLayerDesc, XPathComponents &aoLayerComponents,
2275 : XPathComponents &aoCurComponents, const XPathComponents &aoPrefixComponents,
2276 : const std::set<CPLString> &oSetLayersInIteration, int nRecLevel,
2277 : bool &bAtLeastOneFieldWritten, bool &bCurIsRegularField)
2278 : {
2279 84 : const auto oIter = m_oMapXPathToIdx.find(oField.GetRelatedClassXPath());
2280 84 : if (oIter == m_oMapXPathToIdx.end())
2281 : {
2282 : // Not necessary to be more verbose in case of truncated
2283 : // source dataset
2284 0 : CPLDebug("GMLAS", "No child layer of %s matching xpath = %s",
2285 : oLayerDesc.osName.c_str(),
2286 0 : oField.GetRelatedClassXPath().c_str());
2287 0 : return true;
2288 : }
2289 :
2290 84 : const LayerDescription &oChildLayerDesc = m_aoLayerDesc[oIter->second];
2291 84 : OGRLayer *poRelLayer = GetLayerByName(oChildLayerDesc.osName);
2292 84 : if (poRelLayer == nullptr)
2293 : {
2294 : // Not necessary to be more verbose in case of truncated
2295 : // source dataset
2296 0 : CPLDebug("GMLAS", "Child layer %s of %s not found",
2297 : oChildLayerDesc.osName.c_str(), oLayerDesc.osName.c_str());
2298 0 : return true;
2299 : }
2300 :
2301 84 : if (oLayerDesc.osPKIDName.empty())
2302 : {
2303 0 : CPLError(CE_Failure, CPLE_AppDefined, "Missing %s for layer %s",
2304 : szLAYER_PKID_NAME, oLayerDesc.osName.c_str());
2305 0 : return true;
2306 : }
2307 : int nParentPKIDIdx;
2308 84 : if ((nParentPKIDIdx =
2309 84 : oLayerDesc.GetOGRIdxFromFieldName(oLayerDesc.osPKIDName)) < 0)
2310 : {
2311 0 : CPLError(CE_Failure, CPLE_AppDefined,
2312 : "Cannot find field %s in layer %s",
2313 : oLayerDesc.osPKIDName.c_str(), oLayerDesc.osName.c_str());
2314 0 : return true;
2315 : }
2316 84 : if (!poFeature->IsFieldSetAndNotNull(nParentPKIDIdx))
2317 : {
2318 0 : CPLError(CE_Failure, CPLE_AppDefined,
2319 : "Missing value of %s field for feature " CPL_FRMT_GIB
2320 : " of layer %s",
2321 : oLayerDesc.osPKIDName.c_str(), poFeature->GetFID(),
2322 : oLayerDesc.osName.c_str());
2323 0 : return true;
2324 : }
2325 84 : if (oChildLayerDesc.osParentPKIDName.empty())
2326 : {
2327 0 : CPLError(CE_Failure, CPLE_AppDefined, "Missing %s for layer %s",
2328 : szLAYER_PARENT_PKID_NAME, oChildLayerDesc.osName.c_str());
2329 : }
2330 84 : if (oSetLayersInIteration.find(oChildLayerDesc.osName) !=
2331 168 : oSetLayersInIteration.end())
2332 : {
2333 0 : CPLDebug("GMLAS", "Unexpected at line %d", __LINE__);
2334 0 : return true;
2335 : }
2336 :
2337 168 : std::set<CPLString> oSetLayersInIterationSub(oSetLayersInIteration);
2338 84 : oSetLayersInIterationSub.insert(oChildLayerDesc.osName);
2339 :
2340 84 : if (aoLayerComponents.empty())
2341 : {
2342 64 : aoLayerComponents = SplitXPath(oLayerDesc.osXPath);
2343 64 : aoLayerComponents.insert(aoLayerComponents.begin(),
2344 : aoPrefixComponents.begin(),
2345 128 : aoPrefixComponents.end());
2346 : }
2347 :
2348 168 : XPathComponents aoFieldComponents = SplitXPath(oField.GetXPath());
2349 84 : aoFieldComponents.insert(aoFieldComponents.begin(),
2350 : aoPrefixComponents.begin(),
2351 168 : aoPrefixComponents.end());
2352 :
2353 168 : CPLString osParentPKID(poFeature->GetFieldAsString(nParentPKIDIdx));
2354 84 : poRelLayer->SetAttributeFilter(
2355 : CPLSPrintf("%s = '%s'", oChildLayerDesc.osParentPKIDName.c_str(),
2356 84 : osParentPKID.c_str()));
2357 84 : poRelLayer->ResetReading();
2358 :
2359 : auto poChildFeature =
2360 168 : std::unique_ptr<OGRFeature>(poRelLayer->GetNextFeature());
2361 168 : XPathComponents aoNewInitialContext;
2362 84 : if (poChildFeature != nullptr)
2363 : {
2364 120 : if (aoFieldComponents.size() == aoLayerComponents.size() + 1 &&
2365 44 : oField.GetRepetitionOnSequence())
2366 : {
2367 : /* Case of
2368 : <xs:element name="sequence_unbounded_dt_1">
2369 : <xs:complexType>
2370 : <xs:sequence maxOccurs="unbounded">
2371 : <xs:element name="subelement"
2372 : type="xs:dateTime"/>
2373 : </xs:sequence>
2374 : </xs:complexType>
2375 : </xs:element>
2376 : */
2377 8 : aoNewInitialContext = std::move(aoFieldComponents);
2378 : }
2379 68 : else if (aoFieldComponents.size() == aoLayerComponents.size() + 2)
2380 : {
2381 : /* Case of
2382 : <xs:element name="sequence_1_dt_unbounded">
2383 : <xs:complexType>
2384 : <xs:sequence>
2385 : <xs:element name="subelement"
2386 : type="xs:dateTime"
2387 : maxOccurs="unbounded"/>
2388 : </xs:sequence>
2389 : </xs:complexType>
2390 : </xs:element>
2391 : */
2392 16 : aoNewInitialContext = std::move(aoFieldComponents);
2393 16 : aoNewInitialContext.pop_back();
2394 : }
2395 : else
2396 : {
2397 : /* Case of
2398 : <xs:element name="unbounded_sequence_1_dt"
2399 : maxOccurs="unbounded">
2400 : <xs:complexType>
2401 : <xs:sequence>
2402 : <xs:element name="subelement"
2403 : type="xs:dateTime"/>
2404 : </xs:sequence>
2405 : </xs:complexType>
2406 : </xs:element>
2407 : */
2408 52 : aoNewInitialContext = std::move(aoLayerComponents);
2409 : }
2410 :
2411 76 : WriteClosingAndStartingTags(aoCurComponents, aoNewInitialContext,
2412 76 : bCurIsRegularField);
2413 :
2414 76 : bAtLeastOneFieldWritten = true;
2415 76 : aoCurComponents = aoNewInitialContext;
2416 76 : bCurIsRegularField = false;
2417 : }
2418 :
2419 244 : while (poChildFeature)
2420 : {
2421 160 : bool bRet = WriteFeature(poChildFeature.get(), oChildLayerDesc,
2422 : oSetLayersInIterationSub, aoNewInitialContext,
2423 : aoPrefixComponents, nRecLevel + 1);
2424 :
2425 160 : if (!bRet)
2426 0 : return false;
2427 :
2428 160 : poChildFeature.reset(poRelLayer->GetNextFeature());
2429 : }
2430 84 : poRelLayer->ResetReading();
2431 :
2432 84 : return true;
2433 : }
2434 :
2435 : /************************************************************************/
2436 : /* GetFilteredLayer() */
2437 : /************************************************************************/
2438 :
2439 : OGRLayer *
2440 52 : GMLASWriter::GetFilteredLayer(OGRLayer *poSrcLayer, const CPLString &osFilter,
2441 : const std::set<CPLString> &oSetLayersInIteration)
2442 : {
2443 52 : if (oSetLayersInIteration.find(poSrcLayer->GetName()) ==
2444 104 : oSetLayersInIteration.end())
2445 : {
2446 32 : poSrcLayer->SetAttributeFilter(osFilter);
2447 32 : poSrcLayer->ResetReading();
2448 32 : return poSrcLayer;
2449 : }
2450 :
2451 : // RDBMS drivers will really create a new iterator independent of the
2452 : // underlying layer when using a SELECT statement
2453 20 : GDALDriver *poDriver = m_poSrcDS->GetDriver();
2454 40 : if (poDriver != nullptr &&
2455 20 : (EQUAL(poDriver->GetDescription(), "SQLite") ||
2456 0 : EQUAL(poDriver->GetDescription(), "PostgreSQL")))
2457 : {
2458 40 : CPLString osSQL;
2459 20 : osSQL.Printf("SELECT * FROM \"%s\" WHERE %s", poSrcLayer->GetName(),
2460 20 : osFilter.c_str());
2461 20 : return m_poSrcDS->ExecuteSQL(osSQL, nullptr, nullptr);
2462 : }
2463 :
2464 : // TODO ?
2465 0 : CPLDebug("GMLAS", "Cannot recursively iterate on %s on this driver",
2466 0 : poSrcLayer->GetName());
2467 0 : return nullptr;
2468 : }
2469 :
2470 : /************************************************************************/
2471 : /* ReleaseFilteredLayer() */
2472 : /************************************************************************/
2473 :
2474 52 : void GMLASWriter::ReleaseFilteredLayer(OGRLayer *poSrcLayer,
2475 : OGRLayer *poIterLayer)
2476 : {
2477 52 : if (poIterLayer != poSrcLayer)
2478 20 : m_poSrcDS->ReleaseResultSet(poIterLayer);
2479 : else
2480 32 : poSrcLayer->ResetReading();
2481 52 : }
2482 :
2483 : /************************************************************************/
2484 : /* WriteFieldWithLink() */
2485 : /************************************************************************/
2486 :
2487 36 : bool GMLASWriter::WriteFieldWithLink(
2488 : OGRFeature *poFeature, const GMLASField &oField,
2489 : const LayerDescription &oLayerDesc, XPathComponents &aoLayerComponents,
2490 : XPathComponents &aoCurComponents, const XPathComponents &aoPrefixComponents,
2491 : const std::set<CPLString> &oSetLayersInIteration, int nRecLevel,
2492 : bool &bAtLeastOneFieldWritten, bool &bCurIsRegularField)
2493 : {
2494 36 : const auto oIter = m_oMapXPathToIdx.find(oField.GetRelatedClassXPath());
2495 36 : if (oIter == m_oMapXPathToIdx.end())
2496 : {
2497 : // Not necessary to be more verbose in case of truncated
2498 : // source dataset
2499 0 : CPLDebug("GMLAS", "No child layer of %s matching xpath = %s",
2500 : oLayerDesc.osName.c_str(),
2501 0 : oField.GetRelatedClassXPath().c_str());
2502 0 : return true;
2503 : }
2504 :
2505 36 : const LayerDescription &oChildLayerDesc = m_aoLayerDesc[oIter->second];
2506 36 : OGRLayer *poRelLayer = GetLayerByName(oChildLayerDesc.osName);
2507 36 : if (poRelLayer == nullptr)
2508 : {
2509 : // Not necessary to be more verbose in case of truncated
2510 : // source dataset
2511 0 : CPLDebug("GMLAS", "Referenced layer %s of %s not found",
2512 : oChildLayerDesc.osName.c_str(), oLayerDesc.osName.c_str());
2513 0 : return true;
2514 : }
2515 :
2516 36 : const int nFieldIdx = oLayerDesc.GetOGRIdxFromFieldName(oField.GetName());
2517 72 : XPathComponents aoFieldComponents = SplitXPath(oField.GetXPath());
2518 36 : aoFieldComponents.insert(aoFieldComponents.begin(),
2519 : aoPrefixComponents.begin(),
2520 72 : aoPrefixComponents.end());
2521 :
2522 36 : if (nFieldIdx < 0)
2523 : {
2524 0 : CPLError(CE_Failure, CPLE_AppDefined, "Missing field %s for layer %s",
2525 0 : oField.GetName().c_str(), oLayerDesc.osName.c_str());
2526 0 : return true;
2527 : }
2528 36 : if (!poFeature->IsFieldSetAndNotNull(nFieldIdx))
2529 : {
2530 : // Not an error (unless the field is required)
2531 4 : return true;
2532 : }
2533 32 : if (oLayerDesc.osPKIDName.empty())
2534 : {
2535 0 : CPLError(CE_Failure, CPLE_AppDefined, "Missing %s for layer %s",
2536 : szLAYER_PKID_NAME, oLayerDesc.osName.c_str());
2537 0 : return true;
2538 : }
2539 32 : if (oChildLayerDesc.osPKIDName.empty())
2540 : {
2541 0 : CPLError(CE_Failure, CPLE_AppDefined, "Missing %s for layer %s",
2542 : szLAYER_PKID_NAME, oChildLayerDesc.osName.c_str());
2543 0 : return true;
2544 : }
2545 32 : if (aoFieldComponents.size() < 2)
2546 : {
2547 : // Shouldn't happen for well behaved metadata
2548 0 : CPLDebug("GMLAS", "Unexpected at line %d", __LINE__);
2549 0 : return true;
2550 : }
2551 64 : if (oChildLayerDesc.osXPath.empty() ||
2552 32 : aoFieldComponents.back() != SplitXPath(oChildLayerDesc.osXPath).front())
2553 : {
2554 : // Shouldn't happen for well behaved metadata
2555 0 : CPLDebug("GMLAS", "Unexpected at line %d", __LINE__);
2556 0 : return true;
2557 : }
2558 :
2559 64 : CPLString osChildPKID(poFeature->GetFieldAsString(nFieldIdx));
2560 : const CPLString osFilter(CPLSPrintf(
2561 64 : "%s = '%s'", oChildLayerDesc.osPKIDName.c_str(), osChildPKID.c_str()));
2562 : OGRLayer *poIterLayer =
2563 32 : GetFilteredLayer(poRelLayer, osFilter, oSetLayersInIteration);
2564 32 : if (poIterLayer == nullptr)
2565 : {
2566 0 : return true;
2567 : }
2568 :
2569 64 : std::set<CPLString> oSetLayersInIterationSub(oSetLayersInIteration);
2570 32 : oSetLayersInIterationSub.insert(oChildLayerDesc.osName);
2571 :
2572 64 : XPathComponents aoPrefixComponentsNew(aoFieldComponents);
2573 32 : aoPrefixComponentsNew.pop_back();
2574 :
2575 32 : if (aoLayerComponents.empty())
2576 : {
2577 32 : aoLayerComponents = SplitXPath(oLayerDesc.osXPath);
2578 32 : aoLayerComponents.insert(aoLayerComponents.begin(),
2579 : aoPrefixComponents.begin(),
2580 64 : aoPrefixComponents.end());
2581 : }
2582 :
2583 : auto poChildFeature =
2584 64 : std::unique_ptr<OGRFeature>(poIterLayer->GetNextFeature());
2585 32 : XPathComponents aoInitialComponents;
2586 32 : const bool bHasChild = poChildFeature != nullptr;
2587 32 : if (bHasChild)
2588 : {
2589 32 : aoInitialComponents = std::move(aoFieldComponents);
2590 32 : if (!aoInitialComponents.empty())
2591 32 : aoInitialComponents.pop_back();
2592 32 : WriteClosingAndStartingTags(aoCurComponents, aoInitialComponents,
2593 32 : bCurIsRegularField);
2594 : }
2595 :
2596 32 : bool bRet = true;
2597 64 : while (poChildFeature)
2598 : {
2599 32 : bRet = WriteFeature(poChildFeature.get(), oChildLayerDesc,
2600 : oSetLayersInIterationSub, aoInitialComponents,
2601 : aoPrefixComponentsNew, nRecLevel + 1);
2602 32 : if (!bRet)
2603 0 : break;
2604 32 : poChildFeature.reset(poIterLayer->GetNextFeature());
2605 : }
2606 32 : ReleaseFilteredLayer(poRelLayer, poIterLayer);
2607 :
2608 32 : if (bHasChild)
2609 : {
2610 32 : bAtLeastOneFieldWritten = true;
2611 32 : aoCurComponents = std::move(aoInitialComponents);
2612 32 : bCurIsRegularField = false;
2613 : }
2614 :
2615 32 : return bRet;
2616 : }
2617 :
2618 : /************************************************************************/
2619 : /* WriteFieldJunctionTable() */
2620 : /************************************************************************/
2621 :
2622 16 : bool GMLASWriter::WriteFieldJunctionTable(
2623 : OGRFeature *poFeature, const GMLASField &oField,
2624 : const LayerDescription &oLayerDesc,
2625 : XPathComponents & /*aoLayerComponents */, XPathComponents &aoCurComponents,
2626 : const XPathComponents &aoPrefixComponents,
2627 : const std::set<CPLString> &oSetLayersInIteration, int nRecLevel,
2628 : bool &bAtLeastOneFieldWritten, bool &bCurIsRegularField)
2629 : {
2630 16 : const auto oIter = m_oMapXPathToIdx.find(oField.GetRelatedClassXPath());
2631 16 : if (oIter == m_oMapXPathToIdx.end())
2632 : {
2633 : // Not necessary to be more verbose in case of truncated
2634 : // source dataset
2635 0 : CPLDebug("GMLAS", "No related layer of %s matching xpath = %s",
2636 : oLayerDesc.osName.c_str(),
2637 0 : oField.GetRelatedClassXPath().c_str());
2638 0 : return true;
2639 : }
2640 :
2641 16 : const LayerDescription &oRelLayerDesc = m_aoLayerDesc[oIter->second];
2642 16 : OGRLayer *poRelLayer = GetLayerByName(oRelLayerDesc.osName);
2643 16 : OGRLayer *poJunctionLayer = GetLayerByName(oField.GetJunctionLayer());
2644 16 : if (poRelLayer == nullptr)
2645 : {
2646 : // Not necessary to be more verbose in case of truncated
2647 : // source dataset
2648 0 : CPLDebug("GMLAS", "Referenced layer %s of %s not found",
2649 : oRelLayerDesc.osName.c_str(), oLayerDesc.osName.c_str());
2650 0 : return true;
2651 : }
2652 16 : if (poJunctionLayer == nullptr)
2653 : {
2654 : // Not necessary to be more verbose in case of truncated
2655 : // source dataset
2656 0 : CPLDebug("GMLAS", "Junction layer %s not found",
2657 0 : oField.GetJunctionLayer().c_str());
2658 0 : return true;
2659 : }
2660 :
2661 16 : int nIndexPKID = -1;
2662 16 : if (oLayerDesc.osPKIDName.empty())
2663 : {
2664 0 : CPLError(CE_Failure, CPLE_AppDefined, "Missing %s for layer %s",
2665 : szLAYER_PKID_NAME, oLayerDesc.osName.c_str());
2666 0 : return true;
2667 : }
2668 16 : if ((nIndexPKID =
2669 16 : oLayerDesc.GetOGRIdxFromFieldName(oLayerDesc.osPKIDName)) < 0)
2670 : {
2671 0 : CPLError(CE_Failure, CPLE_AppDefined, "Cannot find %s='%s' in layer %s",
2672 : szLAYER_PKID_NAME, oLayerDesc.osPKIDName.c_str(),
2673 : oLayerDesc.osName.c_str());
2674 0 : return true;
2675 : }
2676 16 : if (!poFeature->IsFieldSetAndNotNull(nIndexPKID))
2677 : {
2678 0 : CPLError(CE_Failure, CPLE_AppDefined,
2679 : "Field '%s' in layer %s is not set for "
2680 : "feature " CPL_FRMT_GIB,
2681 : oLayerDesc.osPKIDName.c_str(), oLayerDesc.osName.c_str(),
2682 : poFeature->GetFID());
2683 0 : return true;
2684 : }
2685 16 : if (oRelLayerDesc.osPKIDName.empty())
2686 : {
2687 0 : CPLError(CE_Failure, CPLE_AppDefined, "Missing %s for layer %s",
2688 : szLAYER_PKID_NAME, oRelLayerDesc.osName.c_str());
2689 0 : return true;
2690 : }
2691 16 : if (oSetLayersInIteration.find(oRelLayerDesc.osName) !=
2692 32 : oSetLayersInIteration.end())
2693 : {
2694 : // TODO... cycle situation. We will need to open a new
2695 : // source dataset or something
2696 0 : return true;
2697 : }
2698 :
2699 32 : std::set<CPLString> oSetLayersInIterationSub(oSetLayersInIteration);
2700 16 : oSetLayersInIterationSub.insert(oRelLayerDesc.osName);
2701 :
2702 16 : poJunctionLayer->SetAttributeFilter(CPLSPrintf(
2703 16 : "%s = '%s'", szPARENT_PKID, poFeature->GetFieldAsString(nIndexPKID)));
2704 16 : poJunctionLayer->ResetReading();
2705 32 : std::vector<CPLString> aoChildPKIDs;
2706 36 : for (auto &&poJunctionFeature : *poJunctionLayer)
2707 : {
2708 20 : aoChildPKIDs.push_back(
2709 : poJunctionFeature->GetFieldAsString(szCHILD_PKID));
2710 : }
2711 16 : poJunctionLayer->ResetReading();
2712 :
2713 16 : bool bRet = true;
2714 16 : bool bHasChild = false;
2715 32 : XPathComponents aoInitialComponents;
2716 36 : for (size_t j = 0; bRet && j < aoChildPKIDs.size(); j++)
2717 : {
2718 20 : CPLString osFilter;
2719 : osFilter.Printf("%s = '%s'", oRelLayerDesc.osPKIDName.c_str(),
2720 20 : aoChildPKIDs[j].c_str());
2721 : OGRLayer *poIterLayer =
2722 20 : GetFilteredLayer(poRelLayer, osFilter, oSetLayersInIteration);
2723 20 : if (poIterLayer == nullptr)
2724 : {
2725 0 : return true;
2726 : }
2727 :
2728 : auto poChildFeature =
2729 40 : std::unique_ptr<OGRFeature>(poIterLayer->GetNextFeature());
2730 20 : if (poChildFeature != nullptr)
2731 : {
2732 20 : if (!bHasChild)
2733 : {
2734 12 : bHasChild = true;
2735 :
2736 12 : aoInitialComponents = SplitXPath(oField.GetXPath());
2737 12 : aoInitialComponents.insert(aoInitialComponents.begin(),
2738 : aoPrefixComponents.begin(),
2739 24 : aoPrefixComponents.end());
2740 :
2741 12 : if (!aoInitialComponents.empty())
2742 12 : aoInitialComponents.pop_back();
2743 12 : WriteClosingAndStartingTags(
2744 12 : aoCurComponents, aoInitialComponents, bCurIsRegularField);
2745 : }
2746 :
2747 20 : bRet = WriteFeature(poChildFeature.get(), oRelLayerDesc,
2748 40 : oSetLayersInIterationSub, XPathComponents(),
2749 40 : XPathComponents(), nRecLevel + 1);
2750 :
2751 20 : ReleaseFilteredLayer(poRelLayer, poIterLayer);
2752 : }
2753 : else
2754 : {
2755 0 : ReleaseFilteredLayer(poRelLayer, poIterLayer);
2756 : }
2757 : }
2758 :
2759 16 : if (bHasChild)
2760 : {
2761 12 : bAtLeastOneFieldWritten = true;
2762 12 : aoCurComponents = std::move(aoInitialComponents);
2763 12 : bCurIsRegularField = false;
2764 : }
2765 :
2766 16 : return bRet;
2767 : }
2768 :
2769 : /************************************************************************/
2770 : /* PrintIndent() */
2771 : /************************************************************************/
2772 :
2773 870 : void GMLASWriter::PrintIndent(VSILFILE *fp)
2774 : {
2775 3888 : for (int i = 0; i < m_nIndentLevel; i++)
2776 : {
2777 3018 : VSIFWriteL(m_osIndentation.c_str(), 1, m_osIndentation.size(), fp);
2778 : }
2779 870 : }
2780 :
2781 : /************************************************************************/
2782 : /* PrintLine() */
2783 : /************************************************************************/
2784 :
2785 1100 : void GMLASWriter::PrintLine(VSILFILE *fp, const char *fmt, ...)
2786 : {
2787 2200 : CPLString osWork;
2788 : va_list args;
2789 :
2790 1100 : va_start(args, fmt);
2791 1100 : osWork.vPrintf(fmt, args);
2792 1100 : va_end(args);
2793 :
2794 1100 : VSIFWriteL(osWork.c_str(), 1, osWork.size(), fp);
2795 1100 : VSIFWriteL(m_osEOL.c_str(), 1, m_osEOL.size(), fp);
2796 1100 : }
2797 :
2798 : } /* namespace GMLAS */
2799 :
2800 : /************************************************************************/
2801 : /* GMLASFakeDataset */
2802 : /************************************************************************/
2803 :
2804 : class GMLASFakeDataset final : public GDALDataset
2805 : {
2806 : public:
2807 0 : GMLASFakeDataset()
2808 0 : {
2809 0 : }
2810 : };
2811 :
2812 : /************************************************************************/
2813 : /* OGRGMLASDriverCreateCopy() */
2814 : /************************************************************************/
2815 :
2816 22 : GDALDataset *OGRGMLASDriverCreateCopy(const char *pszFilename,
2817 : GDALDataset *poSrcDS, int /*bStrict*/,
2818 : char **papszOptions,
2819 : GDALProgressFunc pfnProgress,
2820 : void *pProgressData)
2821 : {
2822 22 : if (strcmp(CPLGetExtensionSafe(pszFilename).c_str(), "xsd") == 0)
2823 : {
2824 1 : CPLError(CE_Failure, CPLE_AppDefined, ".xsd extension is not valid");
2825 1 : return nullptr;
2826 : }
2827 :
2828 : // Strip GMLAS: prefix if specified
2829 21 : if (STARTS_WITH_CI(pszFilename, szGMLAS_PREFIX))
2830 1 : pszFilename += strlen(szGMLAS_PREFIX);
2831 :
2832 42 : GMLAS::GMLASWriter oWriter(pszFilename, poSrcDS, papszOptions);
2833 21 : if (!oWriter.Write(pfnProgress, pProgressData))
2834 11 : return nullptr;
2835 :
2836 20 : if (CPLString(pszFilename) == "/vsistdout/" ||
2837 : // This option is mostly useful for tests where we don't want
2838 : // WFS 2.0 schemas to be pulled from the network
2839 10 : !CPLFetchBool(papszOptions, "REOPEN_DATASET_WITH_GMLAS", true))
2840 : {
2841 0 : return new GMLASFakeDataset();
2842 : }
2843 : else
2844 : {
2845 : GDALOpenInfo oOpenInfo(
2846 30 : (CPLString(szGMLAS_PREFIX) + pszFilename).c_str(), GA_ReadOnly);
2847 20 : auto poOutDS = std::make_unique<OGRGMLASDataSource>();
2848 10 : if (!poOutDS->Open(&oOpenInfo))
2849 : {
2850 0 : poOutDS.reset();
2851 : }
2852 10 : return poOutDS.release();
2853 : }
2854 : }
|