Line data Source code
1 : // SPDX-License-Identifier: MIT
2 : // Copyright 2007, Mateusz Loskot
3 : // Copyright 2008-2024, Even Rouault <even.rouault at spatialys.com>
4 :
5 : /*! @cond Doxygen_Suppress */
6 :
7 : #include "ogrgeojsongeometry.h"
8 : #include "ogrlibjsonutils.h"
9 :
10 : #include "ogr_geometry.h"
11 : #include "ogr_spatialref.h"
12 :
13 : static OGRPoint *OGRGeoJSONReadPoint(json_object *poObj);
14 : static OGRMultiPoint *OGRGeoJSONReadMultiPoint(json_object *poObj);
15 : static OGRLineString *OGRGeoJSONReadLineString(json_object *poObj,
16 : bool bRaw = false);
17 : static OGRMultiLineString *OGRGeoJSONReadMultiLineString(json_object *poObj);
18 : static OGRLinearRing *OGRGeoJSONReadLinearRing(json_object *poObj);
19 : static OGRMultiPolygon *OGRGeoJSONReadMultiPolygon(json_object *poObj);
20 : static OGRGeometryCollection *
21 : OGRGeoJSONReadGeometryCollection(json_object *poObj,
22 : OGRSpatialReference *poSRS = nullptr);
23 :
24 : /************************************************************************/
25 : /* OGRGeoJSONGetType */
26 : /************************************************************************/
27 :
28 2880 : GeoJSONObject::Type OGRGeoJSONGetType(json_object *poObj)
29 : {
30 2880 : if (nullptr == poObj)
31 0 : return GeoJSONObject::eUnknown;
32 :
33 2880 : json_object *poObjType = OGRGeoJSONFindMemberByName(poObj, "type");
34 2880 : if (nullptr == poObjType)
35 2 : return GeoJSONObject::eUnknown;
36 :
37 2878 : const char *name = json_object_get_string(poObjType);
38 2878 : if (EQUAL(name, "Point"))
39 367 : return GeoJSONObject::ePoint;
40 2511 : else if (EQUAL(name, "LineString"))
41 52 : return GeoJSONObject::eLineString;
42 2459 : else if (EQUAL(name, "Polygon"))
43 1005 : return GeoJSONObject::ePolygon;
44 1454 : else if (EQUAL(name, "MultiPoint"))
45 138 : return GeoJSONObject::eMultiPoint;
46 1316 : else if (EQUAL(name, "MultiLineString"))
47 34 : return GeoJSONObject::eMultiLineString;
48 1282 : else if (EQUAL(name, "MultiPolygon"))
49 415 : return GeoJSONObject::eMultiPolygon;
50 867 : else if (EQUAL(name, "GeometryCollection"))
51 14 : return GeoJSONObject::eGeometryCollection;
52 853 : else if (EQUAL(name, "Feature"))
53 567 : return GeoJSONObject::eFeature;
54 286 : else if (EQUAL(name, "FeatureCollection"))
55 286 : return GeoJSONObject::eFeatureCollection;
56 : else
57 0 : return GeoJSONObject::eUnknown;
58 : }
59 :
60 : /************************************************************************/
61 : /* OGRGeoJSONGetOGRGeometryType() */
62 : /************************************************************************/
63 :
64 4627 : OGRwkbGeometryType OGRGeoJSONGetOGRGeometryType(json_object *poObj)
65 : {
66 4627 : if (nullptr == poObj)
67 1 : return wkbUnknown;
68 :
69 4626 : json_object *poObjType = CPL_json_object_object_get(poObj, "type");
70 4626 : if (nullptr == poObjType)
71 0 : return wkbUnknown;
72 :
73 4626 : OGRwkbGeometryType eType = wkbUnknown;
74 4626 : const char *name = json_object_get_string(poObjType);
75 4626 : if (EQUAL(name, "Point"))
76 337 : eType = wkbPoint;
77 4289 : else if (EQUAL(name, "LineString"))
78 53 : eType = wkbLineString;
79 4236 : else if (EQUAL(name, "Polygon"))
80 3936 : eType = wkbPolygon;
81 300 : else if (EQUAL(name, "MultiPoint"))
82 25 : eType = wkbMultiPoint;
83 275 : else if (EQUAL(name, "MultiLineString"))
84 28 : eType = wkbMultiLineString;
85 247 : else if (EQUAL(name, "MultiPolygon"))
86 223 : eType = wkbMultiPolygon;
87 24 : else if (EQUAL(name, "GeometryCollection"))
88 19 : eType = wkbGeometryCollection;
89 : else
90 5 : return wkbUnknown;
91 :
92 : json_object *poCoordinates;
93 4621 : if (eType == wkbGeometryCollection)
94 : {
95 : json_object *poGeometries =
96 19 : CPL_json_object_object_get(poObj, "geometries");
97 37 : if (poGeometries &&
98 37 : json_object_get_type(poGeometries) == json_type_array &&
99 18 : json_object_array_length(poGeometries) > 0)
100 : {
101 17 : if (OGR_GT_HasZ(OGRGeoJSONGetOGRGeometryType(
102 17 : json_object_array_get_idx(poGeometries, 0))))
103 7 : eType = OGR_GT_SetZ(eType);
104 : }
105 : }
106 : else
107 : {
108 4602 : poCoordinates = CPL_json_object_object_get(poObj, "coordinates");
109 9198 : if (poCoordinates &&
110 9198 : json_object_get_type(poCoordinates) == json_type_array &&
111 4596 : json_object_array_length(poCoordinates) > 0)
112 : {
113 : while (true)
114 : {
115 13226 : auto poChild = json_object_array_get_idx(poCoordinates, 0);
116 26442 : if (!(poChild &&
117 13216 : json_object_get_type(poChild) == json_type_array &&
118 8639 : json_object_array_length(poChild) > 0))
119 : {
120 4591 : if (json_object_array_length(poCoordinates) == 3)
121 111 : eType = OGR_GT_SetZ(eType);
122 4591 : break;
123 : }
124 8635 : poCoordinates = poChild;
125 8635 : }
126 : }
127 : }
128 :
129 4621 : return eType;
130 : }
131 :
132 : /************************************************************************/
133 : /* OGRGeoJSONReadGeometry */
134 : /************************************************************************/
135 :
136 1900 : OGRGeometry *OGRGeoJSONReadGeometry(json_object *poObj,
137 : OGRSpatialReference *poParentSRS)
138 : {
139 :
140 1900 : OGRGeometry *poGeometry = nullptr;
141 1900 : OGRSpatialReference *poSRS = nullptr;
142 1900 : lh_entry *entry = OGRGeoJSONFindMemberEntryByName(poObj, "crs");
143 1900 : if (entry != nullptr)
144 : {
145 4 : json_object *poObjSrs =
146 : static_cast<json_object *>(const_cast<void *>(entry->v));
147 4 : if (poObjSrs != nullptr)
148 : {
149 3 : poSRS = OGRGeoJSONReadSpatialReference(poObj);
150 : }
151 : }
152 :
153 1900 : OGRSpatialReference *poSRSToAssign = nullptr;
154 1900 : if (entry != nullptr)
155 : {
156 4 : poSRSToAssign = poSRS;
157 : }
158 1896 : else if (poParentSRS)
159 : {
160 1283 : poSRSToAssign = poParentSRS;
161 : }
162 : else
163 : {
164 : // Assign WGS84 if no CRS defined on geometry.
165 613 : poSRSToAssign = OGRSpatialReference::GetWGS84SRS();
166 : }
167 :
168 1900 : GeoJSONObject::Type objType = OGRGeoJSONGetType(poObj);
169 1900 : if (GeoJSONObject::ePoint == objType)
170 346 : poGeometry = OGRGeoJSONReadPoint(poObj);
171 1554 : else if (GeoJSONObject::eMultiPoint == objType)
172 58 : poGeometry = OGRGeoJSONReadMultiPoint(poObj);
173 1496 : else if (GeoJSONObject::eLineString == objType)
174 48 : poGeometry = OGRGeoJSONReadLineString(poObj);
175 1448 : else if (GeoJSONObject::eMultiLineString == objType)
176 32 : poGeometry = OGRGeoJSONReadMultiLineString(poObj);
177 1416 : else if (GeoJSONObject::ePolygon == objType)
178 1001 : poGeometry = OGRGeoJSONReadPolygon(poObj);
179 415 : else if (GeoJSONObject::eMultiPolygon == objType)
180 405 : poGeometry = OGRGeoJSONReadMultiPolygon(poObj);
181 10 : else if (GeoJSONObject::eGeometryCollection == objType)
182 10 : poGeometry = OGRGeoJSONReadGeometryCollection(poObj, poSRSToAssign);
183 : else
184 : {
185 0 : CPLDebug("GeoJSON", "Unsupported geometry type detected. "
186 : "Feature gets NULL geometry assigned.");
187 : }
188 :
189 1900 : if (poGeometry && GeoJSONObject::eGeometryCollection != objType)
190 1876 : poGeometry->assignSpatialReference(poSRSToAssign);
191 :
192 1900 : if (poSRS)
193 3 : poSRS->Release();
194 :
195 1900 : return poGeometry;
196 : }
197 :
198 : /************************************************************************/
199 : /* OGRGeoJSONGetCoordinate() */
200 : /************************************************************************/
201 :
202 180287 : static double OGRGeoJSONGetCoordinate(json_object *poObj,
203 : const char *pszCoordName, int nIndex,
204 : bool &bValid)
205 : {
206 180287 : json_object *poObjCoord = json_object_array_get_idx(poObj, nIndex);
207 180287 : if (nullptr == poObjCoord)
208 : {
209 5 : CPLDebug("GeoJSON", "Point: got null object for %s.", pszCoordName);
210 5 : bValid = false;
211 5 : return 0.0;
212 : }
213 :
214 180282 : const int iType = json_object_get_type(poObjCoord);
215 180282 : if (json_type_double != iType && json_type_int != iType)
216 : {
217 0 : CPLError(CE_Failure, CPLE_AppDefined,
218 : "Invalid '%s' coordinate. "
219 : "Type is not double or integer for \'%s\'.",
220 : pszCoordName, json_object_to_json_string(poObjCoord));
221 0 : bValid = false;
222 0 : return 0.0;
223 : }
224 :
225 180282 : return json_object_get_double(poObjCoord);
226 : }
227 :
228 : /************************************************************************/
229 : /* OGRGeoJSONReadRawPoint */
230 : /************************************************************************/
231 :
232 89958 : static bool OGRGeoJSONReadRawPoint(json_object *poObj, OGRPoint &point)
233 : {
234 89958 : CPLAssert(nullptr != poObj);
235 :
236 89958 : if (json_type_array == json_object_get_type(poObj))
237 : {
238 89958 : const auto nSize = json_object_array_length(poObj);
239 :
240 89958 : if (nSize < GeoJSONObject::eMinCoordinateDimension)
241 : {
242 1 : CPLDebug("GeoJSON", "Invalid coord dimension. "
243 : "At least 2 dimensions must be present.");
244 1 : return false;
245 : }
246 :
247 89957 : bool bValid = true;
248 89957 : const double dfX = OGRGeoJSONGetCoordinate(poObj, "x", 0, bValid);
249 89957 : const double dfY = OGRGeoJSONGetCoordinate(poObj, "y", 1, bValid);
250 89957 : point.setX(dfX);
251 89957 : point.setY(dfY);
252 :
253 : // Read Z coordinate.
254 89957 : if (nSize >= GeoJSONObject::eMaxCoordinateDimension)
255 : {
256 : // Don't *expect* mixed-dimension geometries, although the
257 : // spec doesn't explicitly forbid this.
258 373 : const double dfZ = OGRGeoJSONGetCoordinate(poObj, "z", 2, bValid);
259 373 : point.setZ(dfZ);
260 : }
261 : else
262 : {
263 89584 : point.flattenTo2D();
264 : }
265 89957 : return bValid;
266 : }
267 :
268 0 : return false;
269 : }
270 :
271 : /************************************************************************/
272 : /* OGRGeoJSONReadPoint */
273 : /************************************************************************/
274 :
275 346 : OGRPoint *OGRGeoJSONReadPoint(json_object *poObj)
276 : {
277 346 : CPLAssert(nullptr != poObj);
278 :
279 346 : json_object *poObjCoords = OGRGeoJSONFindMemberByName(poObj, "coordinates");
280 346 : if (nullptr == poObjCoords)
281 : {
282 2 : CPLError(CE_Failure, CPLE_AppDefined,
283 : "Invalid Point object. Missing \'coordinates\' member.");
284 2 : return nullptr;
285 : }
286 :
287 344 : OGRPoint *poPoint = new OGRPoint();
288 344 : if (!OGRGeoJSONReadRawPoint(poObjCoords, *poPoint))
289 : {
290 5 : CPLDebug("GeoJSON", "Point: raw point parsing failure.");
291 5 : delete poPoint;
292 5 : return nullptr;
293 : }
294 :
295 339 : return poPoint;
296 : }
297 :
298 : /************************************************************************/
299 : /* OGRGeoJSONReadMultiPoint */
300 : /************************************************************************/
301 :
302 58 : OGRMultiPoint *OGRGeoJSONReadMultiPoint(json_object *poObj)
303 : {
304 58 : CPLAssert(nullptr != poObj);
305 :
306 58 : json_object *poObjPoints = OGRGeoJSONFindMemberByName(poObj, "coordinates");
307 58 : if (nullptr == poObjPoints)
308 : {
309 1 : CPLError(CE_Failure, CPLE_AppDefined,
310 : "Invalid MultiPoint object. "
311 : "Missing \'coordinates\' member.");
312 1 : return nullptr;
313 : }
314 :
315 57 : OGRMultiPoint *poMultiPoint = nullptr;
316 57 : if (json_type_array == json_object_get_type(poObjPoints))
317 : {
318 57 : const auto nPoints = json_object_array_length(poObjPoints);
319 :
320 57 : poMultiPoint = new OGRMultiPoint();
321 :
322 259 : for (auto i = decltype(nPoints){0}; i < nPoints; ++i)
323 : {
324 : json_object *poObjCoords =
325 202 : json_object_array_get_idx(poObjPoints, i);
326 :
327 202 : OGRPoint pt;
328 401 : if (poObjCoords != nullptr &&
329 199 : !OGRGeoJSONReadRawPoint(poObjCoords, pt))
330 : {
331 0 : delete poMultiPoint;
332 0 : CPLDebug("GeoJSON", "LineString: raw point parsing failure.");
333 0 : return nullptr;
334 : }
335 202 : poMultiPoint->addGeometry(&pt);
336 : }
337 : }
338 :
339 57 : return poMultiPoint;
340 : }
341 :
342 : /************************************************************************/
343 : /* OGRGeoJSONReadLineString */
344 : /************************************************************************/
345 :
346 101 : OGRLineString *OGRGeoJSONReadLineString(json_object *poObj, bool bRaw)
347 : {
348 101 : CPLAssert(nullptr != poObj);
349 :
350 101 : json_object *poObjPoints = nullptr;
351 :
352 101 : if (!bRaw)
353 : {
354 48 : poObjPoints = OGRGeoJSONFindMemberByName(poObj, "coordinates");
355 48 : if (nullptr == poObjPoints)
356 : {
357 1 : CPLError(CE_Failure, CPLE_AppDefined,
358 : "Invalid LineString object. "
359 : "Missing \'coordinates\' member.");
360 1 : return nullptr;
361 : }
362 : }
363 : else
364 : {
365 53 : poObjPoints = poObj;
366 : }
367 :
368 100 : OGRLineString *poLine = nullptr;
369 :
370 100 : if (json_type_array == json_object_get_type(poObjPoints))
371 : {
372 100 : const auto nPoints = json_object_array_length(poObjPoints);
373 :
374 100 : poLine = new OGRLineString();
375 100 : poLine->setNumPoints(static_cast<int>(nPoints));
376 :
377 295 : for (auto i = decltype(nPoints){0}; i < nPoints; ++i)
378 : {
379 : json_object *poObjCoords =
380 199 : json_object_array_get_idx(poObjPoints, i);
381 199 : if (poObjCoords == nullptr)
382 : {
383 4 : delete poLine;
384 4 : CPLDebug("GeoJSON", "LineString: got null object.");
385 4 : return nullptr;
386 : }
387 :
388 195 : OGRPoint pt;
389 195 : if (!OGRGeoJSONReadRawPoint(poObjCoords, pt))
390 : {
391 0 : delete poLine;
392 0 : CPLDebug("GeoJSON", "LineString: raw point parsing failure.");
393 0 : return nullptr;
394 : }
395 195 : if (pt.getCoordinateDimension() == 2)
396 : {
397 169 : poLine->setPoint(static_cast<int>(i), pt.getX(), pt.getY());
398 : }
399 : else
400 : {
401 26 : poLine->setPoint(static_cast<int>(i), pt.getX(), pt.getY(),
402 : pt.getZ());
403 : }
404 : }
405 : }
406 :
407 96 : return poLine;
408 : }
409 :
410 : /************************************************************************/
411 : /* OGRGeoJSONReadMultiLineString */
412 : /************************************************************************/
413 :
414 32 : OGRMultiLineString *OGRGeoJSONReadMultiLineString(json_object *poObj)
415 : {
416 32 : CPLAssert(nullptr != poObj);
417 :
418 32 : json_object *poObjLines = OGRGeoJSONFindMemberByName(poObj, "coordinates");
419 32 : if (nullptr == poObjLines)
420 : {
421 1 : CPLError(CE_Failure, CPLE_AppDefined,
422 : "Invalid MultiLineString object. "
423 : "Missing \'coordinates\' member.");
424 1 : return nullptr;
425 : }
426 :
427 31 : OGRMultiLineString *poMultiLine = nullptr;
428 :
429 31 : if (json_type_array == json_object_get_type(poObjLines))
430 : {
431 31 : const auto nLines = json_object_array_length(poObjLines);
432 :
433 31 : poMultiLine = new OGRMultiLineString();
434 :
435 86 : for (auto i = decltype(nLines){0}; i < nLines; ++i)
436 : {
437 55 : json_object *poObjLine = json_object_array_get_idx(poObjLines, i);
438 :
439 : OGRLineString *poLine;
440 55 : if (poObjLine != nullptr)
441 53 : poLine = OGRGeoJSONReadLineString(poObjLine, true);
442 : else
443 2 : poLine = new OGRLineString();
444 :
445 55 : if (nullptr != poLine)
446 : {
447 53 : poMultiLine->addGeometryDirectly(poLine);
448 : }
449 : }
450 : }
451 :
452 31 : return poMultiLine;
453 : }
454 :
455 : /************************************************************************/
456 : /* OGRGeoJSONReadLinearRing */
457 : /************************************************************************/
458 :
459 1687 : OGRLinearRing *OGRGeoJSONReadLinearRing(json_object *poObj)
460 : {
461 1687 : CPLAssert(nullptr != poObj);
462 :
463 1687 : OGRLinearRing *poRing = nullptr;
464 :
465 1687 : if (json_type_array == json_object_get_type(poObj))
466 : {
467 1687 : const auto nPoints = json_object_array_length(poObj);
468 :
469 1687 : poRing = new OGRLinearRing();
470 1687 : poRing->setNumPoints(static_cast<int>(nPoints));
471 :
472 90907 : for (auto i = decltype(nPoints){0}; i < nPoints; ++i)
473 : {
474 89220 : json_object *poObjCoords = json_object_array_get_idx(poObj, i);
475 89220 : if (poObjCoords == nullptr)
476 : {
477 0 : delete poRing;
478 0 : CPLDebug("GeoJSON", "LinearRing: got null object.");
479 0 : return nullptr;
480 : }
481 :
482 89220 : OGRPoint pt;
483 89220 : if (!OGRGeoJSONReadRawPoint(poObjCoords, pt))
484 : {
485 0 : delete poRing;
486 0 : CPLDebug("GeoJSON", "LinearRing: raw point parsing failure.");
487 0 : return nullptr;
488 : }
489 :
490 89220 : if (2 == pt.getCoordinateDimension())
491 89079 : poRing->setPoint(static_cast<int>(i), pt.getX(), pt.getY());
492 : else
493 141 : poRing->setPoint(static_cast<int>(i), pt.getX(), pt.getY(),
494 : pt.getZ());
495 : }
496 : }
497 :
498 1687 : return poRing;
499 : }
500 :
501 : /************************************************************************/
502 : /* OGRGeoJSONReadPolygon */
503 : /************************************************************************/
504 :
505 1654 : OGRPolygon *OGRGeoJSONReadPolygon(json_object *poObj, bool bRaw)
506 : {
507 1654 : CPLAssert(nullptr != poObj);
508 :
509 1654 : json_object *poObjRings = nullptr;
510 :
511 1654 : if (!bRaw)
512 : {
513 1001 : poObjRings = OGRGeoJSONFindMemberByName(poObj, "coordinates");
514 1001 : if (nullptr == poObjRings)
515 : {
516 1 : CPLError(CE_Failure, CPLE_AppDefined,
517 : "Invalid Polygon object. "
518 : "Missing \'coordinates\' member.");
519 1 : return nullptr;
520 : }
521 : }
522 : else
523 : {
524 653 : poObjRings = poObj;
525 : }
526 :
527 1653 : OGRPolygon *poPolygon = nullptr;
528 :
529 1653 : if (json_type_array == json_object_get_type(poObjRings))
530 : {
531 1653 : const auto nRings = json_object_array_length(poObjRings);
532 1653 : if (nRings > 0)
533 : {
534 1649 : json_object *poObjPoints = json_object_array_get_idx(poObjRings, 0);
535 1649 : if (poObjPoints == nullptr)
536 : {
537 2 : poPolygon = new OGRPolygon();
538 : }
539 : else
540 : {
541 1647 : OGRLinearRing *poRing = OGRGeoJSONReadLinearRing(poObjPoints);
542 1647 : if (nullptr != poRing)
543 : {
544 1647 : poPolygon = new OGRPolygon();
545 1647 : poPolygon->addRingDirectly(poRing);
546 : }
547 : }
548 :
549 1690 : for (auto i = decltype(nRings){1};
550 1690 : i < nRings && nullptr != poPolygon; ++i)
551 : {
552 41 : poObjPoints = json_object_array_get_idx(poObjRings, i);
553 41 : if (poObjPoints != nullptr)
554 : {
555 : OGRLinearRing *poRing =
556 40 : OGRGeoJSONReadLinearRing(poObjPoints);
557 40 : if (nullptr != poRing)
558 : {
559 40 : poPolygon->addRingDirectly(poRing);
560 : }
561 : }
562 : }
563 : }
564 : else
565 : {
566 4 : poPolygon = new OGRPolygon();
567 : }
568 : }
569 :
570 1653 : return poPolygon;
571 : }
572 :
573 : /************************************************************************/
574 : /* OGRGeoJSONReadMultiPolygon */
575 : /************************************************************************/
576 :
577 405 : OGRMultiPolygon *OGRGeoJSONReadMultiPolygon(json_object *poObj)
578 : {
579 405 : CPLAssert(nullptr != poObj);
580 :
581 405 : json_object *poObjPolys = OGRGeoJSONFindMemberByName(poObj, "coordinates");
582 405 : if (nullptr == poObjPolys)
583 : {
584 1 : CPLError(CE_Failure, CPLE_AppDefined,
585 : "Invalid MultiPolygon object. "
586 : "Missing \'coordinates\' member.");
587 1 : return nullptr;
588 : }
589 :
590 404 : OGRMultiPolygon *poMultiPoly = nullptr;
591 :
592 404 : if (json_type_array == json_object_get_type(poObjPolys))
593 : {
594 404 : const auto nPolys = json_object_array_length(poObjPolys);
595 :
596 404 : poMultiPoly = new OGRMultiPolygon();
597 :
598 1056 : for (auto i = decltype(nPolys){0}; i < nPolys; ++i)
599 : {
600 652 : json_object *poObjPoly = json_object_array_get_idx(poObjPolys, i);
601 652 : if (poObjPoly == nullptr)
602 : {
603 3 : poMultiPoly->addGeometryDirectly(new OGRPolygon());
604 : }
605 : else
606 : {
607 649 : OGRPolygon *poPoly = OGRGeoJSONReadPolygon(poObjPoly, true);
608 649 : if (nullptr != poPoly)
609 : {
610 649 : poMultiPoly->addGeometryDirectly(poPoly);
611 : }
612 : }
613 : }
614 : }
615 :
616 404 : return poMultiPoly;
617 : }
618 :
619 : /************************************************************************/
620 : /* OGRGeoJSONReadGeometryCollection */
621 : /************************************************************************/
622 :
623 : OGRGeometryCollection *
624 10 : OGRGeoJSONReadGeometryCollection(json_object *poObj, OGRSpatialReference *poSRS)
625 : {
626 10 : CPLAssert(nullptr != poObj);
627 :
628 10 : json_object *poObjGeoms = OGRGeoJSONFindMemberByName(poObj, "geometries");
629 10 : if (nullptr == poObjGeoms)
630 : {
631 1 : CPLError(CE_Failure, CPLE_AppDefined,
632 : "Invalid GeometryCollection object. "
633 : "Missing \'geometries\' member.");
634 1 : return nullptr;
635 : }
636 :
637 9 : OGRGeometryCollection *poCollection = nullptr;
638 :
639 9 : if (json_type_array == json_object_get_type(poObjGeoms))
640 : {
641 9 : poCollection = new OGRGeometryCollection();
642 9 : poCollection->assignSpatialReference(poSRS);
643 :
644 9 : const auto nGeoms = json_object_array_length(poObjGeoms);
645 25 : for (auto i = decltype(nGeoms){0}; i < nGeoms; ++i)
646 : {
647 16 : json_object *poObjGeom = json_object_array_get_idx(poObjGeoms, i);
648 16 : if (poObjGeom == nullptr)
649 : {
650 3 : CPLDebug("GeoJSON", "Skipping null sub-geometry");
651 3 : continue;
652 : }
653 :
654 13 : OGRGeometry *poGeometry = OGRGeoJSONReadGeometry(poObjGeom, poSRS);
655 13 : if (nullptr != poGeometry)
656 : {
657 13 : poCollection->addGeometryDirectly(poGeometry);
658 : }
659 : }
660 : }
661 :
662 9 : return poCollection;
663 : }
664 :
665 : /************************************************************************/
666 : /* OGRGeoJSONGetGeometryName() */
667 : /************************************************************************/
668 :
669 1668 : const char *OGRGeoJSONGetGeometryName(OGRGeometry const *poGeometry)
670 : {
671 1668 : CPLAssert(nullptr != poGeometry);
672 :
673 1668 : const OGRwkbGeometryType eType = wkbFlatten(poGeometry->getGeometryType());
674 :
675 1668 : if (wkbPoint == eType)
676 161 : return "Point";
677 1507 : else if (wkbLineString == eType)
678 56 : return "LineString";
679 1451 : else if (wkbPolygon == eType)
680 1282 : return "Polygon";
681 169 : else if (wkbMultiPoint == eType)
682 66 : return "MultiPoint";
683 103 : else if (wkbMultiLineString == eType)
684 34 : return "MultiLineString";
685 69 : else if (wkbMultiPolygon == eType)
686 43 : return "MultiPolygon";
687 26 : else if (wkbGeometryCollection == eType)
688 25 : return "GeometryCollection";
689 :
690 1 : return "Unknown";
691 : }
692 :
693 : /************************************************************************/
694 : /* OGRGeoJSONReadSpatialReference */
695 : /************************************************************************/
696 :
697 458 : OGRSpatialReference *OGRGeoJSONReadSpatialReference(json_object *poObj)
698 : {
699 :
700 : /* -------------------------------------------------------------------- */
701 : /* Read spatial reference definition. */
702 : /* -------------------------------------------------------------------- */
703 458 : OGRSpatialReference *poSRS = nullptr;
704 :
705 458 : json_object *poObjSrs = OGRGeoJSONFindMemberByName(poObj, "crs");
706 458 : if (nullptr != poObjSrs)
707 : {
708 : json_object *poObjSrsType =
709 70 : OGRGeoJSONFindMemberByName(poObjSrs, "type");
710 70 : if (poObjSrsType == nullptr)
711 1 : return nullptr;
712 :
713 69 : const char *pszSrsType = json_object_get_string(poObjSrsType);
714 :
715 : // TODO: Add URL and URN types support.
716 69 : if (STARTS_WITH_CI(pszSrsType, "NAME"))
717 : {
718 : json_object *poObjSrsProps =
719 49 : OGRGeoJSONFindMemberByName(poObjSrs, "properties");
720 49 : if (poObjSrsProps == nullptr)
721 2 : return nullptr;
722 :
723 : json_object *poNameURL =
724 47 : OGRGeoJSONFindMemberByName(poObjSrsProps, "name");
725 47 : if (poNameURL == nullptr)
726 2 : return nullptr;
727 :
728 45 : const char *pszName = json_object_get_string(poNameURL);
729 :
730 : // Mostly to emulate GDAL 2.x behavior
731 : // See https://github.com/OSGeo/gdal/issues/2035
732 45 : if (EQUAL(pszName, "urn:ogc:def:crs:OGC:1.3:CRS84"))
733 9 : pszName = "EPSG:4326";
734 :
735 45 : poSRS = new OGRSpatialReference();
736 45 : poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
737 45 : if (OGRERR_NONE !=
738 45 : poSRS->SetFromUserInput(
739 : pszName,
740 : OGRSpatialReference::SET_FROM_USER_INPUT_LIMITATIONS_get()))
741 : {
742 2 : delete poSRS;
743 2 : poSRS = nullptr;
744 : }
745 : }
746 :
747 20 : else if (STARTS_WITH_CI(pszSrsType, "EPSG"))
748 : {
749 : json_object *poObjSrsProps =
750 7 : OGRGeoJSONFindMemberByName(poObjSrs, "properties");
751 7 : if (poObjSrsProps == nullptr)
752 2 : return nullptr;
753 :
754 : json_object *poObjCode =
755 5 : OGRGeoJSONFindMemberByName(poObjSrsProps, "code");
756 5 : if (poObjCode == nullptr)
757 2 : return nullptr;
758 :
759 3 : int nEPSG = json_object_get_int(poObjCode);
760 :
761 3 : poSRS = new OGRSpatialReference();
762 3 : poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
763 3 : if (OGRERR_NONE != poSRS->importFromEPSG(nEPSG))
764 : {
765 2 : delete poSRS;
766 2 : poSRS = nullptr;
767 : }
768 : }
769 :
770 13 : else if (STARTS_WITH_CI(pszSrsType, "URL") ||
771 13 : STARTS_WITH_CI(pszSrsType, "LINK"))
772 : {
773 : json_object *poObjSrsProps =
774 6 : OGRGeoJSONFindMemberByName(poObjSrs, "properties");
775 6 : if (poObjSrsProps == nullptr)
776 2 : return nullptr;
777 :
778 : json_object *poObjURL =
779 4 : OGRGeoJSONFindMemberByName(poObjSrsProps, "url");
780 :
781 4 : if (nullptr == poObjURL)
782 : {
783 4 : poObjURL = OGRGeoJSONFindMemberByName(poObjSrsProps, "href");
784 : }
785 4 : if (poObjURL == nullptr)
786 2 : return nullptr;
787 :
788 2 : const char *pszURL = json_object_get_string(poObjURL);
789 :
790 2 : poSRS = new OGRSpatialReference();
791 2 : poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
792 2 : if (OGRERR_NONE != poSRS->importFromUrl(pszURL))
793 : {
794 2 : delete poSRS;
795 2 : poSRS = nullptr;
796 2 : }
797 : }
798 :
799 7 : else if (EQUAL(pszSrsType, "OGC"))
800 : {
801 : json_object *poObjSrsProps =
802 7 : OGRGeoJSONFindMemberByName(poObjSrs, "properties");
803 7 : if (poObjSrsProps == nullptr)
804 2 : return nullptr;
805 :
806 : json_object *poObjURN =
807 5 : OGRGeoJSONFindMemberByName(poObjSrsProps, "urn");
808 5 : if (poObjURN == nullptr)
809 2 : return nullptr;
810 :
811 3 : poSRS = new OGRSpatialReference();
812 3 : poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
813 3 : if (OGRERR_NONE !=
814 3 : poSRS->importFromURN(json_object_get_string(poObjURN)))
815 : {
816 2 : delete poSRS;
817 2 : poSRS = nullptr;
818 : }
819 : }
820 : }
821 :
822 : // Strip AXIS, since geojson has (easting, northing) / (longitude, latitude)
823 : // order. According to http://www.geojson.org/geojson-spec.html#id2 :
824 : // "Point coordinates are in x, y order (easting, northing for projected
825 : // coordinates, longitude, latitude for geographic coordinates)".
826 441 : if (poSRS != nullptr)
827 : {
828 45 : OGR_SRSNode *poGEOGCS = poSRS->GetAttrNode("GEOGCS");
829 45 : if (poGEOGCS != nullptr)
830 45 : poGEOGCS->StripNodes("AXIS");
831 : }
832 :
833 441 : return poSRS;
834 : }
835 :
836 : /************************************************************************/
837 : /* OGR_G_CreateGeometryFromJson */
838 : /************************************************************************/
839 :
840 : /** Create a OGR geometry from a GeoJSON geometry object */
841 46 : OGRGeometryH OGR_G_CreateGeometryFromJson(const char *pszJson)
842 : {
843 46 : if (nullptr == pszJson)
844 : {
845 : // Translation failed.
846 0 : return nullptr;
847 : }
848 :
849 46 : json_object *poObj = nullptr;
850 46 : if (!OGRJSonParse(pszJson, &poObj))
851 0 : return nullptr;
852 :
853 46 : OGRGeometry *poGeometry = OGRGeoJSONReadGeometry(poObj);
854 :
855 : // Release JSON tree.
856 46 : json_object_put(poObj);
857 :
858 46 : return OGRGeometry::ToHandle(poGeometry);
859 : }
860 :
861 : /*! @endcond */
|