Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: OpenGIS Simple Features Reference Implementation
4 : * Purpose: GeoJSON feature sequence driver
5 : * Author: Even Rouault <even.rouault at spatialys.com>
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2018, Even Rouault <even.rouault at spatialys.com>
9 : *
10 : * SPDX-License-Identifier: MIT
11 : ****************************************************************************/
12 :
13 : #include "cpl_port.h"
14 : #include "cpl_vsi_virtual.h"
15 : #include "cpl_http.h"
16 : #include "cpl_vsi_error.h"
17 :
18 : #include "ogr_geojson.h"
19 : #include "ogrlibjsonutils.h"
20 : #include "ogrgeojsonreader.h"
21 : #include "ogrgeojsonwriter.h"
22 : #include "ogrgeojsongeometry.h"
23 :
24 : #include <algorithm>
25 : #include <memory>
26 :
27 : constexpr char RS = '\x1e';
28 :
29 : /************************************************************************/
30 : /* OGRGeoJSONSeqDataSource */
31 : /************************************************************************/
32 :
33 : class OGRGeoJSONSeqDataSource final : public GDALDataset
34 : {
35 : friend class OGRGeoJSONSeqLayer;
36 :
37 : std::vector<std::unique_ptr<OGRLayer>> m_apoLayers{};
38 : CPLString m_osTmpFile;
39 : VSILFILE *m_fp = nullptr;
40 : bool m_bSupportsRead = true;
41 : bool m_bAtEOF = false;
42 : bool m_bIsRSSeparated = false;
43 :
44 : public:
45 : OGRGeoJSONSeqDataSource();
46 : ~OGRGeoJSONSeqDataSource();
47 :
48 76 : int GetLayerCount() override
49 : {
50 76 : return static_cast<int>(m_apoLayers.size());
51 : }
52 :
53 : OGRLayer *GetLayer(int) override;
54 : OGRLayer *ICreateLayer(const char *pszName,
55 : const OGRGeomFieldDefn *poGeomFieldDefn,
56 : CSLConstList papszOptions) override;
57 : int TestCapability(const char *pszCap) override;
58 :
59 : bool Open(GDALOpenInfo *poOpenInfo, GeoJSONSourceType nSrcType);
60 : bool Create(const char *pszName, char **papszOptions);
61 : };
62 :
63 : /************************************************************************/
64 : /* OGRGeoJSONSeqLayer */
65 : /************************************************************************/
66 :
67 : class OGRGeoJSONSeqLayer final : public OGRLayer
68 : {
69 : OGRGeoJSONSeqDataSource *m_poDS = nullptr;
70 : OGRFeatureDefn *m_poFeatureDefn = nullptr;
71 : bool m_bLayerDefnEstablished = false;
72 : bool m_bWriteOnlyLayer = false;
73 :
74 : OGRGeoJSONBaseReader m_oReader;
75 : CPLString m_osFIDColumn;
76 :
77 : size_t m_nMaxObjectSize = 0;
78 : std::string m_osBuffer;
79 : std::string m_osFeatureBuffer;
80 : size_t m_nPosInBuffer = 0;
81 : size_t m_nBufferValidSize = 0;
82 :
83 : vsi_l_offset m_nFileSize = 0;
84 : GIntBig m_nIter = 0;
85 :
86 : GIntBig m_nTotalFeatures = 0;
87 : GIntBig m_nNextFID = 0;
88 :
89 : std::unique_ptr<OGRCoordinateTransformation> m_poCT{};
90 : OGRGeometryFactory::TransformWithOptionsCache m_oTransformCache;
91 : OGRGeoJSONWriteOptions m_oWriteOptions;
92 :
93 : json_object *GetNextObject(bool bLooseIdentification);
94 :
95 : public:
96 : OGRGeoJSONSeqLayer(OGRGeoJSONSeqDataSource *poDS, const char *pszName);
97 :
98 : // Write-only constructor
99 : OGRGeoJSONSeqLayer(OGRGeoJSONSeqDataSource *poDS, const char *pszName,
100 : CSLConstList papszOptions,
101 : std::unique_ptr<OGRCoordinateTransformation> &&poCT);
102 :
103 : ~OGRGeoJSONSeqLayer();
104 :
105 : bool Init(bool bLooseIdentification, bool bEstablishLayerDefn);
106 :
107 61 : const char *GetName() override
108 : {
109 61 : return GetDescription();
110 : }
111 :
112 : void ResetReading() override;
113 : OGRFeature *GetNextFeature() override;
114 : OGRFeatureDefn *GetLayerDefn() override;
115 :
116 30 : const char *GetFIDColumn() override
117 : {
118 30 : return m_osFIDColumn.c_str();
119 : }
120 :
121 : GIntBig GetFeatureCount(int) override;
122 : int TestCapability(const char *) override;
123 : OGRErr ICreateFeature(OGRFeature *poFeature) override;
124 : OGRErr CreateField(const OGRFieldDefn *, int) override;
125 :
126 22 : GDALDataset *GetDataset() override
127 : {
128 22 : return m_poDS;
129 : }
130 : };
131 :
132 : /************************************************************************/
133 : /* OGRGeoJSONSeqDataSource() */
134 : /************************************************************************/
135 :
136 99 : OGRGeoJSONSeqDataSource::OGRGeoJSONSeqDataSource()
137 : {
138 99 : }
139 :
140 : /************************************************************************/
141 : /* ~OGRGeoJSONSeqDataSource() */
142 : /************************************************************************/
143 :
144 198 : OGRGeoJSONSeqDataSource::~OGRGeoJSONSeqDataSource()
145 : {
146 99 : if (m_fp)
147 : {
148 89 : VSIFCloseL(m_fp);
149 : }
150 99 : if (!m_osTmpFile.empty())
151 : {
152 3 : VSIUnlink(m_osTmpFile);
153 : }
154 198 : }
155 :
156 : /************************************************************************/
157 : /* GetLayer() */
158 : /************************************************************************/
159 :
160 44 : OGRLayer *OGRGeoJSONSeqDataSource::GetLayer(int nIndex)
161 : {
162 44 : if (nIndex < 0 || nIndex >= GetLayerCount())
163 2 : return nullptr;
164 42 : return m_apoLayers[nIndex].get();
165 : }
166 :
167 : /************************************************************************/
168 : /* ICreateLayer() */
169 : /************************************************************************/
170 :
171 65 : OGRLayer *OGRGeoJSONSeqDataSource::ICreateLayer(
172 : const char *pszNameIn, const OGRGeomFieldDefn *poSrcGeomFieldDefn,
173 : CSLConstList papszOptions)
174 : {
175 65 : if (!TestCapability(ODsCCreateLayer))
176 3 : return nullptr;
177 :
178 : const auto poSRS =
179 62 : poSrcGeomFieldDefn ? poSrcGeomFieldDefn->GetSpatialRef() : nullptr;
180 :
181 62 : std::unique_ptr<OGRCoordinateTransformation> poCT;
182 62 : if (poSRS == nullptr)
183 : {
184 50 : CPLError(
185 : CE_Warning, CPLE_AppDefined,
186 : "No SRS set on layer. Assuming it is long/lat on WGS84 ellipsoid");
187 : }
188 : else
189 : {
190 12 : OGRSpatialReference oSRSWGS84;
191 12 : oSRSWGS84.SetWellKnownGeogCS("WGS84");
192 12 : oSRSWGS84.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
193 12 : const char *const apszOptions[] = {
194 : "IGNORE_DATA_AXIS_TO_SRS_AXIS_MAPPING=YES", nullptr};
195 12 : if (!poSRS->IsSame(&oSRSWGS84, apszOptions))
196 : {
197 3 : poCT.reset(OGRCreateCoordinateTransformation(poSRS, &oSRSWGS84));
198 3 : if (poCT == nullptr)
199 : {
200 0 : CPLError(
201 : CE_Warning, CPLE_AppDefined,
202 : "Failed to create coordinate transformation between the "
203 : "input coordinate system and WGS84.");
204 :
205 0 : return nullptr;
206 : }
207 : }
208 : }
209 :
210 62 : const char *pszRS = CSLFetchNameValue(papszOptions, "RS");
211 62 : if (pszRS)
212 : {
213 1 : m_bIsRSSeparated = CPLTestBool(pszRS);
214 : }
215 :
216 62 : CPLStringList aosOptions(papszOptions);
217 :
218 62 : double dfXYResolution = OGRGeomCoordinatePrecision::UNKNOWN;
219 62 : double dfZResolution = OGRGeomCoordinatePrecision::UNKNOWN;
220 62 : if (const char *pszCoordPrecision =
221 62 : CSLFetchNameValue(papszOptions, "COORDINATE_PRECISION"))
222 : {
223 1 : dfXYResolution = std::pow(10.0, -CPLAtof(pszCoordPrecision));
224 1 : dfZResolution = dfXYResolution;
225 : }
226 61 : else if (poSrcGeomFieldDefn)
227 : {
228 58 : const auto &oCoordPrec = poSrcGeomFieldDefn->GetCoordinatePrecision();
229 116 : OGRSpatialReference oSRSWGS84;
230 58 : oSRSWGS84.SetWellKnownGeogCS("WGS84");
231 : const auto oCoordPrecWGS84 =
232 116 : oCoordPrec.ConvertToOtherSRS(poSRS, &oSRSWGS84);
233 :
234 58 : if (oCoordPrec.dfXYResolution != OGRGeomCoordinatePrecision::UNKNOWN)
235 : {
236 2 : dfXYResolution = oCoordPrecWGS84.dfXYResolution;
237 :
238 : aosOptions.SetNameValue(
239 : "XY_COORD_PRECISION",
240 : CPLSPrintf("%d",
241 : OGRGeomCoordinatePrecision::ResolutionToPrecision(
242 2 : dfXYResolution)));
243 : }
244 58 : if (oCoordPrec.dfZResolution != OGRGeomCoordinatePrecision::UNKNOWN)
245 : {
246 2 : dfZResolution = oCoordPrecWGS84.dfZResolution;
247 :
248 : aosOptions.SetNameValue(
249 : "Z_COORD_PRECISION",
250 : CPLSPrintf("%d",
251 : OGRGeomCoordinatePrecision::ResolutionToPrecision(
252 2 : dfZResolution)));
253 : }
254 : }
255 :
256 62 : m_apoLayers.emplace_back(std::make_unique<OGRGeoJSONSeqLayer>(
257 62 : this, pszNameIn, aosOptions.List(), std::move(poCT)));
258 :
259 62 : auto poLayer = m_apoLayers.back().get();
260 62 : if (poLayer->GetGeomType() != wkbNone &&
261 : dfXYResolution != OGRGeomCoordinatePrecision::UNKNOWN)
262 : {
263 3 : auto poGeomFieldDefn = poLayer->GetLayerDefn()->GetGeomFieldDefn(0);
264 : OGRGeomCoordinatePrecision oCoordPrec(
265 6 : poGeomFieldDefn->GetCoordinatePrecision());
266 3 : oCoordPrec.dfXYResolution = dfXYResolution;
267 3 : poGeomFieldDefn->SetCoordinatePrecision(oCoordPrec);
268 : }
269 :
270 62 : if (poLayer->GetGeomType() != wkbNone &&
271 : dfZResolution != OGRGeomCoordinatePrecision::UNKNOWN)
272 : {
273 3 : auto poGeomFieldDefn = poLayer->GetLayerDefn()->GetGeomFieldDefn(0);
274 : OGRGeomCoordinatePrecision oCoordPrec(
275 6 : poGeomFieldDefn->GetCoordinatePrecision());
276 3 : oCoordPrec.dfZResolution = dfZResolution;
277 3 : poGeomFieldDefn->SetCoordinatePrecision(oCoordPrec);
278 : }
279 :
280 62 : return poLayer;
281 : }
282 :
283 : /************************************************************************/
284 : /* TestCapability() */
285 : /************************************************************************/
286 :
287 128 : int OGRGeoJSONSeqDataSource::TestCapability(const char *pszCap)
288 : {
289 128 : if (EQUAL(pszCap, ODsCCreateLayer))
290 105 : return eAccess == GA_Update;
291 :
292 23 : return FALSE;
293 : }
294 :
295 : /************************************************************************/
296 : /* OGRGeoJSONSeqLayer() */
297 : /************************************************************************/
298 :
299 46 : OGRGeoJSONSeqLayer::OGRGeoJSONSeqLayer(OGRGeoJSONSeqDataSource *poDS,
300 46 : const char *pszName)
301 46 : : m_poDS(poDS)
302 : {
303 46 : SetDescription(pszName);
304 46 : m_poFeatureDefn = new OGRFeatureDefn(pszName);
305 46 : m_poFeatureDefn->Reference();
306 :
307 46 : OGRSpatialReference *poSRSWGS84 = new OGRSpatialReference();
308 46 : poSRSWGS84->SetWellKnownGeogCS("WGS84");
309 46 : poSRSWGS84->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
310 46 : m_poFeatureDefn->GetGeomFieldDefn(0)->SetSpatialRef(poSRSWGS84);
311 46 : poSRSWGS84->Release();
312 :
313 : const double dfTmp =
314 46 : CPLAtof(CPLGetConfigOption("OGR_GEOJSON_MAX_OBJ_SIZE", "200"));
315 46 : m_nMaxObjectSize = dfTmp > 0 ? static_cast<size_t>(dfTmp * 1024 * 1024) : 0;
316 46 : }
317 :
318 : /************************************************************************/
319 : /* OGRGeoJSONSeqLayer() */
320 : /************************************************************************/
321 :
322 : // Write-only constructor
323 62 : OGRGeoJSONSeqLayer::OGRGeoJSONSeqLayer(
324 : OGRGeoJSONSeqDataSource *poDS, const char *pszName,
325 : CSLConstList papszOptions,
326 62 : std::unique_ptr<OGRCoordinateTransformation> &&poCT)
327 62 : : m_poDS(poDS), m_bWriteOnlyLayer(true)
328 : {
329 62 : m_bLayerDefnEstablished = true;
330 :
331 62 : SetDescription(pszName);
332 62 : m_poFeatureDefn = new OGRFeatureDefn(pszName);
333 62 : m_poFeatureDefn->Reference();
334 62 : m_poFeatureDefn->GetGeomFieldDefn(0)->SetSpatialRef(
335 62 : OGRSpatialReference::GetWGS84SRS());
336 62 : m_poCT = std::move(poCT);
337 :
338 62 : m_oWriteOptions.bWriteBBOX =
339 62 : CPLTestBool(CSLFetchNameValueDef(papszOptions, "WRITE_BBOX", "FALSE"));
340 62 : m_oWriteOptions.SetRFC7946Settings();
341 62 : m_oWriteOptions.SetIDOptions(papszOptions);
342 :
343 : const char *pszCoordPrecision =
344 62 : CSLFetchNameValue(papszOptions, "COORDINATE_PRECISION");
345 62 : if (pszCoordPrecision)
346 : {
347 1 : m_oWriteOptions.nXYCoordPrecision = atoi(pszCoordPrecision);
348 1 : m_oWriteOptions.nZCoordPrecision = atoi(pszCoordPrecision);
349 : }
350 : else
351 : {
352 61 : m_oWriteOptions.nXYCoordPrecision =
353 61 : atoi(CSLFetchNameValueDef(papszOptions, "XY_COORD_PRECISION", "7"));
354 61 : m_oWriteOptions.nZCoordPrecision =
355 61 : atoi(CSLFetchNameValueDef(papszOptions, "Z_COORD_PRECISION", "3"));
356 : }
357 :
358 62 : m_oWriteOptions.nSignificantFigures =
359 62 : atoi(CSLFetchNameValueDef(papszOptions, "SIGNIFICANT_FIGURES", "-1"));
360 62 : m_oWriteOptions.bAllowNonFiniteValues = CPLTestBool(
361 : CSLFetchNameValueDef(papszOptions, "WRITE_NON_FINITE_VALUES", "FALSE"));
362 62 : m_oWriteOptions.bAutodetectJsonStrings = CPLTestBool(
363 : CSLFetchNameValueDef(papszOptions, "AUTODETECT_JSON_STRINGS", "TRUE"));
364 62 : }
365 :
366 : /************************************************************************/
367 : /* ~OGRGeoJSONSeqLayer() */
368 : /************************************************************************/
369 :
370 216 : OGRGeoJSONSeqLayer::~OGRGeoJSONSeqLayer()
371 : {
372 108 : m_poFeatureDefn->Release();
373 216 : }
374 :
375 : /************************************************************************/
376 : /* GetLayerDefn() */
377 : /************************************************************************/
378 :
379 1306 : OGRFeatureDefn *OGRGeoJSONSeqLayer::GetLayerDefn()
380 : {
381 1306 : if (!m_bLayerDefnEstablished)
382 : {
383 3 : Init(/* bLooseIdentification = */ false,
384 : /* bEstablishLayerDefn = */ true);
385 : }
386 1306 : return m_poFeatureDefn;
387 : }
388 :
389 : /************************************************************************/
390 : /* Init() */
391 : /************************************************************************/
392 :
393 49 : bool OGRGeoJSONSeqLayer::Init(bool bLooseIdentification,
394 : bool bEstablishLayerDefn)
395 : {
396 55 : if (STARTS_WITH(m_poDS->GetDescription(), "/vsimem/") ||
397 6 : !STARTS_WITH(m_poDS->GetDescription(), "/vsi"))
398 : {
399 49 : VSIFSeekL(m_poDS->m_fp, 0, SEEK_END);
400 49 : m_nFileSize = VSIFTellL(m_poDS->m_fp);
401 : }
402 :
403 : // Set m_bLayerDefnEstablished = true early to avoid infinite recursive
404 : // calls.
405 49 : if (bEstablishLayerDefn)
406 43 : m_bLayerDefnEstablished = true;
407 :
408 49 : ResetReading();
409 :
410 98 : std::map<std::string, int> oMapFieldNameToIdx;
411 98 : std::vector<std::unique_ptr<OGRFieldDefn>> apoFieldDefn;
412 49 : gdal::DirectedAcyclicGraph<int, std::string> dag;
413 49 : bool bOK = false;
414 :
415 : while (true)
416 : {
417 205 : auto poObject = GetNextObject(bLooseIdentification);
418 205 : if (!poObject)
419 43 : break;
420 162 : const auto eObjectType = OGRGeoJSONGetType(poObject);
421 162 : if (bEstablishLayerDefn && eObjectType == GeoJSONObject::eFeature)
422 : {
423 151 : m_oReader.GenerateFeatureDefn(oMapFieldNameToIdx, apoFieldDefn, dag,
424 : this, poObject);
425 : }
426 162 : json_object_put(poObject);
427 162 : if (!bEstablishLayerDefn)
428 : {
429 6 : bOK = (eObjectType == GeoJSONObject::eFeature);
430 6 : break;
431 : }
432 156 : m_nTotalFeatures++;
433 156 : }
434 :
435 49 : if (bEstablishLayerDefn)
436 : {
437 : // CPLDebug("GEOJSONSEQ", "Establish layer definition");
438 :
439 86 : const auto sortedFields = dag.getTopologicalOrdering();
440 43 : CPLAssert(sortedFields.size() == apoFieldDefn.size());
441 158 : for (int idx : sortedFields)
442 : {
443 115 : m_poFeatureDefn->AddFieldDefn(apoFieldDefn[idx].get());
444 : }
445 43 : m_poFeatureDefn->Seal(true);
446 43 : m_oReader.FinalizeLayerDefn(this, m_osFIDColumn);
447 : }
448 :
449 49 : ResetReading();
450 :
451 49 : m_nFileSize = 0;
452 49 : m_nIter = 0;
453 :
454 98 : return bOK || m_nTotalFeatures > 0;
455 : }
456 :
457 : /************************************************************************/
458 : /* ResetReading() */
459 : /************************************************************************/
460 :
461 216 : void OGRGeoJSONSeqLayer::ResetReading()
462 : {
463 432 : if (!m_poDS->m_bSupportsRead ||
464 216 : (m_bWriteOnlyLayer && m_poDS->m_apoLayers.size() > 1))
465 : {
466 0 : return;
467 : }
468 :
469 216 : m_poDS->m_bAtEOF = false;
470 216 : VSIFSeekL(m_poDS->m_fp, 0, SEEK_SET);
471 : // Undocumented: for testing purposes only
472 216 : const size_t nBufferSize = static_cast<size_t>(std::max(
473 216 : 1, atoi(CPLGetConfigOption("OGR_GEOJSONSEQ_CHUNK_SIZE", "40960"))));
474 216 : const size_t nBufferSizeValidated =
475 : nBufferSize > static_cast<size_t>(100 * 1000 * 1000)
476 : ? static_cast<size_t>(100 * 1000 * 1000)
477 : : nBufferSize;
478 216 : m_osBuffer.resize(nBufferSizeValidated);
479 216 : m_osFeatureBuffer.clear();
480 216 : m_nPosInBuffer = nBufferSizeValidated;
481 216 : m_nBufferValidSize = nBufferSizeValidated;
482 216 : m_nNextFID = 0;
483 : }
484 :
485 : /************************************************************************/
486 : /* GetNextObject() */
487 : /************************************************************************/
488 :
489 388 : json_object *OGRGeoJSONSeqLayer::GetNextObject(bool bLooseIdentification)
490 : {
491 388 : m_osFeatureBuffer.clear();
492 : while (true)
493 : {
494 : // If we read all the buffer, then reload it from file
495 444 : if (m_nPosInBuffer >= m_nBufferValidSize)
496 : {
497 259 : if (m_nBufferValidSize < m_osBuffer.size())
498 : {
499 86 : return nullptr;
500 : }
501 173 : m_nBufferValidSize =
502 173 : VSIFReadL(&m_osBuffer[0], 1, m_osBuffer.size(), m_poDS->m_fp);
503 173 : m_nPosInBuffer = 0;
504 298 : if (VSIFTellL(m_poDS->m_fp) == m_nBufferValidSize &&
505 125 : m_nBufferValidSize > 0)
506 : {
507 125 : m_poDS->m_bIsRSSeparated = (m_osBuffer[0] == RS);
508 125 : if (m_poDS->m_bIsRSSeparated)
509 : {
510 26 : m_nPosInBuffer++;
511 : }
512 : }
513 173 : m_nIter++;
514 :
515 219 : if (m_nFileSize > 0 && (m_nBufferValidSize < m_osBuffer.size() ||
516 46 : (m_nIter % 100) == 0))
517 : {
518 48 : CPLDebug("GeoJSONSeq", "First pass: %.2f %%",
519 48 : 100.0 * VSIFTellL(m_poDS->m_fp) / m_nFileSize);
520 : }
521 173 : if (m_nPosInBuffer >= m_nBufferValidSize)
522 : {
523 0 : return nullptr;
524 : }
525 : }
526 :
527 : // Find next feature separator in buffer
528 358 : const size_t nNextSepPos = m_osBuffer.find(
529 358 : m_poDS->m_bIsRSSeparated ? RS : '\n', m_nPosInBuffer);
530 358 : if (nNextSepPos != std::string::npos)
531 : {
532 564 : m_osFeatureBuffer.append(m_osBuffer.data() + m_nPosInBuffer,
533 282 : nNextSepPos - m_nPosInBuffer);
534 282 : m_nPosInBuffer = nNextSepPos + 1;
535 : }
536 : else
537 : {
538 : // No separator ? then accummulate
539 152 : m_osFeatureBuffer.append(m_osBuffer.data() + m_nPosInBuffer,
540 76 : m_nBufferValidSize - m_nPosInBuffer);
541 145 : if (m_nMaxObjectSize > 0 &&
542 69 : m_osFeatureBuffer.size() > m_nMaxObjectSize)
543 : {
544 1 : CPLError(CE_Failure, CPLE_NotSupported,
545 : "Too large feature. You may define the "
546 : "OGR_GEOJSON_MAX_OBJ_SIZE configuration option to "
547 : "a value in megabytes (larger than %u) to allow "
548 : "for larger features, or 0 to remove any size limit.",
549 1 : static_cast<unsigned>(m_osFeatureBuffer.size() / 1024 /
550 : 1024));
551 1 : return nullptr;
552 : }
553 75 : m_nPosInBuffer = m_nBufferValidSize;
554 75 : if (m_nBufferValidSize == m_osBuffer.size())
555 : {
556 48 : continue;
557 : }
558 : }
559 :
560 756 : while (!m_osFeatureBuffer.empty() &&
561 377 : (m_osFeatureBuffer.back() == '\r' ||
562 377 : m_osFeatureBuffer.back() == '\n'))
563 : {
564 70 : m_osFeatureBuffer.pop_back();
565 : }
566 309 : if (!m_osFeatureBuffer.empty())
567 : {
568 307 : json_object *poObject = nullptr;
569 307 : CPL_IGNORE_RET_VAL(
570 307 : OGRJSonParse(m_osFeatureBuffer.c_str(), &poObject));
571 307 : m_osFeatureBuffer.clear();
572 307 : if (json_object_get_type(poObject) == json_type_object)
573 : {
574 301 : return poObject;
575 : }
576 6 : json_object_put(poObject);
577 6 : if (bLooseIdentification)
578 : {
579 0 : return nullptr;
580 : }
581 : }
582 56 : }
583 : }
584 :
585 : /************************************************************************/
586 : /* GetNextFeature() */
587 : /************************************************************************/
588 :
589 163 : OGRFeature *OGRGeoJSONSeqLayer::GetNextFeature()
590 : {
591 163 : if (!m_poDS->m_bSupportsRead)
592 : {
593 0 : return nullptr;
594 : }
595 163 : if (m_bWriteOnlyLayer && m_poDS->m_apoLayers.size() > 1)
596 : {
597 0 : CPLError(CE_Failure, CPLE_NotSupported,
598 : "GetNextFeature() not supported when appending a new layer");
599 0 : return nullptr;
600 : }
601 :
602 163 : GetLayerDefn(); // force scan if not already done
603 : while (true)
604 : {
605 183 : auto poObject = GetNextObject(false);
606 183 : if (!poObject)
607 44 : return nullptr;
608 : OGRFeature *poFeature;
609 139 : auto type = OGRGeoJSONGetType(poObject);
610 139 : if (type == GeoJSONObject::eFeature)
611 : {
612 135 : poFeature = m_oReader.ReadFeature(this, poObject,
613 : m_osFeatureBuffer.c_str());
614 135 : json_object_put(poObject);
615 : }
616 4 : else if (type == GeoJSONObject::eFeatureCollection ||
617 : type == GeoJSONObject::eUnknown)
618 : {
619 0 : json_object_put(poObject);
620 0 : continue;
621 : }
622 : else
623 : {
624 : OGRGeometry *poGeom =
625 4 : m_oReader.ReadGeometry(poObject, GetSpatialRef());
626 4 : json_object_put(poObject);
627 4 : if (!poGeom)
628 : {
629 0 : continue;
630 : }
631 4 : poFeature = new OGRFeature(m_poFeatureDefn);
632 4 : poFeature->SetGeometryDirectly(poGeom);
633 : }
634 :
635 139 : if (poFeature->GetFID() == OGRNullFID)
636 : {
637 139 : poFeature->SetFID(m_nNextFID);
638 139 : m_nNextFID++;
639 : }
640 300 : if ((m_poFilterGeom == nullptr ||
641 270 : FilterGeometry(poFeature->GetGeomFieldRef(m_iGeomFieldFilter))) &&
642 131 : (m_poAttrQuery == nullptr || m_poAttrQuery->Evaluate(poFeature)))
643 : {
644 119 : return poFeature;
645 : }
646 20 : delete poFeature;
647 20 : }
648 : }
649 :
650 : /************************************************************************/
651 : /* GetFeatureCount() */
652 : /************************************************************************/
653 :
654 32 : GIntBig OGRGeoJSONSeqLayer::GetFeatureCount(int bForce)
655 : {
656 32 : if (m_poFilterGeom == nullptr && m_poAttrQuery == nullptr)
657 : {
658 26 : GetLayerDefn(); // force scan if not already done
659 26 : return m_nTotalFeatures;
660 : }
661 6 : return OGRLayer::GetFeatureCount(bForce);
662 : }
663 :
664 : /************************************************************************/
665 : /* TestCapability() */
666 : /************************************************************************/
667 :
668 185 : int OGRGeoJSONSeqLayer::TestCapability(const char *pszCap)
669 : {
670 185 : if (EQUAL(pszCap, OLCStringsAsUTF8))
671 13 : return true;
672 172 : if (m_poFilterGeom == nullptr && m_poAttrQuery == nullptr &&
673 172 : EQUAL(pszCap, OLCFastFeatureCount))
674 : {
675 0 : return true;
676 : }
677 172 : if (EQUAL(pszCap, OLCCreateField) || EQUAL(pszCap, OLCSequentialWrite))
678 : {
679 47 : return m_poDS->GetAccess() == GA_Update;
680 : }
681 :
682 125 : return false;
683 : }
684 :
685 : /************************************************************************/
686 : /* ICreateFeature() */
687 : /************************************************************************/
688 :
689 106 : OGRErr OGRGeoJSONSeqLayer::ICreateFeature(OGRFeature *poFeature)
690 : {
691 106 : if (m_poDS->GetAccess() != GA_Update)
692 3 : return OGRERR_FAILURE;
693 :
694 103 : if (!m_poDS->m_bAtEOF)
695 : {
696 50 : m_poDS->m_bAtEOF = true;
697 50 : VSIFSeekL(m_poDS->m_fp, 0, SEEK_END);
698 : }
699 :
700 103 : std::unique_ptr<OGRFeature> poFeatureToWrite;
701 103 : if (m_poCT != nullptr)
702 : {
703 3 : poFeatureToWrite.reset(new OGRFeature(m_poFeatureDefn));
704 3 : poFeatureToWrite->SetFrom(poFeature);
705 3 : poFeatureToWrite->SetFID(poFeature->GetFID());
706 3 : OGRGeometry *poGeometry = poFeatureToWrite->GetGeometryRef();
707 3 : if (poGeometry)
708 : {
709 3 : const char *const apszOptions[] = {"WRAPDATELINE=YES", nullptr};
710 3 : OGRGeometry *poNewGeom = OGRGeometryFactory::transformWithOptions(
711 : poGeometry, m_poCT.get(), const_cast<char **>(apszOptions),
712 3 : m_oTransformCache);
713 3 : if (poNewGeom == nullptr)
714 : {
715 0 : return OGRERR_FAILURE;
716 : }
717 :
718 3 : OGREnvelope sEnvelope;
719 3 : poNewGeom->getEnvelope(&sEnvelope);
720 3 : if (sEnvelope.MinX < -180.0 || sEnvelope.MaxX > 180.0 ||
721 3 : sEnvelope.MinY < -90.0 || sEnvelope.MaxY > 90.0)
722 : {
723 0 : CPLError(CE_Failure, CPLE_AppDefined,
724 : "Geometry extent outside of "
725 : "[-180.0,180.0]x[-90.0,90.0] bounds");
726 0 : return OGRERR_FAILURE;
727 : }
728 :
729 3 : poFeatureToWrite->SetGeometryDirectly(poNewGeom);
730 : }
731 : }
732 :
733 103 : ++m_nTotalFeatures;
734 :
735 106 : json_object *poObj = OGRGeoJSONWriteFeature(
736 3 : poFeatureToWrite.get() ? poFeatureToWrite.get() : poFeature,
737 103 : m_oWriteOptions);
738 103 : CPLAssert(nullptr != poObj);
739 :
740 103 : const char *pszJson = json_object_to_json_string(poObj);
741 :
742 103 : char chEOL = '\n';
743 103 : OGRErr eErr = OGRERR_NONE;
744 216 : if ((m_poDS->m_bIsRSSeparated &&
745 10 : VSIFWriteL(&RS, 1, 1, m_poDS->m_fp) != 1) ||
746 216 : VSIFWriteL(pszJson, strlen(pszJson), 1, m_poDS->m_fp) != 1 ||
747 103 : VSIFWriteL(&chEOL, 1, 1, m_poDS->m_fp) != 1)
748 : {
749 0 : CPLError(CE_Failure, CPLE_FileIO, "Cannot write feature");
750 0 : eErr = OGRERR_FAILURE;
751 : }
752 :
753 103 : json_object_put(poObj);
754 :
755 103 : return eErr;
756 : }
757 :
758 : /************************************************************************/
759 : /* CreateField() */
760 : /************************************************************************/
761 :
762 111 : OGRErr OGRGeoJSONSeqLayer::CreateField(const OGRFieldDefn *poField,
763 : int /* bApproxOK */)
764 : {
765 111 : if (m_poDS->GetAccess() != GA_Update)
766 3 : return OGRERR_FAILURE;
767 108 : m_poFeatureDefn->AddFieldDefn(poField);
768 108 : return OGRERR_NONE;
769 : }
770 :
771 : /************************************************************************/
772 : /* Open() */
773 : /************************************************************************/
774 :
775 55 : bool OGRGeoJSONSeqDataSource::Open(GDALOpenInfo *poOpenInfo,
776 : GeoJSONSourceType nSrcType)
777 : {
778 55 : CPLAssert(nullptr == m_fp);
779 :
780 110 : CPLString osLayerName("GeoJSONSeq");
781 :
782 55 : const char *pszUnprefixedFilename = poOpenInfo->pszFilename;
783 55 : if (STARTS_WITH_CI(poOpenInfo->pszFilename, "GeoJSONSeq:"))
784 : {
785 1 : pszUnprefixedFilename = poOpenInfo->pszFilename + strlen("GeoJSONSeq:");
786 : }
787 :
788 55 : if (nSrcType == eGeoJSONSourceFile)
789 : {
790 43 : if (pszUnprefixedFilename != poOpenInfo->pszFilename)
791 : {
792 1 : osLayerName = CPLGetBasenameSafe(pszUnprefixedFilename);
793 1 : m_fp = VSIFOpenL(pszUnprefixedFilename,
794 1 : poOpenInfo->eAccess == GA_Update ? "rb+" : "rb");
795 : }
796 : else
797 : {
798 42 : osLayerName = CPLGetBasenameSafe(poOpenInfo->pszFilename);
799 42 : std::swap(m_fp, poOpenInfo->fpL);
800 : }
801 : }
802 12 : else if (nSrcType == eGeoJSONSourceText)
803 : {
804 3 : if (poOpenInfo->eAccess == GA_Update)
805 0 : return false;
806 :
807 3 : m_osTmpFile = VSIMemGenerateHiddenFilename("geojsonseq");
808 3 : m_fp = VSIFileFromMemBuffer(
809 : m_osTmpFile.c_str(),
810 3 : reinterpret_cast<GByte *>(CPLStrdup(poOpenInfo->pszFilename)),
811 3 : strlen(poOpenInfo->pszFilename), true);
812 : }
813 9 : else if (nSrcType == eGeoJSONSourceService)
814 : {
815 9 : if (poOpenInfo->eAccess == GA_Update)
816 0 : return false;
817 :
818 : char *pszStoredContent =
819 9 : OGRGeoJSONDriverStealStoredContent(pszUnprefixedFilename);
820 9 : if (pszStoredContent)
821 : {
822 9 : if (EQUAL(pszStoredContent, INVALID_CONTENT_FOR_JSON_LIKE) ||
823 0 : !GeoJSONSeqIsObject(pszStoredContent, poOpenInfo))
824 : {
825 9 : OGRGeoJSONDriverStoreContent(poOpenInfo->pszFilename,
826 : pszStoredContent);
827 9 : return false;
828 : }
829 : else
830 : {
831 0 : m_osTmpFile = VSIMemGenerateHiddenFilename("geojsonseq");
832 0 : m_fp = VSIFileFromMemBuffer(
833 : m_osTmpFile.c_str(),
834 : reinterpret_cast<GByte *>(pszStoredContent),
835 0 : strlen(pszStoredContent), true);
836 : }
837 : }
838 : else
839 : {
840 : CPLHTTPResult *pResult =
841 0 : GeoJSONHTTPFetchWithContentTypeHeader(pszUnprefixedFilename);
842 0 : if (!pResult)
843 : {
844 0 : return FALSE;
845 : }
846 :
847 0 : m_osTmpFile = VSIMemGenerateHiddenFilename("geojsonseq");
848 0 : m_fp = VSIFileFromMemBuffer(m_osTmpFile.c_str(), pResult->pabyData,
849 0 : pResult->nDataLen, true);
850 0 : pResult->pabyData = nullptr;
851 0 : pResult->nDataLen = 0;
852 0 : CPLHTTPDestroyResult(pResult);
853 : }
854 : }
855 46 : if (m_fp == nullptr)
856 : {
857 0 : return false;
858 : }
859 46 : SetDescription(poOpenInfo->pszFilename);
860 46 : auto poLayer = new OGRGeoJSONSeqLayer(this, osLayerName.c_str());
861 46 : const bool bLooseIdentification =
862 46 : nSrcType == eGeoJSONSourceService &&
863 0 : !STARTS_WITH_CI(poOpenInfo->pszFilename, "GeoJSONSeq:");
864 46 : if (bLooseIdentification)
865 : {
866 0 : CPLPushErrorHandler(CPLQuietErrorHandler);
867 : }
868 46 : const bool bEstablishLayerDefn = poOpenInfo->eAccess != GA_Update;
869 46 : auto ret = poLayer->Init(bLooseIdentification, bEstablishLayerDefn);
870 46 : if (bLooseIdentification)
871 : {
872 0 : CPLPopErrorHandler();
873 0 : CPLErrorReset();
874 : }
875 46 : if (!ret)
876 : {
877 1 : delete poLayer;
878 1 : return false;
879 : }
880 45 : m_apoLayers.emplace_back(std::move(poLayer));
881 45 : eAccess = poOpenInfo->eAccess;
882 45 : return true;
883 : }
884 :
885 : /************************************************************************/
886 : /* Create() */
887 : /************************************************************************/
888 :
889 44 : bool OGRGeoJSONSeqDataSource::Create(const char *pszName,
890 : char ** /* papszOptions */)
891 : {
892 44 : CPLAssert(nullptr == m_fp);
893 :
894 44 : if (strcmp(pszName, "/dev/stdout") == 0)
895 0 : pszName = "/vsistdout/";
896 :
897 : /* -------------------------------------------------------------------- */
898 : /* Create the output file. */
899 : /* -------------------------------------------------------------------- */
900 44 : m_bSupportsRead =
901 87 : VSIFileManager::GetHandler(pszName)->SupportsRead(pszName) &&
902 86 : VSIFileManager::GetHandler(pszName)->SupportsRandomWrite(pszName,
903 43 : false);
904 44 : m_bAtEOF = !m_bSupportsRead;
905 44 : m_fp = VSIFOpenExL(pszName, m_bSupportsRead ? "wb+" : "wb", true);
906 44 : if (nullptr == m_fp)
907 : {
908 1 : CPLError(CE_Failure, CPLE_OpenFailed, "Failed to create %s: %s",
909 : pszName, VSIGetLastErrorMsg());
910 1 : return false;
911 : }
912 :
913 43 : eAccess = GA_Update;
914 :
915 43 : m_bIsRSSeparated = EQUAL(CPLGetExtensionSafe(pszName).c_str(), "GEOJSONS");
916 :
917 43 : return true;
918 : }
919 :
920 : /************************************************************************/
921 : /* OGRGeoJSONSeqDriverIdentify() */
922 : /************************************************************************/
923 :
924 48030 : static int OGRGeoJSONSeqDriverIdentifyInternal(GDALOpenInfo *poOpenInfo,
925 : GeoJSONSourceType &nSrcType)
926 : {
927 48030 : nSrcType = GeoJSONSeqGetSourceType(poOpenInfo);
928 48030 : if (nSrcType == eGeoJSONSourceUnknown)
929 47911 : return FALSE;
930 119 : if (nSrcType == eGeoJSONSourceService)
931 : {
932 27 : if (poOpenInfo->IsSingleAllowedDriver("GeoJSONSeq"))
933 1 : return TRUE;
934 26 : if (!STARTS_WITH_CI(poOpenInfo->pszFilename, "GeoJSONSeq:"))
935 : {
936 26 : return -1;
937 : }
938 : }
939 92 : return TRUE;
940 : }
941 :
942 : /************************************************************************/
943 : /* OGRGeoJSONSeqDriverIdentify() */
944 : /************************************************************************/
945 :
946 47975 : static int OGRGeoJSONSeqDriverIdentify(GDALOpenInfo *poOpenInfo)
947 : {
948 : GeoJSONSourceType nSrcType;
949 95950 : return OGRGeoJSONSeqDriverIdentifyInternal(poOpenInfo, nSrcType);
950 : }
951 :
952 : /************************************************************************/
953 : /* Open() */
954 : /************************************************************************/
955 :
956 55 : static GDALDataset *OGRGeoJSONSeqDriverOpen(GDALOpenInfo *poOpenInfo)
957 : {
958 : GeoJSONSourceType nSrcType;
959 55 : if (OGRGeoJSONSeqDriverIdentifyInternal(poOpenInfo, nSrcType) == FALSE)
960 : {
961 0 : return nullptr;
962 : }
963 :
964 55 : OGRGeoJSONSeqDataSource *poDS = new OGRGeoJSONSeqDataSource();
965 :
966 55 : if (!poDS->Open(poOpenInfo, nSrcType))
967 : {
968 10 : delete poDS;
969 10 : poDS = nullptr;
970 : }
971 :
972 55 : return poDS;
973 : }
974 :
975 : /************************************************************************/
976 : /* Create() */
977 : /************************************************************************/
978 :
979 : static GDALDataset *
980 44 : OGRGeoJSONSeqDriverCreate(const char *pszName, int /* nBands */,
981 : int /* nXSize */, int /* nYSize */,
982 : GDALDataType /* eDT */, char **papszOptions)
983 : {
984 44 : OGRGeoJSONSeqDataSource *poDS = new OGRGeoJSONSeqDataSource();
985 :
986 44 : if (!poDS->Create(pszName, papszOptions))
987 : {
988 1 : delete poDS;
989 1 : poDS = nullptr;
990 : }
991 :
992 44 : return poDS;
993 : }
994 :
995 : /************************************************************************/
996 : /* RegisterOGRGeoJSONSeq() */
997 : /************************************************************************/
998 :
999 1682 : void RegisterOGRGeoJSONSeq()
1000 : {
1001 1682 : if (GDALGetDriverByName("GeoJSONSeq") != nullptr)
1002 301 : return;
1003 :
1004 1381 : GDALDriver *poDriver = new GDALDriver();
1005 :
1006 1381 : poDriver->SetDescription("GeoJSONSeq");
1007 1381 : poDriver->SetMetadataItem(GDAL_DCAP_VECTOR, "YES");
1008 1381 : poDriver->SetMetadataItem(GDAL_DCAP_CREATE_LAYER, "YES");
1009 1381 : poDriver->SetMetadataItem(GDAL_DCAP_CREATE_FIELD, "YES");
1010 1381 : poDriver->SetMetadataItem(GDAL_DCAP_Z_GEOMETRIES, "YES");
1011 1381 : poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "GeoJSON Sequence");
1012 1381 : poDriver->SetMetadataItem(GDAL_DMD_EXTENSIONS, "geojsonl geojsons");
1013 1381 : poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC,
1014 1381 : "drivers/vector/geojsonseq.html");
1015 :
1016 1381 : poDriver->SetMetadataItem(
1017 : GDAL_DS_LAYER_CREATIONOPTIONLIST,
1018 : "<LayerCreationOptionList>"
1019 : " <Option name='RS' type='boolean' description='whether to prefix "
1020 : "records with RS=0x1e character' default='NO'/>"
1021 : " <Option name='COORDINATE_PRECISION' type='int' description='Number "
1022 : "of decimal for coordinates. Default is 7'/>"
1023 : " <Option name='SIGNIFICANT_FIGURES' type='int' description='Number "
1024 : "of significant figures for floating-point values' default='17'/>"
1025 : " <Option name='ID_FIELD' type='string' description='Name of the "
1026 : "source field that must be used as the id member of Feature features'/>"
1027 : " <Option name='ID_TYPE' type='string-select' description='Type of "
1028 : "the id member of Feature features'>"
1029 : " <Value>AUTO</Value>"
1030 : " <Value>String</Value>"
1031 : " <Value>Integer</Value>"
1032 : " </Option>"
1033 : " <Option name='WRITE_BBOX' type='boolean' description='whether to "
1034 : "write a bbox property with the bounding box of each geometry' "
1035 : "default='NO'/>"
1036 1381 : "</LayerCreationOptionList>");
1037 :
1038 1381 : poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
1039 1381 : poDriver->SetMetadataItem(GDAL_DMD_CREATIONFIELDDATATYPES,
1040 : "Integer Integer64 Real String IntegerList "
1041 1381 : "Integer64List RealList StringList");
1042 1381 : poDriver->SetMetadataItem(GDAL_DMD_CREATIONFIELDDATASUBTYPES, "Boolean");
1043 1381 : poDriver->SetMetadataItem(GDAL_DMD_SUPPORTED_SQL_DIALECTS, "OGRSQL SQLITE");
1044 1381 : poDriver->SetMetadataItem(GDAL_DCAP_HONOR_GEOM_COORDINATE_PRECISION, "YES");
1045 :
1046 1381 : poDriver->pfnOpen = OGRGeoJSONSeqDriverOpen;
1047 1381 : poDriver->pfnIdentify = OGRGeoJSONSeqDriverIdentify;
1048 1381 : poDriver->pfnCreate = OGRGeoJSONSeqDriverCreate;
1049 :
1050 1381 : GetGDALDriverManager()->RegisterDriver(poDriver);
1051 : }
|