Skip to content

Commit

Permalink
Merge pull request OSGeo#11420 from rouault/flatgeobuf_empty_geoms
Browse files Browse the repository at this point in the history
FlatGeobuf writing: in SPATIAL_INDEX=NO mode, deal with empty geometries as if there were null
  • Loading branch information
rouault authored Dec 5, 2024
2 parents d9ac524 + c59ee22 commit 436ae05
Show file tree
Hide file tree
Showing 2 changed files with 59 additions and 9 deletions.
38 changes: 38 additions & 0 deletions autotest/ogr/ogr_flatgeobuf.py
Original file line number Diff line number Diff line change
Expand Up @@ -1612,3 +1612,41 @@ def test_ogr_flatgeobuf_arrow_stream_numpy_datetime_as_string(tmp_vsimem):
assert batch["datetime"][1] == b"2022-05-31T12:34:56.789Z"
assert batch["datetime"][2] == b"2022-05-31T12:34:56"
assert batch["datetime"][3] == b"2022-05-31T12:34:56+12:30"


def test_ogr_flatgeobuf_write_empty_geometries_no_spatial_index(tmp_vsimem):
# Verify empty geom handling without spatial index
out_filename = str(tmp_vsimem / "out.fgb")
with ogr.GetDriverByName("FlatGeobuf").CreateDataSource(out_filename) as ds:
lyr = ds.CreateLayer(
"test", geom_type=ogr.wkbPolygon, options=["SPATIAL_INDEX=NO"]
)

f = ogr.Feature(lyr.GetLayerDefn())
f.SetGeometry(ogr.CreateGeometryFromWkt("POLYGON EMPTY"))
lyr.CreateFeature(f)
f = ogr.Feature(lyr.GetLayerDefn())
lyr.CreateFeature(f)

with gdal.OpenEx(out_filename) as ds:
lyr = ds.GetLayer(0)
f = lyr.GetNextFeature()
g = f.GetGeometryRef()
assert g is None


def test_ogr_flatgeobuf_write_empty_file_no_spatial_index(tmp_vsimem):
# Verify empty geom handling without spatial index
out_filename = str(tmp_vsimem / "out.fgb")
with ogr.GetDriverByName("FlatGeobuf").CreateDataSource(out_filename) as ds:
srs = osr.SpatialReference()
srs.ImportFromEPSG(32631)
ds.CreateLayer(
"test", geom_type=ogr.wkbPolygon, srs=srs, options=["SPATIAL_INDEX=NO"]
)

with gdal.OpenEx(out_filename) as ds:
lyr = ds.GetLayer(0)
assert lyr.GetFeatureCount() == 0
assert lyr.GetExtent(can_return_null=True) is None
assert lyr.GetSpatialRef().GetAuthorityCode(None) == "32631"
30 changes: 21 additions & 9 deletions ogr/ogrsf_frmts/flatgeobuf/ogrflatgeobuflayer.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
#include "geometrywriter.h"

#include <algorithm>
#include <cmath>
#include <limits>
#include <new>
#include <stdexcept>
Expand Down Expand Up @@ -71,7 +72,9 @@ OGRFlatGeobufLayer::OGRFlatGeobufLayer(const Header *poHeader, GByte *headerBuf,
m_hasM = m_poHeader->has_m();
m_hasT = m_poHeader->has_t();
const auto envelope = m_poHeader->envelope();
if (envelope && envelope->size() == 4)
if (envelope && envelope->size() == 4 && std::isfinite((*envelope)[0]) &&
std::isfinite((*envelope)[1]) && std::isfinite((*envelope)[2]) &&
std::isfinite((*envelope)[3]))
{
m_sExtent.MinX = (*envelope)[0];
m_sExtent.MinY = (*envelope)[1];
Expand Down Expand Up @@ -536,8 +539,7 @@ bool OGRFlatGeobufLayer::CreateFinalFile()
// no spatial index requested, we are (almost) done
if (!m_bCreateSpatialIndexAtClose)
{
if (m_poFpWrite == nullptr || m_featuresCount == 0 ||
!SupportsSeekWhileWriting(m_osFilename))
if (m_poFpWrite == nullptr || !SupportsSeekWhileWriting(m_osFilename))
{
return true;
}
Expand All @@ -546,14 +548,24 @@ bool OGRFlatGeobufLayer::CreateFinalFile()
VSIFSeekL(m_poFpWrite, 0, SEEK_SET);
m_writeOffset = 0;
std::vector<double> extentVector;
extentVector.push_back(m_sExtent.MinX);
extentVector.push_back(m_sExtent.MinY);
extentVector.push_back(m_sExtent.MaxX);
extentVector.push_back(m_sExtent.MaxY);
if (!m_sExtent.IsInit())
{
extentVector.resize(4, std::numeric_limits<double>::quiet_NaN());
}
else
{
extentVector.push_back(m_sExtent.MinX);
extentVector.push_back(m_sExtent.MinY);
extentVector.push_back(m_sExtent.MaxX);
extentVector.push_back(m_sExtent.MaxY);
}
writeHeader(m_poFpWrite, m_featuresCount, &extentVector);
// Sanity check to verify that the dummy header and the real header
// have the same size.
CPLAssert(m_writeOffset == m_offsetAfterHeader);
if (m_featuresCount)
{
CPLAssert(m_writeOffset == m_offsetAfterHeader);
}
CPL_IGNORE_RET_VAL(m_writeOffset); // otherwise checkers might tell the
// member is not used
return true;
Expand Down Expand Up @@ -2295,7 +2307,7 @@ OGRErr OGRFlatGeobufLayer::ICreateFeature(OGRFeature *poNewFeature)
// the size of the FlatBuffer, but WKB might be a good approximation.
// Takes an extra security margin of 10%
flatbuffers::Offset<FlatGeobuf::Geometry> geometryOffset = 0;
if (ogrGeometry != nullptr)
if (ogrGeometry && !ogrGeometry->IsEmpty())
{
const auto nWKBSize = ogrGeometry->WkbSize();
if (nWKBSize > feature_max_buffer_size - nWKBSize / 10)
Expand Down

0 comments on commit 436ae05

Please sign in to comment.