Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: OGR
4 : * Purpose: Implements OGRGMLLayer class.
5 : * Author: Frank Warmerdam, warmerdam@pobox.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2002, Frank Warmerdam <warmerdam@pobox.com>
9 : * Copyright (c) 2009-2013, Even Rouault <even dot rouault at spatialys.com>
10 : *
11 : * SPDX-License-Identifier: MIT
12 : ****************************************************************************/
13 :
14 : #include "ogr_gml.h"
15 : #include "gmlutils.h"
16 : #include "cpl_conv.h"
17 : #include "cpl_port.h"
18 : #include "cpl_string.h"
19 : #include "ogr_p.h"
20 : #include "ogr_api.h"
21 :
22 : /************************************************************************/
23 : /* OGRGMLLayer() */
24 : /************************************************************************/
25 :
26 682 : OGRGMLLayer::OGRGMLLayer(const char *pszName, bool bWriterIn,
27 682 : OGRGMLDataSource *poDSIn)
28 : : poFeatureDefn(new OGRFeatureDefn(
29 682 : pszName + (STARTS_WITH_CI(pszName, "ogr:") ? 4 : 0))),
30 : iNextGMLId(0), bInvalidFIDFound(false), pszFIDPrefix(nullptr),
31 : bWriter(bWriterIn), poDS(poDSIn),
32 682 : poFClass(!bWriter ? poDS->GetReader()->GetClass(pszName) : nullptr),
33 : // Reader's should get the corresponding GMLFeatureClass and cache it.
34 1364 : hCacheSRS(GML_BuildOGRGeometryFromList_CreateCache()),
35 : // Compatibility option. Not advertized, because hopefully won't be
36 : // needed. Just put here in case.
37 : bUseOldFIDFormat(
38 682 : CPLTestBool(CPLGetConfigOption("GML_USE_OLD_FID_FORMAT", "FALSE"))),
39 : // Must be in synced in OGR_G_CreateFromGML(), OGRGMLLayer::OGRGMLLayer()
40 : // and GMLReader::GMLReader().
41 : bFaceHoleNegative(
42 2728 : CPLTestBool(CPLGetConfigOption("GML_FACE_HOLE_NEGATIVE", "NO")))
43 : {
44 682 : SetDescription(poFeatureDefn->GetName());
45 682 : poFeatureDefn->Reference();
46 682 : poFeatureDefn->SetGeomType(wkbNone);
47 682 : }
48 :
49 : /************************************************************************/
50 : /* ~OGRGMLLayer() */
51 : /************************************************************************/
52 :
53 1364 : OGRGMLLayer::~OGRGMLLayer()
54 :
55 : {
56 682 : CPLFree(pszFIDPrefix);
57 :
58 682 : if (poFeatureDefn)
59 682 : poFeatureDefn->Release();
60 :
61 682 : GML_BuildOGRGeometryFromList_DestroyCache(hCacheSRS);
62 1364 : }
63 :
64 : /************************************************************************/
65 : /* ResetReading() */
66 : /************************************************************************/
67 :
68 750 : void OGRGMLLayer::ResetReading()
69 :
70 : {
71 750 : if (bWriter)
72 16 : return;
73 :
74 1468 : if (poDS->GetReadMode() == INTERLEAVED_LAYERS ||
75 734 : poDS->GetReadMode() == SEQUENTIAL_LAYERS)
76 : {
77 : // Does the last stored feature belong to our layer ? If so, no
78 : // need to reset the reader.
79 94 : if (iNextGMLId == 0 && poDS->PeekStoredGMLFeature() != nullptr &&
80 0 : poDS->PeekStoredGMLFeature()->GetClass() == poFClass)
81 0 : return;
82 :
83 94 : delete poDS->PeekStoredGMLFeature();
84 94 : poDS->SetStoredGMLFeature(nullptr);
85 : }
86 :
87 734 : iNextGMLId = 0;
88 734 : poDS->GetReader()->ResetReading();
89 734 : CPLDebug("GML", "ResetReading()");
90 734 : if (poDS->GetLayerCount() > 1 && poDS->GetReadMode() == STANDARD)
91 : {
92 78 : const char *pszElementName = poFClass->GetElementName();
93 78 : const char *pszLastPipe = strrchr(pszElementName, '|');
94 78 : if (pszLastPipe != nullptr)
95 33 : pszElementName = pszLastPipe + 1;
96 78 : poDS->GetReader()->SetFilteredClassName(pszElementName);
97 : }
98 : }
99 :
100 : /************************************************************************/
101 : /* Increment() */
102 : /************************************************************************/
103 :
104 563 : static GIntBig Increment(GIntBig nVal)
105 : {
106 563 : if (nVal <= GINTBIG_MAX - 1)
107 563 : return nVal + 1;
108 0 : return nVal;
109 : }
110 :
111 : /************************************************************************/
112 : /* GetNextFeature() */
113 : /************************************************************************/
114 :
115 824 : OGRFeature *OGRGMLLayer::GetNextFeature()
116 :
117 : {
118 824 : if (bWriter)
119 : {
120 16 : CPLError(CE_Failure, CPLE_NotSupported,
121 : "Cannot read features when writing a GML file");
122 16 : return nullptr;
123 : }
124 :
125 808 : if (poDS->GetLastReadLayer() != this)
126 : {
127 380 : if (poDS->GetReadMode() != INTERLEAVED_LAYERS)
128 371 : ResetReading();
129 380 : poDS->SetLastReadLayer(this);
130 : }
131 :
132 : /* ==================================================================== */
133 : /* Loop till we find and translate a feature meeting all our */
134 : /* requirements. */
135 : /* ==================================================================== */
136 : while (true)
137 : {
138 1466 : GMLFeature *poGMLFeature = poDS->PeekStoredGMLFeature();
139 1466 : if (poGMLFeature != nullptr)
140 : {
141 3 : poDS->SetStoredGMLFeature(nullptr);
142 : }
143 : else
144 : {
145 1463 : poGMLFeature = poDS->GetReader()->NextFeature();
146 1463 : if (poGMLFeature == nullptr)
147 808 : return nullptr;
148 :
149 : // We count reading low level GML features as a feature read for
150 : // work checking purposes, though at least we didn't necessary
151 : // have to turn it into an OGRFeature.
152 1334 : m_nFeaturesRead++;
153 : }
154 :
155 : /* --------------------------------------------------------------------
156 : */
157 : /* Is it of the proper feature class? */
158 : /* --------------------------------------------------------------------
159 : */
160 :
161 1337 : if (poGMLFeature->GetClass() != poFClass)
162 : {
163 1836 : if (poDS->GetReadMode() == INTERLEAVED_LAYERS ||
164 1222 : (poDS->GetReadMode() == SEQUENTIAL_LAYERS && iNextGMLId != 0))
165 : {
166 4 : CPLAssert(poDS->PeekStoredGMLFeature() == nullptr);
167 4 : poDS->SetStoredGMLFeature(poGMLFeature);
168 4 : return nullptr;
169 : }
170 : else
171 : {
172 610 : delete poGMLFeature;
173 610 : continue;
174 : }
175 : }
176 :
177 : /* --------------------------------------------------------------------
178 : */
179 : /* Extract the fid: */
180 : /* -Assumes the fids are non-negative integers with an optional */
181 : /* prefix */
182 : /* -If a prefix differs from the prefix of the first feature from
183 : */
184 : /* the poDS then the fids from the poDS are ignored and are */
185 : /* assigned serially thereafter */
186 : /* --------------------------------------------------------------------
187 : */
188 723 : GIntBig nFID = -1;
189 723 : const char *pszGML_FID = poGMLFeature->GetFID();
190 723 : if (bInvalidFIDFound)
191 : {
192 44 : nFID = iNextGMLId;
193 44 : iNextGMLId = Increment(iNextGMLId);
194 : }
195 679 : else if (pszGML_FID == nullptr)
196 : {
197 109 : bInvalidFIDFound = true;
198 109 : nFID = iNextGMLId;
199 109 : iNextGMLId = Increment(iNextGMLId);
200 : }
201 570 : else if (iNextGMLId == 0)
202 : {
203 293 : int j = 0;
204 293 : int i = static_cast<int>(strlen(pszGML_FID)) - 1;
205 710 : while (i >= 0 && pszGML_FID[i] >= '0' && pszGML_FID[i] <= '9' &&
206 : j < 20)
207 : {
208 417 : i--;
209 417 : j++;
210 : }
211 : // i points the last character of the fid.
212 293 : if (i >= 0 && j < 20 && pszFIDPrefix == nullptr)
213 : {
214 230 : pszFIDPrefix = static_cast<char *>(CPLMalloc(i + 2));
215 230 : pszFIDPrefix[i + 1] = '\0';
216 230 : strncpy(pszFIDPrefix, pszGML_FID, i + 1);
217 : }
218 : // pszFIDPrefix now contains the prefix or NULL if no prefix is
219 : // found.
220 293 : if (j < 20 && sscanf(pszGML_FID + i + 1, CPL_FRMT_GIB, &nFID) == 1)
221 : {
222 284 : if (iNextGMLId <= nFID)
223 284 : iNextGMLId = Increment(nFID);
224 : }
225 : else
226 : {
227 9 : bInvalidFIDFound = true;
228 9 : nFID = iNextGMLId;
229 9 : iNextGMLId = Increment(iNextGMLId);
230 : }
231 : }
232 : else // if( iNextGMLId != 0 ).
233 : {
234 277 : const char *pszFIDPrefix_notnull = pszFIDPrefix;
235 277 : if (pszFIDPrefix_notnull == nullptr)
236 6 : pszFIDPrefix_notnull = "";
237 277 : int nLenPrefix = static_cast<int>(strlen(pszFIDPrefix_notnull));
238 :
239 827 : if (strncmp(pszGML_FID, pszFIDPrefix_notnull, nLenPrefix) == 0 &&
240 550 : strlen(pszGML_FID + nLenPrefix) < 20 &&
241 273 : sscanf(pszGML_FID + nLenPrefix, CPL_FRMT_GIB, &nFID) == 1)
242 : {
243 : // fid with the prefix. Using its numerical part.
244 272 : if (iNextGMLId < nFID)
245 112 : iNextGMLId = Increment(nFID);
246 : }
247 : else
248 : {
249 : // fid without the aforementioned prefix or a valid numerical
250 : // part.
251 5 : bInvalidFIDFound = true;
252 5 : nFID = iNextGMLId;
253 5 : iNextGMLId = Increment(iNextGMLId);
254 : }
255 : }
256 :
257 : /* --------------------------------------------------------------------
258 : */
259 : /* Does it satisfy the spatial query, if there is one? */
260 : /* --------------------------------------------------------------------
261 : */
262 :
263 723 : OGRGeometry **papoGeometries = nullptr;
264 723 : const CPLXMLNode *const *papsGeometry = poGMLFeature->GetGeometryList();
265 :
266 723 : const CPLXMLNode *apsGeometries[2] = {nullptr, nullptr};
267 : const CPLXMLNode *psBoundedByGeometry =
268 723 : poGMLFeature->GetBoundedByGeometry();
269 723 : if (psBoundedByGeometry && !(papsGeometry && papsGeometry[0]))
270 : {
271 6 : apsGeometries[0] = psBoundedByGeometry;
272 6 : papsGeometry = apsGeometries;
273 : }
274 :
275 723 : OGRGeometry *poGeom = nullptr;
276 :
277 723 : if (poFeatureDefn->GetGeomFieldCount() > 1)
278 : {
279 188 : papoGeometries = static_cast<OGRGeometry **>(CPLCalloc(
280 94 : poFeatureDefn->GetGeomFieldCount(), sizeof(OGRGeometry *)));
281 94 : const char *pszSRSName = poDS->GetGlobalSRSName();
282 301 : for (int i = 0; i < poFeatureDefn->GetGeomFieldCount(); i++)
283 : {
284 207 : const CPLXMLNode *psGeom = poGMLFeature->GetGeometryRef(i);
285 207 : if (psGeom != nullptr)
286 : {
287 163 : const CPLXMLNode *myGeometryList[2] = {psGeom, nullptr};
288 163 : poGeom = GML_BuildOGRGeometryFromList(
289 : myGeometryList, true,
290 163 : poDS->GetInvertAxisOrderIfLatLong(), pszSRSName,
291 163 : poDS->GetConsiderEPSGAsURN(),
292 163 : poDS->GetSwapCoordinates(),
293 163 : poDS->GetSecondaryGeometryOption(), hCacheSRS,
294 163 : bFaceHoleNegative);
295 :
296 : // Do geometry type changes if needed to match layer
297 : // geometry type.
298 163 : if (poGeom != nullptr)
299 : {
300 163 : papoGeometries[i] = OGRGeometryFactory::forceTo(
301 : poGeom,
302 163 : poFeatureDefn->GetGeomFieldDefn(i)->GetType());
303 163 : poGeom = nullptr;
304 : }
305 : else
306 : {
307 : // We assume the createFromGML() function would have
308 : // already reported the error.
309 0 : for (int j = 0; j < poFeatureDefn->GetGeomFieldCount();
310 : j++)
311 : {
312 0 : delete papoGeometries[j];
313 : }
314 0 : CPLFree(papoGeometries);
315 0 : delete poGMLFeature;
316 0 : return nullptr;
317 : }
318 : }
319 : }
320 :
321 0 : if (m_poFilterGeom != nullptr && m_iGeomFieldFilter >= 0 &&
322 0 : m_iGeomFieldFilter < poFeatureDefn->GetGeomFieldCount() &&
323 94 : papoGeometries[m_iGeomFieldFilter] &&
324 0 : !FilterGeometry(papoGeometries[m_iGeomFieldFilter]))
325 : {
326 0 : for (int j = 0; j < poFeatureDefn->GetGeomFieldCount(); j++)
327 : {
328 0 : delete papoGeometries[j];
329 : }
330 0 : CPLFree(papoGeometries);
331 0 : delete poGMLFeature;
332 0 : continue;
333 : }
334 : }
335 629 : else if (papsGeometry[0] &&
336 531 : strcmp(papsGeometry[0]->pszValue, "null") == 0)
337 : {
338 : // do nothing
339 : }
340 625 : else if (papsGeometry[0] != nullptr)
341 : {
342 527 : const char *pszSRSName = poDS->GetGlobalSRSName();
343 527 : CPLPushErrorHandler(CPLQuietErrorHandler);
344 527 : poGeom = GML_BuildOGRGeometryFromList(
345 527 : papsGeometry, true, poDS->GetInvertAxisOrderIfLatLong(),
346 527 : pszSRSName, poDS->GetConsiderEPSGAsURN(),
347 527 : poDS->GetSwapCoordinates(), poDS->GetSecondaryGeometryOption(),
348 527 : hCacheSRS, bFaceHoleNegative);
349 527 : CPLPopErrorHandler();
350 :
351 : // Do geometry type changes if needed to match layer geometry type.
352 527 : if (poGeom != nullptr)
353 : {
354 527 : poGeom = OGRGeometryFactory::forceTo(poGeom, GetGeomType());
355 : }
356 : else
357 : {
358 0 : const CPLString osLastErrorMsg(CPLGetLastErrorMsg());
359 :
360 0 : const bool bGoOn = CPLTestBool(
361 : CPLGetConfigOption("GML_SKIP_CORRUPTED_FEATURES", "NO"));
362 :
363 0 : CPLError(
364 : bGoOn ? CE_Warning : CE_Failure, CPLE_AppDefined,
365 : "Geometry of feature " CPL_FRMT_GIB
366 : " %scannot be parsed: %s%s",
367 0 : nFID, pszGML_FID ? CPLSPrintf("%s ", pszGML_FID) : "",
368 : osLastErrorMsg.c_str(),
369 : bGoOn ? ". Skipping to next feature."
370 : : ". You may set the GML_SKIP_CORRUPTED_FEATURES "
371 : "configuration option to YES to skip to the next "
372 : "feature");
373 0 : delete poGMLFeature;
374 0 : if (bGoOn)
375 0 : continue;
376 0 : return nullptr;
377 : }
378 :
379 527 : if (m_poFilterGeom != nullptr && !FilterGeometry(poGeom))
380 : {
381 20 : delete poGMLFeature;
382 20 : delete poGeom;
383 20 : continue;
384 : }
385 : }
386 :
387 : /* --------------------------------------------------------------------
388 : */
389 : /* Convert the whole feature into an OGRFeature. */
390 : /* --------------------------------------------------------------------
391 : */
392 703 : int iDstField = 0;
393 703 : OGRFeature *poOGRFeature = new OGRFeature(poFeatureDefn);
394 :
395 703 : poOGRFeature->SetFID(nFID);
396 703 : if (poDS->ExposeId())
397 : {
398 595 : if (pszGML_FID)
399 558 : poOGRFeature->SetField(iDstField, pszGML_FID);
400 595 : iDstField++;
401 : }
402 :
403 703 : const int nPropertyCount = poFClass->GetPropertyCount();
404 3430 : for (int iField = 0; iField < nPropertyCount; iField++, iDstField++)
405 : {
406 : const GMLProperty *psGMLProperty =
407 2727 : poGMLFeature->GetProperty(iField);
408 2727 : if (psGMLProperty == nullptr || psGMLProperty->nSubProperties == 0)
409 614 : continue;
410 :
411 2113 : if (EQUAL(psGMLProperty->papszSubProperties[0], OGR_GML_NULL))
412 : {
413 8 : poOGRFeature->SetFieldNull(iDstField);
414 8 : continue;
415 : }
416 :
417 2105 : switch (poFClass->GetProperty(iField)->GetType())
418 : {
419 354 : case GMLPT_Real:
420 : {
421 354 : poOGRFeature->SetField(
422 : iDstField,
423 354 : CPLAtof(psGMLProperty->papszSubProperties[0]));
424 : }
425 354 : break;
426 :
427 13 : case GMLPT_IntegerList:
428 : {
429 13 : const int nCount = psGMLProperty->nSubProperties;
430 : int *panIntList =
431 13 : static_cast<int *>(CPLMalloc(sizeof(int) * nCount));
432 :
433 39 : for (int i = 0; i < nCount; i++)
434 26 : panIntList[i] =
435 26 : atoi(psGMLProperty->papszSubProperties[i]);
436 :
437 13 : poOGRFeature->SetField(iDstField, nCount, panIntList);
438 13 : CPLFree(panIntList);
439 : }
440 13 : break;
441 :
442 6 : case GMLPT_Integer64List:
443 : {
444 6 : const int nCount = psGMLProperty->nSubProperties;
445 : GIntBig *panIntList = static_cast<GIntBig *>(
446 6 : CPLMalloc(sizeof(GIntBig) * nCount));
447 :
448 15 : for (int i = 0; i < nCount; i++)
449 18 : panIntList[i] =
450 9 : CPLAtoGIntBig(psGMLProperty->papszSubProperties[i]);
451 :
452 6 : poOGRFeature->SetField(iDstField, nCount, panIntList);
453 6 : CPLFree(panIntList);
454 : }
455 6 : break;
456 :
457 25 : case GMLPT_RealList:
458 : {
459 25 : const int nCount = psGMLProperty->nSubProperties;
460 : double *padfList = static_cast<double *>(
461 25 : CPLMalloc(sizeof(double) * nCount));
462 :
463 59 : for (int i = 0; i < nCount; i++)
464 68 : padfList[i] =
465 34 : CPLAtof(psGMLProperty->papszSubProperties[i]);
466 :
467 25 : poOGRFeature->SetField(iDstField, nCount, padfList);
468 25 : CPLFree(padfList);
469 : }
470 25 : break;
471 :
472 106 : case GMLPT_StringList:
473 : case GMLPT_FeaturePropertyList:
474 : {
475 106 : poOGRFeature->SetField(iDstField,
476 106 : psGMLProperty->papszSubProperties);
477 : }
478 106 : break;
479 :
480 46 : case GMLPT_Boolean:
481 : {
482 46 : if (strcmp(psGMLProperty->papszSubProperties[0], "true") ==
483 7 : 0 ||
484 7 : strcmp(psGMLProperty->papszSubProperties[0], "1") == 0)
485 : {
486 39 : poOGRFeature->SetField(iDstField, 1);
487 : }
488 7 : else if (strcmp(psGMLProperty->papszSubProperties[0],
489 0 : "false") == 0 ||
490 0 : strcmp(psGMLProperty->papszSubProperties[0],
491 : "0") == 0)
492 : {
493 7 : poOGRFeature->SetField(iDstField, 0);
494 : }
495 : else
496 : {
497 0 : poOGRFeature->SetField(
498 0 : iDstField, psGMLProperty->papszSubProperties[0]);
499 : }
500 46 : break;
501 : }
502 :
503 3 : case GMLPT_BooleanList:
504 : {
505 3 : const int nCount = psGMLProperty->nSubProperties;
506 : int *panIntList =
507 3 : static_cast<int *>(CPLMalloc(sizeof(int) * nCount));
508 :
509 9 : for (int i = 0; i < nCount; i++)
510 : {
511 6 : panIntList[i] =
512 6 : (strcmp(psGMLProperty->papszSubProperties[i],
513 9 : "true") == 0 ||
514 3 : strcmp(psGMLProperty->papszSubProperties[i],
515 : "1") == 0);
516 : }
517 :
518 3 : poOGRFeature->SetField(iDstField, nCount, panIntList);
519 3 : CPLFree(panIntList);
520 3 : break;
521 : }
522 :
523 1552 : default:
524 1552 : poOGRFeature->SetField(
525 1552 : iDstField, psGMLProperty->papszSubProperties[0]);
526 1552 : break;
527 : }
528 : }
529 :
530 703 : delete poGMLFeature;
531 703 : poGMLFeature = nullptr;
532 :
533 : // Assign the geometry before the attribute filter because
534 : // the attribute filter may use a special field like OGR_GEOMETRY.
535 703 : if (papoGeometries != nullptr)
536 : {
537 301 : for (int i = 0; i < poFeatureDefn->GetGeomFieldCount(); i++)
538 : {
539 207 : poOGRFeature->SetGeomFieldDirectly(i, papoGeometries[i]);
540 : }
541 94 : CPLFree(papoGeometries);
542 94 : papoGeometries = nullptr;
543 : }
544 : else
545 : {
546 609 : poOGRFeature->SetGeometryDirectly(poGeom);
547 : }
548 :
549 : // Assign SRS.
550 1460 : for (int i = 0; i < poFeatureDefn->GetGeomFieldCount(); i++)
551 : {
552 757 : poGeom = poOGRFeature->GetGeomFieldRef(i);
553 757 : if (poGeom != nullptr)
554 : {
555 : const OGRSpatialReference *poSRS =
556 670 : poFeatureDefn->GetGeomFieldDefn(i)->GetSpatialRef();
557 670 : if (poSRS != nullptr)
558 137 : poGeom->assignSpatialReference(poSRS);
559 : }
560 : }
561 :
562 : /* --------------------------------------------------------------------
563 : */
564 : /* Test against the attribute query. */
565 : /* --------------------------------------------------------------------
566 : */
567 703 : if (m_poAttrQuery != nullptr && !m_poAttrQuery->Evaluate(poOGRFeature))
568 : {
569 28 : delete poOGRFeature;
570 28 : continue;
571 : }
572 :
573 : // Got the desired feature.
574 675 : return poOGRFeature;
575 658 : }
576 :
577 : return nullptr;
578 : }
579 :
580 : /************************************************************************/
581 : /* GetFeatureCount() */
582 : /************************************************************************/
583 :
584 62 : GIntBig OGRGMLLayer::GetFeatureCount(int bForce)
585 :
586 : {
587 62 : if (poFClass == nullptr)
588 0 : return 0;
589 :
590 62 : if (m_poFilterGeom != nullptr || m_poAttrQuery != nullptr)
591 6 : return OGRLayer::GetFeatureCount(bForce);
592 :
593 : // If the schema is read from a .xsd file, we haven't read
594 : // the feature count, so compute it now.
595 56 : GIntBig nFeatureCount = poFClass->GetFeatureCount();
596 56 : if (nFeatureCount < 0)
597 : {
598 26 : nFeatureCount = OGRLayer::GetFeatureCount(bForce);
599 26 : poFClass->SetFeatureCount(nFeatureCount);
600 : }
601 :
602 56 : return nFeatureCount;
603 : }
604 :
605 : /************************************************************************/
606 : /* GetExtent() */
607 : /************************************************************************/
608 :
609 10 : OGRErr OGRGMLLayer::GetExtent(OGREnvelope *psExtent, int bForce)
610 :
611 : {
612 10 : if (GetGeomType() == wkbNone)
613 0 : return OGRERR_FAILURE;
614 :
615 10 : double dfXMin = 0.0;
616 10 : double dfXMax = 0.0;
617 10 : double dfYMin = 0.0;
618 10 : double dfYMax = 0.0;
619 20 : if (poFClass != nullptr &&
620 10 : poFClass->GetExtents(&dfXMin, &dfXMax, &dfYMin, &dfYMax))
621 : {
622 8 : psExtent->MinX = dfXMin;
623 8 : psExtent->MaxX = dfXMax;
624 8 : psExtent->MinY = dfYMin;
625 8 : psExtent->MaxY = dfYMax;
626 :
627 8 : return OGRERR_NONE;
628 : }
629 :
630 2 : return OGRLayer::GetExtent(psExtent, bForce);
631 : }
632 :
633 : /************************************************************************/
634 : /* GetExtent() */
635 : /************************************************************************/
636 :
637 505 : static void GMLWriteField(OGRGMLDataSource *poDS, VSILFILE *fp,
638 : bool bWriteSpaceIndentation, const char *pszPrefix,
639 : bool bRemoveAppPrefix, OGRFieldDefn *poFieldDefn,
640 : const char *pszVal)
641 :
642 : {
643 505 : const char *pszFieldName = poFieldDefn->GetNameRef();
644 :
645 505 : while (*pszVal == ' ')
646 0 : pszVal++;
647 :
648 505 : if (bWriteSpaceIndentation)
649 505 : VSIFPrintfL(fp, " ");
650 :
651 505 : if (bRemoveAppPrefix)
652 60 : poDS->PrintLine(fp, "<%s>%s</%s>", pszFieldName, pszVal, pszFieldName);
653 : else
654 445 : poDS->PrintLine(fp, "<%s:%s>%s</%s:%s>", pszPrefix, pszFieldName,
655 : pszVal, pszPrefix, pszFieldName);
656 505 : }
657 :
658 : /************************************************************************/
659 : /* ICreateFeature() */
660 : /************************************************************************/
661 :
662 266 : OGRErr OGRGMLLayer::ICreateFeature(OGRFeature *poFeature)
663 :
664 : {
665 266 : const bool bIsGML3Output = poDS->IsGML3Output();
666 266 : VSILFILE *fp = poDS->GetOutputFP();
667 266 : const bool bWriteSpaceIndentation = poDS->WriteSpaceIndentation();
668 266 : const char *pszPrefix = poDS->GetAppPrefix();
669 266 : const bool bRemoveAppPrefix = poDS->RemoveAppPrefix();
670 266 : const bool bGMLFeatureCollection = poDS->GMLFeatureCollection();
671 :
672 266 : if (!bWriter || poDS->HasWriteError())
673 0 : return OGRERR_FAILURE;
674 :
675 266 : poFeature->FillUnsetWithDefault(TRUE, nullptr);
676 266 : if (!poFeature->Validate(OGR_F_VAL_ALL & ~OGR_F_VAL_GEOM_TYPE &
677 : ~OGR_F_VAL_ALLOW_NULL_WHEN_DEFAULT,
678 : TRUE))
679 2 : return OGRERR_FAILURE;
680 :
681 264 : if (bWriteSpaceIndentation)
682 264 : VSIFPrintfL(fp, " ");
683 264 : if (bIsGML3Output && !bGMLFeatureCollection)
684 : {
685 220 : if (bRemoveAppPrefix)
686 10 : poDS->PrintLine(fp, "<featureMember>");
687 : else
688 210 : poDS->PrintLine(fp, "<%s:featureMember>", pszPrefix);
689 : }
690 : else
691 : {
692 44 : poDS->PrintLine(fp, "<gml:featureMember>");
693 : }
694 :
695 264 : if (poFeature->GetFID() == OGRNullFID)
696 251 : poFeature->SetFID(iNextGMLId++);
697 :
698 264 : if (bWriteSpaceIndentation)
699 264 : VSIFPrintfL(fp, " ");
700 264 : VSIFPrintfL(fp, "<");
701 264 : if (!bRemoveAppPrefix)
702 244 : VSIFPrintfL(fp, "%s:", pszPrefix);
703 :
704 264 : int nGMLIdIndex = -1;
705 264 : if (bIsGML3Output)
706 : {
707 230 : nGMLIdIndex = poFeatureDefn->GetFieldIndex("gml_id");
708 230 : if (nGMLIdIndex >= 0 && poFeature->IsFieldSetAndNotNull(nGMLIdIndex))
709 3 : poDS->PrintLine(fp, "%s gml:id=\"%s\">", poFeatureDefn->GetName(),
710 : poFeature->GetFieldAsString(nGMLIdIndex));
711 : else
712 454 : poDS->PrintLine(fp, "%s gml:id=\"%s." CPL_FRMT_GIB "\">",
713 227 : poFeatureDefn->GetName(), poFeatureDefn->GetName(),
714 : poFeature->GetFID());
715 : }
716 : else
717 : {
718 34 : nGMLIdIndex = poFeatureDefn->GetFieldIndex("fid");
719 34 : if (bUseOldFIDFormat)
720 : {
721 0 : poDS->PrintLine(fp, "%s fid=\"F" CPL_FRMT_GIB "\">",
722 0 : poFeatureDefn->GetName(), poFeature->GetFID());
723 : }
724 44 : else if (nGMLIdIndex >= 0 &&
725 10 : poFeature->IsFieldSetAndNotNull(nGMLIdIndex))
726 : {
727 10 : poDS->PrintLine(fp, "%s fid=\"%s\">", poFeatureDefn->GetName(),
728 : poFeature->GetFieldAsString(nGMLIdIndex));
729 : }
730 : else
731 : {
732 48 : poDS->PrintLine(fp, "%s fid=\"%s." CPL_FRMT_GIB "\">",
733 24 : poFeatureDefn->GetName(), poFeatureDefn->GetName(),
734 : poFeature->GetFID());
735 : }
736 : }
737 :
738 517 : for (int iGeomField = 0; iGeomField < poFeatureDefn->GetGeomFieldCount();
739 : iGeomField++)
740 : {
741 : const OGRGeomFieldDefn *poFieldDefn =
742 253 : poFeatureDefn->GetGeomFieldDefn(iGeomField);
743 :
744 : // Write out Geometry - for now it isn't indented properly.
745 : // GML geometries don't like very much the concept of empty geometry.
746 253 : OGRGeometry *poGeom = poFeature->GetGeomFieldRef(iGeomField);
747 253 : if (poGeom != nullptr && !poGeom->IsEmpty())
748 : {
749 215 : OGREnvelope3D sGeomBounds;
750 :
751 215 : const int nCoordDimension = poGeom->getCoordinateDimension();
752 :
753 215 : poGeom->getEnvelope(&sGeomBounds);
754 215 : if (poDS->HasWriteGlobalSRS())
755 211 : poDS->GrowExtents(&sGeomBounds, nCoordDimension);
756 :
757 356 : if (poGeom->getSpatialReference() == nullptr &&
758 141 : poFieldDefn->GetSpatialRef() != nullptr)
759 18 : poGeom->assignSpatialReference(poFieldDefn->GetSpatialRef());
760 :
761 215 : const auto &oCoordPrec = poFieldDefn->GetCoordinatePrecision();
762 :
763 215 : if (bIsGML3Output && poDS->WriteFeatureBoundedBy())
764 : {
765 171 : bool bCoordSwap = false;
766 :
767 : char *pszSRSName =
768 171 : GML_GetSRSName(poGeom->getSpatialReference(),
769 171 : poDS->GetSRSNameFormat(), &bCoordSwap);
770 171 : char szLowerCorner[75] = {};
771 171 : char szUpperCorner[75] = {};
772 :
773 171 : OGRWktOptions coordOpts;
774 :
775 171 : if (oCoordPrec.dfXYResolution !=
776 : OGRGeomCoordinatePrecision::UNKNOWN)
777 : {
778 3 : coordOpts.format = OGRWktFormat::F;
779 3 : coordOpts.xyPrecision =
780 3 : OGRGeomCoordinatePrecision::ResolutionToPrecision(
781 3 : oCoordPrec.dfXYResolution);
782 : }
783 171 : if (oCoordPrec.dfZResolution !=
784 : OGRGeomCoordinatePrecision::UNKNOWN)
785 : {
786 3 : coordOpts.format = OGRWktFormat::F;
787 3 : coordOpts.zPrecision =
788 3 : OGRGeomCoordinatePrecision::ResolutionToPrecision(
789 3 : oCoordPrec.dfZResolution);
790 : }
791 :
792 342 : std::string wkt;
793 171 : if (bCoordSwap)
794 : {
795 36 : wkt = OGRMakeWktCoordinate(
796 : sGeomBounds.MinY, sGeomBounds.MinX, sGeomBounds.MinZ,
797 18 : nCoordDimension, coordOpts);
798 18 : memcpy(szLowerCorner, wkt.data(), wkt.size() + 1);
799 :
800 36 : wkt = OGRMakeWktCoordinate(
801 : sGeomBounds.MaxY, sGeomBounds.MaxX, sGeomBounds.MaxZ,
802 18 : nCoordDimension, coordOpts);
803 18 : memcpy(szUpperCorner, wkt.data(), wkt.size() + 1);
804 : }
805 : else
806 : {
807 306 : wkt = OGRMakeWktCoordinate(
808 : sGeomBounds.MinX, sGeomBounds.MinY, sGeomBounds.MinZ,
809 153 : nCoordDimension, coordOpts);
810 153 : memcpy(szLowerCorner, wkt.data(), wkt.size() + 1);
811 :
812 306 : wkt = OGRMakeWktCoordinate(
813 : sGeomBounds.MaxX, sGeomBounds.MaxY, sGeomBounds.MaxZ,
814 153 : nCoordDimension, coordOpts);
815 153 : memcpy(szUpperCorner, wkt.data(), wkt.size() + 1);
816 : }
817 171 : if (bWriteSpaceIndentation)
818 171 : VSIFPrintfL(fp, " ");
819 171 : poDS->PrintLine(
820 : fp,
821 : "<gml:boundedBy><gml:Envelope%s%s><gml:lowerCorner>%s"
822 : "</gml:lowerCorner><gml:upperCorner>%s</gml:upperCorner>"
823 : "</gml:Envelope></gml:boundedBy>",
824 : (nCoordDimension == 3) ? " srsDimension=\"3\"" : "",
825 : pszSRSName, szLowerCorner, szUpperCorner);
826 171 : CPLFree(pszSRSName);
827 : }
828 :
829 215 : char **papszOptions = nullptr;
830 215 : if (bIsGML3Output)
831 : {
832 181 : papszOptions = CSLAddString(papszOptions, "FORMAT=GML3");
833 181 : if (poDS->GetSRSNameFormat() == SRSNAME_SHORT)
834 : papszOptions =
835 1 : CSLAddString(papszOptions, "SRSNAME_FORMAT=SHORT");
836 180 : else if (poDS->GetSRSNameFormat() == SRSNAME_OGC_URN)
837 : papszOptions =
838 167 : CSLAddString(papszOptions, "SRSNAME_FORMAT=OGC_URN");
839 13 : else if (poDS->GetSRSNameFormat() == SRSNAME_OGC_URL)
840 : papszOptions =
841 13 : CSLAddString(papszOptions, "SRSNAME_FORMAT=OGC_URL");
842 : }
843 215 : const char *pszSRSDimensionLoc = poDS->GetSRSDimensionLoc();
844 215 : if (pszSRSDimensionLoc != nullptr)
845 3 : papszOptions = CSLSetNameValue(papszOptions, "SRSDIMENSION_LOC",
846 : pszSRSDimensionLoc);
847 215 : if (poDS->IsGML32Output())
848 : {
849 115 : if (poFeatureDefn->GetGeomFieldCount() > 1)
850 18 : papszOptions = CSLAddString(
851 : papszOptions, CPLSPrintf("GMLID=%s.%s." CPL_FRMT_GIB,
852 9 : poFeatureDefn->GetName(),
853 : poFieldDefn->GetNameRef(),
854 : poFeature->GetFID()));
855 : else
856 212 : papszOptions = CSLAddString(
857 : papszOptions, CPLSPrintf("GMLID=%s.geom." CPL_FRMT_GIB,
858 106 : poFeatureDefn->GetName(),
859 : poFeature->GetFID()));
860 : }
861 :
862 215 : if (oCoordPrec.dfXYResolution !=
863 : OGRGeomCoordinatePrecision::UNKNOWN)
864 : {
865 3 : papszOptions = CSLAddString(
866 : papszOptions, CPLSPrintf("XY_COORD_RESOLUTION=%g",
867 3 : oCoordPrec.dfXYResolution));
868 : }
869 215 : if (oCoordPrec.dfZResolution != OGRGeomCoordinatePrecision::UNKNOWN)
870 : {
871 3 : papszOptions = CSLAddString(
872 : papszOptions, CPLSPrintf("Z_COORD_RESOLUTION=%g",
873 3 : oCoordPrec.dfZResolution));
874 : }
875 :
876 215 : char *pszGeometry = nullptr;
877 215 : if (!bIsGML3Output && OGR_GT_IsNonLinear(poGeom->getGeometryType()))
878 : {
879 0 : OGRGeometry *poGeomTmp = OGRGeometryFactory::forceTo(
880 0 : poGeom->clone(),
881 0 : OGR_GT_GetLinear(poGeom->getGeometryType()));
882 0 : pszGeometry = poGeomTmp->exportToGML(papszOptions);
883 0 : delete poGeomTmp;
884 : }
885 : else
886 : {
887 215 : if (wkbFlatten(poGeom->getGeometryType()) == wkbTriangle)
888 : {
889 0 : pszGeometry = poGeom->exportToGML(papszOptions);
890 :
891 : const char *pszGMLID =
892 0 : poDS->IsGML32Output()
893 0 : ? CPLSPrintf(
894 : " gml:id=\"%s\"",
895 : CSLFetchNameValue(papszOptions, "GMLID"))
896 0 : : "";
897 0 : char *pszNewGeom = CPLStrdup(
898 : CPLSPrintf("<gml:TriangulatedSurface%s><gml:patches>%s<"
899 : "/gml:patches></gml:TriangulatedSurface>",
900 : pszGMLID, pszGeometry));
901 0 : CPLFree(pszGeometry);
902 0 : pszGeometry = pszNewGeom;
903 : }
904 : else
905 : {
906 215 : pszGeometry = poGeom->exportToGML(papszOptions);
907 : }
908 : }
909 215 : CSLDestroy(papszOptions);
910 215 : if (pszGeometry)
911 : {
912 214 : if (bWriteSpaceIndentation)
913 214 : VSIFPrintfL(fp, " ");
914 214 : if (bRemoveAppPrefix)
915 20 : poDS->PrintLine(fp, "<%s>%s</%s>",
916 : poFieldDefn->GetNameRef(), pszGeometry,
917 : poFieldDefn->GetNameRef());
918 : else
919 194 : poDS->PrintLine(fp, "<%s:%s>%s</%s:%s>", pszPrefix,
920 : poFieldDefn->GetNameRef(), pszGeometry,
921 : pszPrefix, poFieldDefn->GetNameRef());
922 : }
923 : else
924 : {
925 1 : CPLError(CE_Failure, CPLE_AppDefined,
926 : "Export of geometry to GML failed");
927 : }
928 215 : CPLFree(pszGeometry);
929 : }
930 : }
931 :
932 : // Write all "set" fields.
933 885 : for (int iField = 0; iField < poFeatureDefn->GetFieldCount(); iField++)
934 : {
935 621 : if (iField == nGMLIdIndex)
936 13 : continue;
937 608 : OGRFieldDefn *poFieldDefn = poFeatureDefn->GetFieldDefn(iField);
938 :
939 608 : if (poFeature->IsFieldNull(iField))
940 : {
941 1 : const char *pszFieldName = poFieldDefn->GetNameRef();
942 :
943 1 : if (bWriteSpaceIndentation)
944 1 : VSIFPrintfL(fp, " ");
945 :
946 1 : if (bRemoveAppPrefix)
947 0 : poDS->PrintLine(fp, "<%s xsi:nil=\"true\"/>", pszFieldName);
948 : else
949 1 : poDS->PrintLine(fp, "<%s:%s xsi:nil=\"true\"/>", pszPrefix,
950 : pszFieldName);
951 : }
952 607 : else if (poFeature->IsFieldSet(iField))
953 : {
954 500 : OGRFieldType eType = poFieldDefn->GetType();
955 500 : if (eType == OFTStringList)
956 : {
957 1 : char **papszIter = poFeature->GetFieldAsStringList(iField);
958 3 : while (papszIter != nullptr && *papszIter != nullptr)
959 : {
960 2 : char *pszEscaped = OGRGetXML_UTF8_EscapedString(*papszIter);
961 2 : GMLWriteField(poDS, fp, bWriteSpaceIndentation, pszPrefix,
962 : bRemoveAppPrefix, poFieldDefn, pszEscaped);
963 2 : CPLFree(pszEscaped);
964 :
965 2 : papszIter++;
966 : }
967 : }
968 499 : else if (eType == OFTIntegerList)
969 : {
970 2 : int nCount = 0;
971 : const int *panVals =
972 2 : poFeature->GetFieldAsIntegerList(iField, &nCount);
973 2 : if (poFieldDefn->GetSubType() == OFSTBoolean)
974 : {
975 3 : for (int i = 0; i < nCount; i++)
976 : {
977 : // 0 and 1 are OK, but the canonical representation is
978 : // false and true.
979 2 : GMLWriteField(poDS, fp, bWriteSpaceIndentation,
980 : pszPrefix, bRemoveAppPrefix, poFieldDefn,
981 2 : panVals[i] ? "true" : "false");
982 : }
983 : }
984 : else
985 : {
986 3 : for (int i = 0; i < nCount; i++)
987 : {
988 2 : GMLWriteField(poDS, fp, bWriteSpaceIndentation,
989 : pszPrefix, bRemoveAppPrefix, poFieldDefn,
990 2 : CPLSPrintf("%d", panVals[i]));
991 : }
992 : }
993 : }
994 497 : else if (eType == OFTInteger64List)
995 : {
996 2 : int nCount = 0;
997 : const GIntBig *panVals =
998 2 : poFeature->GetFieldAsInteger64List(iField, &nCount);
999 2 : if (poFieldDefn->GetSubType() == OFSTBoolean)
1000 : {
1001 0 : for (int i = 0; i < nCount; i++)
1002 : {
1003 : // 0 and 1 are OK, but the canonical representation is
1004 : // false and true.
1005 0 : GMLWriteField(poDS, fp, bWriteSpaceIndentation,
1006 : pszPrefix, bRemoveAppPrefix, poFieldDefn,
1007 0 : panVals[i] ? "true" : "false");
1008 : }
1009 : }
1010 : else
1011 : {
1012 5 : for (int i = 0; i < nCount; i++)
1013 : {
1014 3 : GMLWriteField(poDS, fp, bWriteSpaceIndentation,
1015 : pszPrefix, bRemoveAppPrefix, poFieldDefn,
1016 3 : CPLSPrintf(CPL_FRMT_GIB, panVals[i]));
1017 : }
1018 : }
1019 : }
1020 495 : else if (eType == OFTRealList)
1021 : {
1022 1 : int nCount = 0;
1023 : const double *padfVals =
1024 1 : poFeature->GetFieldAsDoubleList(iField, &nCount);
1025 3 : for (int i = 0; i < nCount; i++)
1026 : {
1027 2 : char szBuffer[80] = {};
1028 2 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.15g",
1029 2 : padfVals[i]);
1030 2 : GMLWriteField(poDS, fp, bWriteSpaceIndentation, pszPrefix,
1031 : bRemoveAppPrefix, poFieldDefn, szBuffer);
1032 : }
1033 : }
1034 625 : else if ((eType == OFTInteger || eType == OFTInteger64) &&
1035 131 : poFieldDefn->GetSubType() == OFSTBoolean)
1036 : {
1037 : // 0 and 1 are OK, but the canonical representation is false and
1038 : // true.
1039 2 : GMLWriteField(poDS, fp, bWriteSpaceIndentation, pszPrefix,
1040 : bRemoveAppPrefix, poFieldDefn,
1041 2 : (poFeature->GetFieldAsInteger(iField)) ? "true"
1042 : : "false");
1043 : }
1044 492 : else if (eType == OFTDate)
1045 : {
1046 49 : const OGRField *poField = poFeature->GetRawFieldRef(iField);
1047 : const char *pszXML =
1048 98 : CPLSPrintf("%04d-%02d-%02d", poField->Date.Year,
1049 49 : poField->Date.Month, poField->Date.Day);
1050 49 : GMLWriteField(poDS, fp, bWriteSpaceIndentation, pszPrefix,
1051 : bRemoveAppPrefix, poFieldDefn, pszXML);
1052 : }
1053 443 : else if (eType == OFTDateTime)
1054 : {
1055 : char *pszXML =
1056 49 : OGRGetXMLDateTime(poFeature->GetRawFieldRef(iField));
1057 49 : GMLWriteField(poDS, fp, bWriteSpaceIndentation, pszPrefix,
1058 : bRemoveAppPrefix, poFieldDefn, pszXML);
1059 49 : CPLFree(pszXML);
1060 : }
1061 : else
1062 : {
1063 394 : const char *pszRaw = poFeature->GetFieldAsString(iField);
1064 :
1065 394 : char *pszEscaped = OGRGetXML_UTF8_EscapedString(pszRaw);
1066 :
1067 394 : GMLWriteField(poDS, fp, bWriteSpaceIndentation, pszPrefix,
1068 : bRemoveAppPrefix, poFieldDefn, pszEscaped);
1069 394 : CPLFree(pszEscaped);
1070 : }
1071 : }
1072 : }
1073 :
1074 264 : if (bWriteSpaceIndentation)
1075 264 : VSIFPrintfL(fp, " ");
1076 264 : if (bRemoveAppPrefix)
1077 20 : poDS->PrintLine(fp, "</%s>", poFeatureDefn->GetName());
1078 : else
1079 244 : poDS->PrintLine(fp, "</%s:%s>", pszPrefix, poFeatureDefn->GetName());
1080 264 : if (bWriteSpaceIndentation)
1081 264 : VSIFPrintfL(fp, " ");
1082 264 : if (bIsGML3Output && !bGMLFeatureCollection)
1083 : {
1084 220 : if (bRemoveAppPrefix)
1085 10 : poDS->PrintLine(fp, "</featureMember>");
1086 : else
1087 210 : poDS->PrintLine(fp, "</%s:featureMember>", pszPrefix);
1088 : }
1089 : else
1090 : {
1091 44 : poDS->PrintLine(fp, "</gml:featureMember>");
1092 : }
1093 :
1094 264 : return !poDS->HasWriteError() ? OGRERR_NONE : OGRERR_FAILURE;
1095 : }
1096 :
1097 : /************************************************************************/
1098 : /* TestCapability() */
1099 : /************************************************************************/
1100 :
1101 340 : int OGRGMLLayer::TestCapability(const char *pszCap)
1102 :
1103 : {
1104 340 : if (EQUAL(pszCap, OLCSequentialWrite))
1105 19 : return bWriter;
1106 :
1107 321 : else if (EQUAL(pszCap, OLCCreateField))
1108 18 : return bWriter && iNextGMLId == 0;
1109 :
1110 303 : else if (EQUAL(pszCap, OLCCreateGeomField))
1111 4 : return bWriter && iNextGMLId == 0;
1112 :
1113 299 : else if (EQUAL(pszCap, OLCFastGetExtent))
1114 : {
1115 4 : if (poFClass == nullptr)
1116 0 : return FALSE;
1117 :
1118 4 : double dfXMin = 0.0;
1119 4 : double dfXMax = 0.0;
1120 4 : double dfYMin = 0.0;
1121 4 : double dfYMax = 0.0;
1122 :
1123 4 : return poFClass->GetExtents(&dfXMin, &dfXMax, &dfYMin, &dfYMax);
1124 : }
1125 :
1126 295 : else if (EQUAL(pszCap, OLCFastFeatureCount))
1127 : {
1128 1 : if (poFClass == nullptr || m_poFilterGeom != nullptr ||
1129 1 : m_poAttrQuery != nullptr)
1130 0 : return FALSE;
1131 :
1132 1 : return poFClass->GetFeatureCount() != -1;
1133 : }
1134 :
1135 294 : else if (EQUAL(pszCap, OLCStringsAsUTF8))
1136 14 : return TRUE;
1137 :
1138 280 : else if (EQUAL(pszCap, OLCCurveGeometries))
1139 146 : return poDS->IsGML3Output();
1140 :
1141 134 : else if (EQUAL(pszCap, OLCZGeometries))
1142 3 : return TRUE;
1143 :
1144 : else
1145 131 : return FALSE;
1146 : }
1147 :
1148 : /************************************************************************/
1149 : /* CreateField() */
1150 : /************************************************************************/
1151 :
1152 181 : OGRErr OGRGMLLayer::CreateField(const OGRFieldDefn *poField, int bApproxOK)
1153 :
1154 : {
1155 181 : if (!bWriter || iNextGMLId != 0)
1156 0 : return OGRERR_FAILURE;
1157 :
1158 : /* -------------------------------------------------------------------- */
1159 : /* Enforce XML naming semantics on element name. */
1160 : /* -------------------------------------------------------------------- */
1161 362 : OGRFieldDefn oCleanCopy(poField);
1162 181 : char *pszName = CPLStrdup(poField->GetNameRef());
1163 181 : CPLCleanXMLElementName(pszName);
1164 :
1165 181 : if (strcmp(pszName, poField->GetNameRef()) != 0)
1166 : {
1167 0 : if (!bApproxOK)
1168 : {
1169 0 : CPLFree(pszName);
1170 0 : CPLError(CE_Failure, CPLE_AppDefined,
1171 : "Unable to create field with name '%s', it would not\n"
1172 : "be valid as an XML element name.",
1173 : poField->GetNameRef());
1174 0 : return OGRERR_FAILURE;
1175 : }
1176 :
1177 0 : oCleanCopy.SetName(pszName);
1178 0 : CPLError(CE_Warning, CPLE_AppDefined,
1179 : "Field name '%s' adjusted to '%s' to be a valid\n"
1180 : "XML element name.",
1181 : poField->GetNameRef(), pszName);
1182 : }
1183 :
1184 181 : CPLFree(pszName);
1185 :
1186 181 : poFeatureDefn->AddFieldDefn(&oCleanCopy);
1187 :
1188 181 : return OGRERR_NONE;
1189 : }
1190 :
1191 : /************************************************************************/
1192 : /* CreateGeomField() */
1193 : /************************************************************************/
1194 :
1195 19 : OGRErr OGRGMLLayer::CreateGeomField(const OGRGeomFieldDefn *poField,
1196 : int bApproxOK)
1197 :
1198 : {
1199 19 : if (!bWriter || iNextGMLId != 0)
1200 0 : return OGRERR_FAILURE;
1201 :
1202 : /* -------------------------------------------------------------------- */
1203 : /* Enforce XML naming semantics on element name. */
1204 : /* -------------------------------------------------------------------- */
1205 38 : OGRGeomFieldDefn oCleanCopy(poField);
1206 19 : const auto poSRSOri = poField->GetSpatialRef();
1207 19 : poDS->DeclareNewWriteSRS(poSRSOri);
1208 19 : if (poSRSOri)
1209 : {
1210 6 : auto poSRS = poSRSOri->Clone();
1211 6 : poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
1212 6 : oCleanCopy.SetSpatialRef(poSRS);
1213 6 : poSRS->Release();
1214 : }
1215 19 : char *pszName = CPLStrdup(poField->GetNameRef());
1216 19 : CPLCleanXMLElementName(pszName);
1217 :
1218 19 : if (strcmp(pszName, poField->GetNameRef()) != 0)
1219 : {
1220 0 : if (!bApproxOK)
1221 : {
1222 0 : CPLFree(pszName);
1223 0 : CPLError(CE_Failure, CPLE_AppDefined,
1224 : "Unable to create field with name '%s', it would not\n"
1225 : "be valid as an XML element name.",
1226 : poField->GetNameRef());
1227 0 : return OGRERR_FAILURE;
1228 : }
1229 :
1230 0 : oCleanCopy.SetName(pszName);
1231 0 : CPLError(CE_Warning, CPLE_AppDefined,
1232 : "Field name '%s' adjusted to '%s' to be a valid\n"
1233 : "XML element name.",
1234 : poField->GetNameRef(), pszName);
1235 : }
1236 :
1237 19 : CPLFree(pszName);
1238 :
1239 19 : poFeatureDefn->AddGeomFieldDefn(&oCleanCopy);
1240 :
1241 19 : return OGRERR_NONE;
1242 : }
1243 :
1244 : /************************************************************************/
1245 : /* GetDataset() */
1246 : /************************************************************************/
1247 :
1248 18 : GDALDataset *OGRGMLLayer::GetDataset()
1249 : {
1250 18 : return poDS;
1251 : }
|