LCOV - code coverage report
Current view: top level - frmts/netcdf - netcdflayersg.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 168 187 89.8 %
Date: 2025-01-18 12:42:00 Functions: 9 9 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  netCDF read/write Driver
       4             :  * Purpose:  GDAL bindings over netCDF library.
       5             :  * Author:   Winor Chen <wchen329 at wisc.edu>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2019, Winor Chen <wchen329 at wisc.edu>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : #include "netcdfsg.h"
      13             : #include "netcdfdataset.h"
      14             : #include "ogr_core.h"
      15             : 
      16             : namespace nccfdriver
      17             : {
      18          67 : OGRwkbGeometryType RawToOGR(geom_t type, int axis_count)
      19             : {
      20          67 :     OGRwkbGeometryType ret = wkbNone;
      21             : 
      22          67 :     switch (type)
      23             :     {
      24           0 :         case NONE:
      25           0 :             break;
      26           4 :         case LINE:
      27           6 :             ret = axis_count == 2   ? wkbLineString
      28           2 :                   : axis_count == 3 ? wkbLineString25D
      29             :                                     : wkbNone;
      30           4 :             break;
      31           6 :         case MULTILINE:
      32           8 :             ret = axis_count == 2   ? wkbMultiLineString
      33           2 :                   : axis_count == 3 ? wkbMultiLineString25D
      34             :                                     : wkbNone;
      35           6 :             break;
      36           8 :         case POLYGON:
      37          11 :             ret = axis_count == 2   ? wkbPolygon
      38           3 :                   : axis_count == 3 ? wkbPolygon25D
      39             :                                     : wkbNone;
      40           8 :             break;
      41          30 :         case MULTIPOLYGON:
      42          36 :             ret = axis_count == 2   ? wkbMultiPolygon
      43           6 :                   : axis_count == 3 ? wkbMultiPolygon25D
      44             :                                     : wkbNone;
      45          30 :             break;
      46          12 :         case POINT:
      47          21 :             ret = axis_count == 2   ? wkbPoint
      48           9 :                   : axis_count == 3 ? wkbPoint25D
      49             :                                     : wkbNone;
      50          12 :             break;
      51           6 :         case MULTIPOINT:
      52           8 :             ret = axis_count == 2   ? wkbMultiPoint
      53           2 :                   : axis_count == 3 ? wkbMultiPoint25D
      54             :                                     : wkbNone;
      55           6 :             break;
      56           1 :         case UNSUPPORTED:
      57           1 :             break;
      58             :     }
      59             : 
      60          67 :     return ret;
      61             : }
      62             : 
      63         620 : geom_t OGRtoRaw(OGRwkbGeometryType type)
      64             : {
      65         620 :     geom_t ret = NONE;
      66         620 :     auto eFlatType = wkbFlatten(type);
      67             : 
      68         620 :     if (eFlatType == wkbPoint)
      69             :     {
      70         102 :         ret = POINT;
      71             :     }
      72             : 
      73         518 :     else if (eFlatType == wkbLineString)
      74             :     {
      75          12 :         ret = LINE;
      76             :     }
      77             : 
      78         506 :     else if (eFlatType == wkbPolygon)
      79             :     {
      80          42 :         ret = POLYGON;
      81             :     }
      82             : 
      83         464 :     else if (eFlatType == wkbMultiPoint)
      84             :     {
      85          19 :         ret = MULTIPOINT;
      86             :     }
      87             : 
      88         445 :     else if (eFlatType == wkbMultiLineString)
      89             :     {
      90          16 :         ret = MULTILINE;
      91             :     }
      92             : 
      93         429 :     else if (eFlatType == wkbMultiPolygon)
      94             :     {
      95         412 :         ret = MULTIPOLYGON;
      96             :     }
      97             : 
      98             :     // if the feature type isn't NONE potentially give a warning about measures
      99         620 :     if (ret != NONE && wkbHasM(type))
     100             :     {
     101           0 :         CPLError(CE_Warning, CPLE_NotSupported,
     102             :                  "A partially supported measured feature type was detected. X, "
     103             :                  "Y, Z Geometry will be preserved but the measure axis and "
     104             :                  "related information will be removed.");
     105             :     }
     106             : 
     107         620 :     return ret;
     108             : }
     109             : 
     110          42 : bool OGRHasZandSupported(OGRwkbGeometryType type)
     111             : {
     112          34 :     return type == wkbPoint25D || type == wkbLineString25D ||
     113          31 :            type == wkbPolygon25D || type == wkbMultiPoint25D ||
     114          76 :            type == wkbMultiLineString25D || type == wkbMultiPolygon25D;
     115             : }
     116             : 
     117             : }  // namespace nccfdriver
     118             : 
     119          75 : bool netCDFDataset::DetectAndFillSGLayers(int ncid)
     120             : {
     121             :     // Discover simple geometry variables
     122             :     int var_count;
     123          75 :     nc_inq_nvars(ncid, &var_count);
     124          75 :     std::set<int> vidList;
     125             : 
     126          75 :     nccfdriver::scanForGeometryContainers(ncid, vidList);
     127             : 
     128          75 :     if (!vidList.empty())
     129             :     {
     130         140 :         for (auto vid : vidList)
     131             :         {
     132             :             try
     133             :             {
     134          73 :                 LoadSGVarIntoLayer(ncid, vid);
     135             :             }
     136             : 
     137          14 :             catch (nccfdriver::SG_Exception &e)
     138             :             {
     139           7 :                 CPLError(CE_Warning, CPLE_AppDefined,
     140             :                          "Translation of a simple geometry layer has been "
     141             :                          "terminated prematurely due to an error.\n%s",
     142           7 :                          e.get_err_msg());
     143             :             }
     144             :         }
     145             :     }
     146             : 
     147         150 :     return !vidList.empty();
     148             : }
     149             : 
     150          73 : CPLErr netCDFDataset::LoadSGVarIntoLayer(int ncid, int nc_basevarId)
     151             : {
     152             :     std::shared_ptr<nccfdriver::SGeometry_Reader> sg(
     153         140 :         new nccfdriver::SGeometry_Reader(ncid, nc_basevarId));
     154          67 :     int cont_id = sg->getContainerId();
     155         134 :     nccfdriver::SGeometry_PropertyScanner pr(ncid, cont_id);
     156             :     OGRwkbGeometryType owgt =
     157          67 :         nccfdriver::RawToOGR(sg->getGeometryType(), sg->get_axisCount());
     158             : 
     159         134 :     std::string return_gm = "";
     160             : 
     161          67 :     if (sg->getGridMappingVarID() != nccfdriver::INVALID_VAR_ID)
     162          86 :         SetProjectionFromVar(ncid, nc_basevarId, true,
     163          43 :                              sg->getGridMappingName().c_str(), &return_gm,
     164             :                              sg.get(), /*paosRemovedMDItems=*/nullptr);
     165             : 
     166             :     // Geometry Type invalid, avoid further processing
     167          67 :     if (owgt == wkbNone)
     168             :     {
     169           1 :         throw nccfdriver::SG_Exception_BadFeature();
     170             :     }
     171             : 
     172             :     char baseName[NC_MAX_CHAR + 1];
     173          66 :     memset(baseName, 0, NC_MAX_CHAR + 1);
     174          66 :     nc_inq_varname(ncid, nc_basevarId, baseName);
     175             : 
     176          66 :     OGRSpatialReference *poSRS = nullptr;
     177          66 :     if (return_gm != "")
     178             :     {
     179          43 :         poSRS = new OGRSpatialReference();
     180          43 :         if (poSRS->importFromWkt(return_gm.c_str()) != OGRERR_NONE)
     181             :         {
     182           0 :             delete poSRS;
     183           0 :             throw nccfdriver::SG_Exception_General_Malformed("SRS settings");
     184             :         }
     185             : 
     186          43 :         poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
     187             :     }
     188             : 
     189             :     std::shared_ptr<netCDFLayer> poL(
     190         132 :         new netCDFLayer(this, ncid, baseName, owgt, poSRS));
     191             : 
     192          66 :     if (poSRS != nullptr)
     193             :     {
     194          43 :         poSRS->Release();
     195             :     }
     196             : 
     197          66 :     poL->EnableSGBypass();
     198          66 :     OGRFeatureDefn *defn = poL->GetLayerDefn();
     199          66 :     defn->SetGeomType(owgt);
     200             : 
     201             :     // Add properties
     202          66 :     std::vector<int> props = pr.ids();
     203         174 :     for (size_t itr = 0; itr < props.size(); itr++)
     204             :     {
     205         108 :         poL->AddField(props[itr]);
     206             :     }
     207             : 
     208             :     // Set simple geometry object
     209          66 :     poL->SetSGeometryRepresentation(sg);
     210             : 
     211             :     // Create layer
     212          66 :     papoLayers.push_back(poL);
     213             : 
     214         132 :     return CE_None;
     215             : }
     216             : 
     217             : /* Creates and fills any needed variables that haven't already been created
     218             :  */
     219         314 : bool netCDFDataset::SGCommitPendingTransaction()
     220             : {
     221             :     try
     222             :     {
     223         314 :         if (bSGSupport)
     224             :         {
     225             :             // Go through all the layers and resize dimensions accordingly
     226          85 :             for (size_t layerInd = 0; layerInd < papoLayers.size(); layerInd++)
     227             :             {
     228             :                 auto poLayer =
     229          45 :                     dynamic_cast<netCDFLayer *>(papoLayers[layerInd].get());
     230          45 :                 if (!poLayer)
     231           0 :                     continue;
     232             :                 nccfdriver::ncLayer_SG_Metadata &layerMD =
     233          45 :                     poLayer->getLayerSGMetadata();
     234          45 :                 nccfdriver::geom_t wType = layerMD.getWritableType();
     235             : 
     236             :                 // Resize node coordinates
     237          45 :                 int ncoord_did = layerMD.get_node_coord_dimID();
     238          45 :                 if (ncoord_did != nccfdriver::INVALID_DIM_ID)
     239             :                 {
     240          42 :                     vcdf.nc_resize_vdim(
     241             :                         ncoord_did, layerMD.get_next_write_pos_node_coord());
     242             :                 }
     243             : 
     244             :                 // Resize node count (for all except POINT)
     245          45 :                 if (wType != nccfdriver::POINT)
     246             :                 {
     247          35 :                     int ncount_did = layerMD.get_node_count_dimID();
     248          35 :                     if (ncount_did != nccfdriver::INVALID_DIM_ID)
     249             :                     {
     250          32 :                         vcdf.nc_resize_vdim(
     251             :                             ncount_did,
     252             :                             layerMD.get_next_write_pos_node_count());
     253             :                     }
     254             :                 }
     255             : 
     256             :                 // Resize part node count (for MULTILINE, POLYGON, MULTIPOLYGON)
     257          45 :                 if (wType == nccfdriver::MULTILINE ||
     258          36 :                     wType == nccfdriver::POLYGON ||
     259             :                     wType == nccfdriver::MULTIPOLYGON)
     260             :                 {
     261          30 :                     int pnc_did = layerMD.get_pnc_dimID();
     262          30 :                     if (pnc_did != nccfdriver::INVALID_DIM_ID)
     263             :                     {
     264          27 :                         vcdf.nc_resize_vdim(pnc_did,
     265             :                                             layerMD.get_next_write_pos_pnc());
     266             :                     }
     267             :                 }
     268             : 
     269          45 :                 nccfdriver::geom_t geometry_type = layerMD.getWritableType();
     270             : 
     271             :                 /* Delete interior ring stuff if not detected
     272             :                  */
     273             : 
     274          83 :                 if (!layerMD.getInteriorRingDetected() &&
     275          22 :                     (geometry_type == nccfdriver::MULTIPOLYGON ||
     276          83 :                      geometry_type == nccfdriver::POLYGON) &&
     277          20 :                     layerMD.get_containerRealID() != nccfdriver::INVALID_VAR_ID)
     278             :                 {
     279          17 :                     SetDefineMode(true);
     280             : 
     281             :                     int err_code =
     282          17 :                         nc_del_att(cdfid, layerMD.get_containerRealID(),
     283             :                                    CF_SG_INTERIOR_RING);
     284          17 :                     NCDF_ERR(err_code);
     285          17 :                     if (err_code != NC_NOERR)
     286             :                     {
     287           0 :                         std::string frmt = std::string("attribute: ") +
     288           0 :                                            std::string(CF_SG_INTERIOR_RING);
     289             :                         throw nccfdriver::SGWriter_Exception_NCDelFailure(
     290           0 :                             layerMD.get_containerName().c_str(), frmt.c_str());
     291             :                     }
     292             : 
     293             :                     // Invalidate variable writes as well - Interior Ring
     294          17 :                     vcdf.nc_del_vvar(layerMD.get_intring_varID());
     295             : 
     296          17 :                     if (geometry_type == nccfdriver::POLYGON)
     297             :                     {
     298             :                         err_code =
     299           4 :                             nc_del_att(cdfid, layerMD.get_containerRealID(),
     300             :                                        CF_SG_PART_NODE_COUNT);
     301           4 :                         NCDF_ERR(err_code);
     302           4 :                         if (err_code != NC_NOERR)
     303             :                         {
     304             :                             std::string frmt =
     305           0 :                                 std::string("attribute: ") +
     306           0 :                                 std::string(CF_SG_PART_NODE_COUNT);
     307             :                             throw nccfdriver::SGWriter_Exception_NCDelFailure(
     308           0 :                                 layerMD.get_containerName().c_str(),
     309           0 :                                 frmt.c_str());
     310             :                         }
     311             : 
     312             :                         // Invalidate variable writes as well - Part Node Count
     313           4 :                         vcdf.nc_del_vvar(layerMD.get_pnc_varID());
     314             : 
     315             :                         // Invalidate dimension as well - Part Node Count
     316           4 :                         vcdf.nc_del_vdim(layerMD.get_pnc_dimID());
     317             :                     }
     318             : 
     319          17 :                     SetDefineMode(false);
     320             :                 }
     321             :             }
     322             : 
     323          40 :             vcdf.nc_vmap();
     324          40 :             this->FieldScribe.commit_transaction();
     325          40 :             this->GeometryScribe.commit_transaction();
     326             :         }
     327             :     }
     328             : 
     329           0 :     catch (nccfdriver::SG_Exception &sge)
     330             :     {
     331           0 :         CPLError(CE_Fatal, CPLE_FileIO,
     332             :                  "An error occurred while writing the target netCDF File. "
     333             :                  "Translation will be terminated.\n%s",
     334           0 :                  sge.get_err_msg());
     335           0 :         return false;
     336             :     }
     337         314 :     return true;
     338             : }
     339             : 
     340         180 : void netCDFDataset::SGLogPendingTransaction()
     341             : {
     342         180 :     GeometryScribe.log_transaction();
     343         180 :     FieldScribe.log_transaction();
     344         180 : }
     345             : 
     346             : /* Takes an index and using the layer geometry builds the equivalent
     347             :  * OGRFeature.
     348             :  */
     349         492 : OGRFeature *netCDFLayer::buildSGeometryFeature(size_t featureInd)
     350             : {
     351             :     OGRGeometry *geometry;
     352             : 
     353         492 :     switch (m_simpleGeometryReader->getGeometryType())
     354             :     {
     355          39 :         case nccfdriver::POINT:
     356          39 :             geometry = new OGRPoint;
     357          39 :             break;
     358          16 :         case nccfdriver::LINE:
     359          16 :             geometry = new OGRLineString;
     360          16 :             break;
     361          13 :         case nccfdriver::POLYGON:
     362          13 :             geometry = new OGRPolygon;
     363          13 :             break;
     364          25 :         case nccfdriver::MULTIPOINT:
     365          25 :             geometry = new OGRMultiPoint;
     366          25 :             break;
     367          18 :         case nccfdriver::MULTILINE:
     368          18 :             geometry = new OGRMultiLineString;
     369          18 :             break;
     370         381 :         case nccfdriver::MULTIPOLYGON:
     371         381 :             geometry = new OGRMultiPolygon;
     372         381 :             break;
     373           0 :         default:
     374           0 :             throw nccfdriver::SG_Exception_BadFeature();
     375             :             break;
     376             :     }
     377             : 
     378         492 :     const auto wkb = m_simpleGeometryReader->serializeToWKB(featureInd);
     379         492 :     geometry->importFromWkb(wkb.data(), wkb.size(), wkbVariantIso);
     380         492 :     geometry->assignSpatialReference(this->GetSpatialRef());
     381             : 
     382         492 :     OGRFeatureDefn *defn = this->GetLayerDefn();
     383         492 :     OGRFeature *feat = new OGRFeature(defn);
     384         492 :     feat->SetGeometryDirectly(geometry);
     385             : 
     386         492 :     int dimId = m_simpleGeometryReader->getInstDim();
     387             : 
     388         492 :     this->FillFeatureFromVar(feat, dimId, featureInd);
     389             : 
     390         492 :     feat->SetFID(featureInd);
     391         984 :     return feat;
     392             : }
     393             : 
     394        1942 : std::string netCDFDataset::generateLogName()
     395             : {
     396        1942 :     return std::string(CPLGenerateTempFilenameSafe(nullptr));
     397             : }

Generated by: LCOV version 1.14