LCOV - code coverage report
Current view: top level - port - cpl_json.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 549 597 92.0 %
Date: 2025-01-18 12:42:00 Functions: 91 92 98.9 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  * Project:  Common Portability Library
       3             :  * Purpose:  Function wrapper for libjson-c access.
       4             :  * Author:   Dmitry Baryshnikov, dmitry.baryshnikov@nextgis.com
       5             :  *
       6             :  ******************************************************************************
       7             :  * Copyright (c) 2017-2018 NextGIS, <info@nextgis.com>
       8             :  *
       9             :  * SPDX-License-Identifier: MIT
      10             :  ****************************************************************************/
      11             : 
      12             : #include "cpl_json.h"
      13             : 
      14             : #include "cpl_error.h"
      15             : #include "cpl_json_header.h"
      16             : #include "cpl_vsi.h"
      17             : 
      18             : #include "cpl_http.h"
      19             : #include "cpl_multiproc.h"
      20             : 
      21             : #define TO_JSONOBJ(x) static_cast<json_object *>(x)
      22             : 
      23             : static const char *JSON_PATH_DELIMITER = "/";
      24             : 
      25             : static const char *INVALID_OBJ_KEY = "__INVALID_OBJ_KEY__";
      26             : 
      27             : #define JSON_C_VER_014 (14 << 8)
      28             : 
      29             : // json_object_new_uint64() was added in libjson-c 0.14
      30             : #if (!defined(JSON_C_VERSION_NUM)) || (JSON_C_VERSION_NUM < JSON_C_VER_014)
      31             : 
      32             : static int CPLJSON_json_object_new_uint64_formatter(struct json_object *jso,
      33             :                                                     struct printbuf *pb,
      34             :                                                     int /* level */,
      35             :                                                     int /* flags */)
      36             : {
      37             :     const char *pszStr = json_object_get_string(jso);
      38             :     return printbuf_memappend(pb, pszStr, static_cast<int>(strlen(pszStr)));
      39             : }
      40             : 
      41             : static json_object *CPLJSON_json_object_new_uint64(uint64_t nVal)
      42             : {
      43             :     json_object *jso = json_object_new_string(
      44             :         CPLSPrintf(CPL_FRMT_GUIB, static_cast<GUIntBig>(nVal)));
      45             :     json_object_set_serializer(jso, CPLJSON_json_object_new_uint64_formatter,
      46             :                                nullptr, nullptr);
      47             :     return jso;
      48             : }
      49             : 
      50             : #define json_object_new_uint64 CPLJSON_json_object_new_uint64
      51             : 
      52             : #endif
      53             : 
      54             : //------------------------------------------------------------------------------
      55             : // JSONDocument
      56             : //------------------------------------------------------------------------------
      57             : /*! @cond Doxygen_Suppress */
      58        8895 : CPLJSONDocument::CPLJSONDocument() : m_poRootJsonObject(nullptr)
      59             : {
      60        8895 : }
      61             : 
      62       17804 : CPLJSONDocument::~CPLJSONDocument()
      63             : {
      64        8902 :     if (m_poRootJsonObject)
      65        8098 :         json_object_put(TO_JSONOBJ(m_poRootJsonObject));
      66        8902 : }
      67             : 
      68           6 : CPLJSONDocument::CPLJSONDocument(const CPLJSONDocument &other)
      69           6 :     : m_poRootJsonObject(json_object_get(TO_JSONOBJ(other.m_poRootJsonObject)))
      70             : {
      71           6 : }
      72             : 
      73           2 : CPLJSONDocument &CPLJSONDocument::operator=(const CPLJSONDocument &other)
      74             : {
      75           2 :     if (this == &other)
      76           1 :         return *this;
      77             : 
      78           1 :     if (m_poRootJsonObject)
      79           1 :         json_object_put(TO_JSONOBJ(m_poRootJsonObject));
      80           1 :     m_poRootJsonObject = json_object_get(TO_JSONOBJ(other.m_poRootJsonObject));
      81             : 
      82           1 :     return *this;
      83             : }
      84             : 
      85           1 : CPLJSONDocument::CPLJSONDocument(CPLJSONDocument &&other)
      86           1 :     : m_poRootJsonObject(other.m_poRootJsonObject)
      87             : {
      88           1 :     other.m_poRootJsonObject = nullptr;
      89           1 : }
      90             : 
      91         118 : CPLJSONDocument &CPLJSONDocument::operator=(CPLJSONDocument &&other)
      92             : {
      93         118 :     if (this == &other)
      94           0 :         return *this;
      95             : 
      96         118 :     if (m_poRootJsonObject)
      97          21 :         json_object_put(TO_JSONOBJ(m_poRootJsonObject));
      98         118 :     m_poRootJsonObject = other.m_poRootJsonObject;
      99         118 :     other.m_poRootJsonObject = nullptr;
     100             : 
     101         118 :     return *this;
     102             : }
     103             : 
     104             : /*! @endcond */
     105             : 
     106             : /**
     107             :  * Save json document at specified path
     108             :  * @param  osPath Path to save json document
     109             :  * @return         true on success. If error occurred it can be received using
     110             :  * CPLGetLastErrorMsg method.
     111             :  *
     112             :  * @since GDAL 2.3
     113             :  */
     114        1233 : bool CPLJSONDocument::Save(const std::string &osPath) const
     115             : {
     116        1233 :     VSILFILE *fp = VSIFOpenL(osPath.c_str(), "wt");
     117        1233 :     if (nullptr == fp)
     118             :     {
     119           2 :         CPLError(CE_Failure, CPLE_NoWriteAccess, "Open file %s to write failed",
     120             :                  osPath.c_str());
     121           2 :         return false;
     122             :     }
     123             : 
     124        2462 :     const char *pabyData = json_object_to_json_string_ext(
     125        1231 :         TO_JSONOBJ(m_poRootJsonObject), JSON_C_TO_STRING_PRETTY);
     126        1231 :     VSIFWriteL(pabyData, 1, strlen(pabyData), fp);
     127             : 
     128        1231 :     VSIFCloseL(fp);
     129             : 
     130        1231 :     return true;
     131             : }
     132             : 
     133             : /**
     134             :  * Return the json document as a serialized string.
     135             :  * @return         serialized document.
     136             :  *
     137             :  * @since GDAL 2.3
     138             :  */
     139        1363 : std::string CPLJSONDocument::SaveAsString() const
     140             : {
     141        1363 :     return json_object_to_json_string_ext(TO_JSONOBJ(m_poRootJsonObject),
     142        1363 :                                           JSON_C_TO_STRING_PRETTY);
     143             : }
     144             : 
     145             : /**
     146             :  * Get json document root object
     147             :  * @return CPLJSONObject class instance
     148             :  *
     149             :  * @since GDAL 3.1
     150             :  */
     151          90 : const CPLJSONObject CPLJSONDocument::GetRoot() const
     152             : {
     153          90 :     return const_cast<CPLJSONDocument *>(this)->GetRoot();
     154             : }
     155             : 
     156             : /**
     157             :  * Get json document root object
     158             :  * @return CPLJSONObject class instance
     159             :  *
     160             :  * @since GDAL 2.3
     161             :  */
     162       13960 : CPLJSONObject CPLJSONDocument::GetRoot()
     163             : {
     164       13960 :     if (nullptr == m_poRootJsonObject)
     165             :     {
     166        1046 :         m_poRootJsonObject = json_object_new_object();
     167             :     }
     168             : 
     169       13960 :     if (json_object_get_type(TO_JSONOBJ(m_poRootJsonObject)) == json_type_array)
     170             :     {
     171         378 :         return CPLJSONArray("", m_poRootJsonObject);
     172             :     }
     173             :     else
     174             :     {
     175       13771 :         return CPLJSONObject("", m_poRootJsonObject);
     176             :     }
     177             : }
     178             : 
     179             : /**
     180             :  * Set json document root object
     181             :  * @param oRoot CPLJSONObject root object
     182             :  *
     183             :  * @since GDAL 3.4
     184             :  */
     185        1716 : void CPLJSONDocument::SetRoot(const CPLJSONObject &oRoot)
     186             : {
     187        1716 :     if (m_poRootJsonObject)
     188           0 :         json_object_put(TO_JSONOBJ(m_poRootJsonObject));
     189        1716 :     m_poRootJsonObject = json_object_get(TO_JSONOBJ(oRoot.m_poJsonObject));
     190        1716 : }
     191             : 
     192             : /**
     193             :  * Load json document from file by provided path
     194             :  * @param  osPath Path to json file.
     195             :  * @return         true on success. If error occurred it can be received using
     196             :  * CPLGetLastErrorMsg method.
     197             :  *
     198             :  * @since GDAL 2.3
     199             :  */
     200        2716 : bool CPLJSONDocument::Load(const std::string &osPath)
     201             : {
     202        2716 :     GByte *pabyOut = nullptr;
     203        2716 :     vsi_l_offset nSize = 0;
     204        2716 :     if (!VSIIngestFile(nullptr, osPath.c_str(), &pabyOut, &nSize,
     205             :                        100 * 1024 * 1024))  // Maximum 100 Mb allowed
     206             :     {
     207         259 :         CPLError(CE_Failure, CPLE_FileIO, "Load json file %s failed",
     208             :                  osPath.c_str());
     209         259 :         return false;
     210             :     }
     211             : 
     212        2457 :     bool bResult = LoadMemory(pabyOut, static_cast<int>(nSize));
     213        2457 :     VSIFree(pabyOut);
     214        2457 :     return bResult;
     215             : }
     216             : 
     217             : /**
     218             :  * Load json document from memory buffer.
     219             :  * @param  pabyData Buffer.data.
     220             :  * @param  nLength  Buffer size.
     221             :  * @return          true on success. If error occurred it can be received using
     222             :  * CPLGetLastErrorMsg method.
     223             :  *
     224             :  * @since GDAL 2.3
     225             :  */
     226        6584 : bool CPLJSONDocument::LoadMemory(const GByte *pabyData, int nLength)
     227             : {
     228        6584 :     if (nullptr == pabyData)
     229             :     {
     230           1 :         return false;
     231             :     }
     232             : 
     233        6583 :     if (m_poRootJsonObject)
     234        1217 :         json_object_put(TO_JSONOBJ(m_poRootJsonObject));
     235             : 
     236        6583 :     if (nLength == 4 &&
     237           5 :         memcmp(reinterpret_cast<const char *>(pabyData), "true", nLength) == 0)
     238             :     {
     239           5 :         m_poRootJsonObject = json_object_new_boolean(true);
     240           5 :         return true;
     241             :     }
     242             : 
     243        6578 :     if (nLength == 5 &&
     244           5 :         memcmp(reinterpret_cast<const char *>(pabyData), "false", nLength) == 0)
     245             :     {
     246           3 :         m_poRootJsonObject = json_object_new_boolean(false);
     247           3 :         return true;
     248             :     }
     249             : 
     250        6575 :     json_tokener *jstok = json_tokener_new();
     251        6575 :     m_poRootJsonObject = json_tokener_parse_ex(
     252             :         jstok, reinterpret_cast<const char *>(pabyData), nLength);
     253        6575 :     bool bParsed = jstok->err == json_tokener_success;
     254        6575 :     if (!bParsed)
     255             :     {
     256          13 :         CPLError(CE_Failure, CPLE_AppDefined,
     257             :                  "JSON parsing error: %s (at offset %d)",
     258             :                  json_tokener_error_desc(jstok->err), jstok->char_offset);
     259          13 :         json_tokener_free(jstok);
     260          13 :         return false;
     261             :     }
     262        6562 :     json_tokener_free(jstok);
     263        6562 :     return bParsed;
     264             : }
     265             : 
     266             : /**
     267             :  * Load json document from memory buffer.
     268             :  * @param  osStr    String
     269             :  * @return          true on success. If error occurred it can be received using
     270             :  * CPLGetLastErrorMsg method.
     271             :  *
     272             :  * @since GDAL 2.3
     273             :  */
     274        4031 : bool CPLJSONDocument::LoadMemory(const std::string &osStr)
     275             : {
     276        4031 :     if (osStr.empty())
     277           3 :         return false;
     278        4028 :     return LoadMemory(reinterpret_cast<const GByte *>(osStr.data()),
     279        8056 :                       static_cast<int>(osStr.size()));
     280             : }
     281             : 
     282             : /**
     283             :  * Load json document from file using small chunks of data.
     284             :  * @param  osPath      Path to json document file.
     285             :  * @param  nChunkSize   Chunk size.
     286             :  * @param  pfnProgress  a function to report progress of the json data loading.
     287             :  * @param  pProgressArg application data passed into progress function.
     288             :  * @return              true on success. If error occurred it can be received
     289             :  * using CPLGetLastErrorMsg method.
     290             :  *
     291             :  * @since GDAL 2.3
     292             :  */
     293           3 : bool CPLJSONDocument::LoadChunks(const std::string &osPath, size_t nChunkSize,
     294             :                                  GDALProgressFunc pfnProgress,
     295             :                                  void *pProgressArg)
     296             : {
     297             :     VSIStatBufL sStatBuf;
     298           3 :     if (VSIStatL(osPath.c_str(), &sStatBuf) != 0)
     299             :     {
     300           2 :         CPLError(CE_Failure, CPLE_FileIO, "Cannot open %s", osPath.c_str());
     301           2 :         return false;
     302             :     }
     303             : 
     304           1 :     VSILFILE *fp = VSIFOpenL(osPath.c_str(), "rb");
     305           1 :     if (fp == nullptr)
     306             :     {
     307           0 :         CPLError(CE_Failure, CPLE_FileIO, "Cannot open %s", osPath.c_str());
     308           0 :         return false;
     309             :     }
     310             : 
     311           1 :     void *pBuffer = CPLMalloc(nChunkSize);
     312           1 :     json_tokener *tok = json_tokener_new();
     313           1 :     bool bSuccess = true;
     314           1 :     GUInt32 nFileSize = static_cast<GUInt32>(sStatBuf.st_size);
     315           1 :     double dfTotalRead = 0.0;
     316             : 
     317             :     while (true)
     318             :     {
     319           6 :         size_t nRead = VSIFReadL(pBuffer, 1, nChunkSize, fp);
     320           6 :         dfTotalRead += nRead;
     321             : 
     322           6 :         if (m_poRootJsonObject)
     323           1 :             json_object_put(TO_JSONOBJ(m_poRootJsonObject));
     324             : 
     325           6 :         m_poRootJsonObject = json_tokener_parse_ex(
     326             :             tok, static_cast<const char *>(pBuffer), static_cast<int>(nRead));
     327             : 
     328           6 :         enum json_tokener_error jerr = json_tokener_get_error(tok);
     329           6 :         if (jerr != json_tokener_continue && jerr != json_tokener_success)
     330             :         {
     331           0 :             CPLError(CE_Failure, CPLE_AppDefined, "JSON error: %s",
     332             :                      json_tokener_error_desc(jerr));
     333           0 :             bSuccess = false;
     334           0 :             break;
     335             :         }
     336             : 
     337           6 :         if (nRead < nChunkSize)
     338             :         {
     339           1 :             break;
     340             :         }
     341             : 
     342           5 :         if (nullptr != pfnProgress)
     343             :         {
     344           0 :             pfnProgress(dfTotalRead / nFileSize, "Loading ...", pProgressArg);
     345             :         }
     346           5 :     }
     347             : 
     348           1 :     json_tokener_free(tok);
     349           1 :     CPLFree(pBuffer);
     350           1 :     VSIFCloseL(fp);
     351             : 
     352           1 :     if (nullptr != pfnProgress)
     353             :     {
     354           0 :         pfnProgress(1.0, "Loading ...", pProgressArg);
     355             :     }
     356             : 
     357           1 :     return bSuccess;
     358             : }
     359             : 
     360             : /*! @cond Doxygen_Suppress */
     361             : #ifdef HAVE_CURL
     362             : 
     363             : typedef struct
     364             : {
     365             :     json_object *pObject;
     366             :     json_tokener *pTokener;
     367             : } JsonContext, *JsonContextL;
     368             : 
     369           0 : static size_t CPLJSONWriteFunction(void *pBuffer, size_t nSize, size_t nMemb,
     370             :                                    void *pUserData)
     371             : {
     372           0 :     size_t nLength = nSize * nMemb;
     373           0 :     JsonContextL ctx = static_cast<JsonContextL>(pUserData);
     374           0 :     if (ctx->pObject != nullptr)
     375             :     {
     376           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     377             :                  "A complete JSon object had already been parsed before new "
     378             :                  "content is appended to it");
     379           0 :         return 0;
     380             :     }
     381           0 :     ctx->pObject =
     382           0 :         json_tokener_parse_ex(ctx->pTokener, static_cast<const char *>(pBuffer),
     383             :                               static_cast<int>(nLength));
     384           0 :     switch (json_tokener_get_error(ctx->pTokener))
     385             :     {
     386           0 :         case json_tokener_continue:
     387             :         case json_tokener_success:
     388           0 :             return nLength;
     389           0 :         default:
     390           0 :             return 0; /* error: interrupt the transfer */
     391             :     }
     392             : }
     393             : 
     394             : #endif  // HAVE_CURL
     395             : /*! @endcond */
     396             : 
     397             : /**
     398             :  * Load json document from web.
     399             :  * @param  osUrl       Url to json document.
     400             :  * @param  papszOptions Option list as a NULL-terminated array of strings. May
     401             :  * be NULL. The available keys are same for CPLHTTPFetch method. Additional key
     402             :  * JSON_DEPTH define json parse depth. Default is 10.
     403             :  * @param  pfnProgress  a function to report progress of the json data loading.
     404             :  * @param  pProgressArg application data passed into progress function.
     405             :  * @return              true on success. If error occurred it can be received
     406             :  * using CPLGetLastErrorMsg method.
     407             :  *
     408             :  * @since GDAL 2.3
     409             :  */
     410             : 
     411             : #ifdef HAVE_CURL
     412           2 : bool CPLJSONDocument::LoadUrl(const std::string &osUrl,
     413             :                               const char *const *papszOptions,
     414             :                               GDALProgressFunc pfnProgress, void *pProgressArg)
     415             : #else
     416             : bool CPLJSONDocument::LoadUrl(const std::string & /*osUrl*/,
     417             :                               const char *const * /*papszOptions*/,
     418             :                               GDALProgressFunc /*pfnProgress*/,
     419             :                               void * /*pProgressArg*/)
     420             : #endif  // HAVE_CURL
     421             : {
     422             : #ifdef HAVE_CURL
     423             :     int nDepth =
     424           2 :         atoi(CSLFetchNameValueDef(papszOptions, "JSON_DEPTH",
     425             :                                   "32"));  // Same as JSON_TOKENER_DEFAULT_DEPTH
     426           2 :     JsonContext ctx = {nullptr, json_tokener_new_ex(nDepth)};
     427             : 
     428           2 :     CPLHTTPFetchWriteFunc pWriteFunc = CPLJSONWriteFunction;
     429             :     CPLHTTPResult *psResult =
     430           2 :         CPLHTTPFetchEx(osUrl.c_str(), papszOptions, pfnProgress, pProgressArg,
     431             :                        pWriteFunc, &ctx);
     432             : 
     433           2 :     bool bResult =
     434           2 :         psResult->nStatus == 0 /*CURLE_OK*/ && psResult->pszErrBuf == nullptr;
     435             : 
     436           2 :     CPLHTTPDestroyResult(psResult);
     437             : 
     438             :     enum json_tokener_error jerr;
     439           2 :     if ((jerr = json_tokener_get_error(ctx.pTokener)) != json_tokener_success)
     440             :     {
     441           0 :         CPLError(CE_Failure, CPLE_AppDefined, "JSON error: %s\n",
     442             :                  json_tokener_error_desc(jerr));
     443           0 :         bResult = false;
     444             :     }
     445             :     else
     446             :     {
     447           2 :         if (m_poRootJsonObject)
     448           1 :             json_object_put(TO_JSONOBJ(m_poRootJsonObject));
     449             : 
     450           2 :         m_poRootJsonObject = ctx.pObject;
     451             :     }
     452           2 :     json_tokener_free(ctx.pTokener);
     453             : 
     454           2 :     return bResult;
     455             : #else
     456             :     CPLError(CE_Failure, CPLE_NotSupported,
     457             :              "LoadUrl() not supported in a build without Curl");
     458             :     return false;
     459             : #endif
     460             : }
     461             : 
     462             : //------------------------------------------------------------------------------
     463             : // JSONObject
     464             : //------------------------------------------------------------------------------
     465             : /*! @cond Doxygen_Suppress */
     466      127543 : CPLJSONObject::CPLJSONObject() : m_poJsonObject(json_object_new_object())
     467             : {
     468      127543 : }
     469             : 
     470           1 : CPLJSONObject::CPLJSONObject(std::nullptr_t) : m_poJsonObject(nullptr)
     471             : {
     472           1 : }
     473             : 
     474           1 : CPLJSONObject::CPLJSONObject(const std::string &osVal)
     475           1 :     : m_poJsonObject(json_object_new_string(osVal.c_str()))
     476             : {
     477           1 : }
     478             : 
     479           1 : CPLJSONObject::CPLJSONObject(const char *pszValue)
     480           1 :     : m_poJsonObject(json_object_new_string(pszValue))
     481             : {
     482           1 : }
     483             : 
     484           1 : CPLJSONObject::CPLJSONObject(bool bVal)
     485           1 :     : m_poJsonObject(json_object_new_boolean(bVal))
     486             : {
     487           1 : }
     488             : 
     489           1 : CPLJSONObject::CPLJSONObject(int nVal)
     490           1 :     : m_poJsonObject(json_object_new_int(nVal))
     491             : {
     492           1 : }
     493             : 
     494           1 : CPLJSONObject::CPLJSONObject(int64_t nVal)
     495           1 :     : m_poJsonObject(json_object_new_int64(nVal))
     496             : {
     497           1 : }
     498             : 
     499           1 : CPLJSONObject::CPLJSONObject(uint64_t nVal)
     500           1 :     : m_poJsonObject(json_object_new_uint64(nVal))
     501             : {
     502           1 : }
     503             : 
     504           1 : CPLJSONObject::CPLJSONObject(double dfVal)
     505           1 :     : m_poJsonObject(json_object_new_double(dfVal))
     506             : {
     507           1 : }
     508             : 
     509         817 : CPLJSONObject::CPLJSONObject(const std::string &osName,
     510         817 :                              const CPLJSONObject &oParent)
     511         817 :     : m_poJsonObject(json_object_get(json_object_new_object())), m_osKey(osName)
     512             : {
     513         817 :     json_object_object_add(TO_JSONOBJ(oParent.m_poJsonObject), osName.c_str(),
     514         817 :                            TO_JSONOBJ(m_poJsonObject));
     515         817 : }
     516             : 
     517      804053 : CPLJSONObject::CPLJSONObject(const std::string &osName,
     518      804053 :                              JSONObjectH poJsonObject)
     519      804053 :     : m_poJsonObject(json_object_get(TO_JSONOBJ(poJsonObject))), m_osKey(osName)
     520             : {
     521      804053 : }
     522             : 
     523         499 : CPLJSONObject CPLJSONObject::Clone() const
     524             : {
     525         499 :     CPLJSONObject oRet;
     526         499 :     if (IsValid())
     527             :     {
     528         998 :         CPLJSONDocument oTmpDoc;
     529         499 :         oTmpDoc.SetRoot(*this);
     530         499 :         std::string osStr = oTmpDoc.SaveAsString();
     531         499 :         CPL_IGNORE_RET_VAL(oTmpDoc.LoadMemory(osStr));
     532         499 :         oRet = oTmpDoc.GetRoot();
     533             :     }
     534         499 :     return oRet;
     535             : }
     536             : 
     537     1705920 : CPLJSONObject::~CPLJSONObject()
     538             : {
     539             :     // Should delete m_poJsonObject only if CPLJSONObject has no parent
     540     1705920 :     if (m_poJsonObject)
     541             :     {
     542     1440300 :         json_object_put(TO_JSONOBJ(m_poJsonObject));
     543     1440300 :         m_poJsonObject = nullptr;
     544             :     }
     545     1705920 : }
     546             : 
     547      573668 : CPLJSONObject::CPLJSONObject(const CPLJSONObject &other)
     548      573668 :     : m_poJsonObject(json_object_get(TO_JSONOBJ(other.m_poJsonObject))),
     549      573668 :       m_osKey(other.m_osKey)
     550             : {
     551      573668 : }
     552             : 
     553      199828 : CPLJSONObject::CPLJSONObject(CPLJSONObject &&other)
     554      199828 :     : m_poJsonObject(other.m_poJsonObject), m_osKey(std::move(other.m_osKey))
     555             : {
     556      199828 :     other.m_poJsonObject = nullptr;
     557      199828 : }
     558             : 
     559       52978 : CPLJSONObject &CPLJSONObject::operator=(const CPLJSONObject &other)
     560             : {
     561       52978 :     if (this == &other)
     562           1 :         return *this;
     563             : 
     564       52977 :     m_osKey = other.m_osKey;
     565       52977 :     if (m_poJsonObject)
     566       52224 :         json_object_put(TO_JSONOBJ(m_poJsonObject));
     567       52977 :     m_poJsonObject = json_object_get(TO_JSONOBJ(other.m_poJsonObject));
     568       52977 :     return *this;
     569             : }
     570             : 
     571       11445 : CPLJSONObject &CPLJSONObject::operator=(CPLJSONObject &&other)
     572             : {
     573       11445 :     if (this == &other)
     574           0 :         return *this;
     575             : 
     576       11445 :     m_osKey = std::move(other.m_osKey);
     577       11445 :     if (m_poJsonObject)
     578        6238 :         json_object_put(TO_JSONOBJ(m_poJsonObject));
     579       11445 :     m_poJsonObject = other.m_poJsonObject;
     580       11445 :     other.m_poJsonObject = nullptr;
     581       11445 :     return *this;
     582             : }
     583             : 
     584             : /*! @endcond */
     585             : 
     586             : /**
     587             :  * Add new key - value pair to json object.
     588             :  * @param osName Key name.
     589             :  * @param osValue String value.
     590             :  *
     591             :  * @since GDAL 2.3
     592             :  */
     593       13031 : void CPLJSONObject::Add(const std::string &osName, const std::string &osValue)
     594             : {
     595       26062 :     std::string objectName;
     596       13031 :     if (m_osKey == INVALID_OBJ_KEY)
     597           0 :         m_osKey.clear();
     598       26062 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     599       13031 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     600             :                                 object.m_poJsonObject)) == json_type_object)
     601             :     {
     602       13031 :         json_object *poVal = json_object_new_string(osValue.c_str());
     603       13031 :         json_object_object_add(TO_JSONOBJ(object.GetInternalHandle()),
     604             :                                objectName.c_str(), poVal);
     605             :     }
     606       13031 : }
     607             : 
     608             : /**
     609             :  * Add new key - value pair to json object.
     610             :  * @param osName Key name.
     611             :  * @param pszValue String value.
     612             :  *
     613             :  * @since GDAL 2.3
     614             :  */
     615       13671 : void CPLJSONObject::Add(const std::string &osName, const char *pszValue)
     616             : {
     617       13671 :     if (nullptr == pszValue)
     618             :     {
     619           1 :         return;
     620             :     }
     621       13670 :     if (m_osKey == INVALID_OBJ_KEY)
     622           0 :         m_osKey.clear();
     623       27340 :     std::string objectName;
     624       27340 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     625       13670 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     626             :                                 object.m_poJsonObject)) == json_type_object)
     627             :     {
     628       13670 :         json_object *poVal = json_object_new_string(pszValue);
     629       13670 :         json_object_object_add(TO_JSONOBJ(object.GetInternalHandle()),
     630             :                                objectName.c_str(), poVal);
     631             :     }
     632             : }
     633             : 
     634             : // defined in ogr/ogrsf_frmts/geojson/ogrgeojsonwriter.cpp
     635             : CPL_C_START
     636             : /* %.XXXg formatting */
     637             : json_object CPL_DLL *
     638             : json_object_new_double_with_significant_figures(double dfVal,
     639             :                                                 int nSignificantFigures);
     640             : CPL_C_END
     641             : 
     642             : /**
     643             :  * Add new key - value pair to json object.
     644             :  * @param osName  Key name.
     645             :  * @param dfValue Double value.
     646             :  *
     647             :  * @since GDAL 2.3
     648             :  */
     649       10995 : void CPLJSONObject::Add(const std::string &osName, double dfValue)
     650             : {
     651       21990 :     std::string objectName;
     652       10995 :     if (m_osKey == INVALID_OBJ_KEY)
     653           0 :         m_osKey.clear();
     654       21990 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     655       10995 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     656             :                                 object.m_poJsonObject)) == json_type_object)
     657             :     {
     658             :         json_object *poVal =
     659       10995 :             json_object_new_double_with_significant_figures(dfValue, -1);
     660       10995 :         json_object_object_add(TO_JSONOBJ(object.GetInternalHandle()),
     661             :                                objectName.c_str(), poVal);
     662             :     }
     663       10995 : }
     664             : 
     665             : /**
     666             :  * Add new key - value pair to json object.
     667             :  * @param osName  Key name.
     668             :  * @param nValue Integer value.
     669             :  *
     670             :  * @since GDAL 2.3
     671             :  */
     672       14136 : void CPLJSONObject::Add(const std::string &osName, int nValue)
     673             : {
     674       28272 :     std::string objectName;
     675       14136 :     if (m_osKey == INVALID_OBJ_KEY)
     676           0 :         m_osKey.clear();
     677       28272 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     678       14136 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     679             :                                 object.m_poJsonObject)) == json_type_object)
     680             :     {
     681       14136 :         json_object *poVal = json_object_new_int(nValue);
     682       14136 :         json_object_object_add(TO_JSONOBJ(object.GetInternalHandle()),
     683             :                                objectName.c_str(), poVal);
     684             :     }
     685       14136 : }
     686             : 
     687             : /**
     688             :  * Add new key - value pair to json object.
     689             :  * @param osName  Key name.
     690             :  * @param nValue Long value.
     691             :  *
     692             :  * @since GDAL 2.3
     693             :  */
     694        5644 : void CPLJSONObject::Add(const std::string &osName, GInt64 nValue)
     695             : {
     696       11288 :     std::string objectName;
     697        5644 :     if (m_osKey == INVALID_OBJ_KEY)
     698           0 :         m_osKey.clear();
     699       11288 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     700        5644 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     701             :                                 object.m_poJsonObject)) == json_type_object)
     702             :     {
     703             :         json_object *poVal =
     704        5644 :             json_object_new_int64(static_cast<int64_t>(nValue));
     705        5644 :         json_object_object_add(TO_JSONOBJ(object.GetInternalHandle()),
     706             :                                objectName.c_str(), poVal);
     707             :     }
     708        5644 : }
     709             : 
     710             : /**
     711             :  * Add new key - value pair to json object.
     712             :  * @param osName  Key name.
     713             :  * @param nValue uint64_t value.
     714             :  *
     715             :  * @since GDAL 3.8
     716             :  */
     717         795 : void CPLJSONObject::Add(const std::string &osName, uint64_t nValue)
     718             : {
     719        1590 :     std::string objectName;
     720         795 :     if (m_osKey == INVALID_OBJ_KEY)
     721           0 :         m_osKey.clear();
     722        1590 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     723         795 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     724             :                                 object.m_poJsonObject)) == json_type_object)
     725             :     {
     726         795 :         json_object *poVal = json_object_new_uint64(nValue);
     727         795 :         json_object_object_add(TO_JSONOBJ(object.GetInternalHandle()),
     728             :                                objectName.c_str(), poVal);
     729             :     }
     730         795 : }
     731             : 
     732             : /**
     733             :  * Add new key - value pair to json object.
     734             :  * @param osName  Key name.
     735             :  * @param oValue   Array value.
     736             :  *
     737             :  * @since GDAL 2.3
     738             :  */
     739        5882 : void CPLJSONObject::Add(const std::string &osName, const CPLJSONArray &oValue)
     740             : {
     741       11764 :     std::string objectName;
     742        5882 :     if (m_osKey == INVALID_OBJ_KEY)
     743           0 :         m_osKey.clear();
     744       11764 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     745        5882 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     746             :                                 object.m_poJsonObject)) == json_type_object)
     747             :     {
     748       11764 :         json_object_object_add(
     749        5882 :             TO_JSONOBJ(object.GetInternalHandle()), objectName.c_str(),
     750        5882 :             json_object_get(TO_JSONOBJ(oValue.GetInternalHandle())));
     751             :     }
     752        5882 : }
     753             : 
     754             : /**
     755             :  * Add new key - value pair to json object.
     756             :  * @param osName  Key name.
     757             :  * @param oValue   Json object value.
     758             :  *
     759             :  * @since GDAL 2.3
     760             :  */
     761       13417 : void CPLJSONObject::Add(const std::string &osName, const CPLJSONObject &oValue)
     762             : {
     763       13417 :     std::string objectName;
     764       13417 :     if (m_osKey == INVALID_OBJ_KEY)
     765           0 :         m_osKey.clear();
     766       13417 :     if (osName.empty())
     767             :     {
     768          20 :         json_object_object_add(
     769          10 :             TO_JSONOBJ(GetInternalHandle()), "",
     770          10 :             json_object_get(TO_JSONOBJ(oValue.GetInternalHandle())));
     771          10 :         return;
     772             :     }
     773       26814 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     774       13407 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     775             :                                 object.m_poJsonObject)) == json_type_object)
     776             :     {
     777       26814 :         json_object_object_add(
     778       13407 :             TO_JSONOBJ(object.GetInternalHandle()), objectName.c_str(),
     779       13407 :             json_object_get(TO_JSONOBJ(oValue.GetInternalHandle())));
     780             :     }
     781             : }
     782             : 
     783             : /**
     784             :  * Add new key - value pair to json object.
     785             :  * @param osName  Key name (do not split it on '/')
     786             :  * @param oValue   Json object value.
     787             :  *
     788             :  * @since GDAL 3.2
     789             :  */
     790         845 : void CPLJSONObject::AddNoSplitName(const std::string &osName,
     791             :                                    const CPLJSONObject &oValue)
     792             : {
     793         845 :     if (m_osKey == INVALID_OBJ_KEY)
     794           0 :         m_osKey.clear();
     795        1690 :     if (IsValid() &&
     796         845 :         json_object_get_type(TO_JSONOBJ(m_poJsonObject)) == json_type_object)
     797             :     {
     798        1690 :         json_object_object_add(
     799         845 :             TO_JSONOBJ(GetInternalHandle()), osName.c_str(),
     800         845 :             json_object_get(TO_JSONOBJ(oValue.GetInternalHandle())));
     801             :     }
     802         845 : }
     803             : 
     804             : /**
     805             :  * Add new key - value pair to json object.
     806             :  * @param osName  Key name.
     807             :  * @param bValue   Boolean value.
     808             :  *
     809             :  * @since GDAL 2.3
     810             :  */
     811        2019 : void CPLJSONObject::Add(const std::string &osName, bool bValue)
     812             : {
     813        4038 :     std::string objectName;
     814        2019 :     if (m_osKey == INVALID_OBJ_KEY)
     815           0 :         m_osKey.clear();
     816        4038 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     817        2019 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     818             :                                 object.m_poJsonObject)) == json_type_object)
     819             :     {
     820        2019 :         json_object *poVal = json_object_new_boolean(bValue);
     821        2019 :         json_object_object_add(TO_JSONOBJ(object.GetInternalHandle()),
     822             :                                objectName.c_str(), poVal);
     823             :     }
     824        2019 : }
     825             : 
     826             : /**
     827             :  * Add new key - null pair to json object.
     828             :  * @param osName  Key name.
     829             :  *
     830             :  * @since GDAL 2.3
     831             :  */
     832       11408 : void CPLJSONObject::AddNull(const std::string &osName)
     833             : {
     834       22816 :     std::string objectName;
     835       11408 :     if (m_osKey == INVALID_OBJ_KEY)
     836           0 :         m_osKey.clear();
     837       22816 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     838       11408 :     if (object.IsValid() && json_object_get_type(TO_JSONOBJ(
     839             :                                 object.m_poJsonObject)) == json_type_object)
     840             :     {
     841       11408 :         json_object_object_add(TO_JSONOBJ(object.GetInternalHandle()),
     842             :                                objectName.c_str(), nullptr);
     843             :     }
     844       11408 : }
     845             : 
     846             : /**
     847             :  * Change value by key.
     848             :  * @param osName  Key name.
     849             :  * @param osValue String value.
     850             :  *
     851             :  * @since GDAL 2.3
     852             :  */
     853         500 : void CPLJSONObject::Set(const std::string &osName, const std::string &osValue)
     854             : {
     855         500 :     Delete(osName);
     856         500 :     Add(osName, osValue);
     857         500 : }
     858             : 
     859             : /**
     860             :  * Change value by key.
     861             :  * @param osName  Key name.
     862             :  * @param pszValue String value.
     863             :  *
     864             :  * @since GDAL 2.3
     865             :  */
     866        4156 : void CPLJSONObject::Set(const std::string &osName, const char *pszValue)
     867             : {
     868        4156 :     if (nullptr == pszValue)
     869           1 :         return;
     870        4155 :     Delete(osName);
     871        4155 :     Add(osName, pszValue);
     872             : }
     873             : 
     874             : /**
     875             :  * Change value by key.
     876             :  * @param osName  Key name.
     877             :  * @param dfValue  Double value.
     878             :  *
     879             :  * @since GDAL 2.3
     880             :  */
     881         517 : void CPLJSONObject::Set(const std::string &osName, double dfValue)
     882             : {
     883         517 :     Delete(osName);
     884         517 :     Add(osName, dfValue);
     885         517 : }
     886             : 
     887             : /**
     888             :  * Change value by key.
     889             :  * @param osName  Key name.
     890             :  * @param nValue   Integer value.
     891             :  *
     892             :  * @since GDAL 2.3
     893             :  */
     894        1394 : void CPLJSONObject::Set(const std::string &osName, int nValue)
     895             : {
     896        1394 :     Delete(osName);
     897        1394 :     Add(osName, nValue);
     898        1394 : }
     899             : 
     900             : /**
     901             :  * Change value by key.
     902             :  * @param osName  Key name.
     903             :  * @param nValue   Long value.
     904             :  *
     905             :  * @since GDAL 2.3
     906             :  */
     907         228 : void CPLJSONObject::Set(const std::string &osName, GInt64 nValue)
     908             : {
     909         228 :     Delete(osName);
     910         228 :     Add(osName, nValue);
     911         228 : }
     912             : 
     913             : /**
     914             :  * Change value by key.
     915             :  * @param osName  Key name.
     916             :  * @param nValue   uint64_t value.
     917             :  *
     918             :  * @since GDAL 3.8
     919             :  */
     920          40 : void CPLJSONObject::Set(const std::string &osName, uint64_t nValue)
     921             : {
     922          40 :     Delete(osName);
     923          40 :     Add(osName, nValue);
     924          40 : }
     925             : 
     926             : /**
     927             :  * Change value by key.
     928             :  * @param osName  Key name.
     929             :  * @param bValue   Boolean value.
     930             :  *
     931             :  * @since GDAL 2.3
     932             :  */
     933         451 : void CPLJSONObject::Set(const std::string &osName, bool bValue)
     934             : {
     935         451 :     Delete(osName);
     936         451 :     Add(osName, bValue);
     937         451 : }
     938             : 
     939             : /**
     940             :  * Change value by key.
     941             :  * @param osName  Key name.
     942             :  *
     943             :  * @since GDAL 2.3
     944             :  */
     945          15 : void CPLJSONObject::SetNull(const std::string &osName)
     946             : {
     947          15 :     Delete(osName);
     948          15 :     AddNull(osName);
     949          15 : }
     950             : 
     951             : /**
     952             :  * Get value by key.
     953             :  * @param  osName Key name.
     954             :  * @return         Json array object.
     955             :  *
     956             :  * @since GDAL 2.3
     957             :  */
     958       22257 : CPLJSONArray CPLJSONObject::GetArray(const std::string &osName) const
     959             : {
     960       44514 :     std::string objectName;
     961       44514 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     962       22257 :     if (object.IsValid())
     963             :     {
     964       22257 :         json_object *poVal = nullptr;
     965       22257 :         if (json_object_object_get_ex(TO_JSONOBJ(object.GetInternalHandle()),
     966       22257 :                                       objectName.c_str(), &poVal))
     967             :         {
     968       15908 :             if (poVal && json_object_get_type(poVal) == json_type_array)
     969             :             {
     970       15866 :                 return CPLJSONArray(objectName, poVal);
     971             :             }
     972             :         }
     973             :     }
     974        6391 :     return CPLJSONArray(INVALID_OBJ_KEY, nullptr);
     975             : }
     976             : 
     977             : /**
     978             :  * Get value by key.
     979             :  * @param  osName Key name.
     980             :  * @return         Json object.
     981             :  *
     982             :  * @since GDAL 2.3
     983             :  */
     984      407794 : CPLJSONObject CPLJSONObject::GetObj(const std::string &osName) const
     985             : {
     986      815588 :     std::string objectName;
     987      815588 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
     988      407794 :     if (object.IsValid())
     989             :     {
     990      407050 :         json_object *poVal = nullptr;
     991      407050 :         if (json_object_object_get_ex(TO_JSONOBJ(object.GetInternalHandle()),
     992      407050 :                                       objectName.c_str(), &poVal))
     993             :         {
     994      365857 :             return CPLJSONObject(objectName, poVal);
     995             :         }
     996             :     }
     997       41937 :     return CPLJSONObject(INVALID_OBJ_KEY, nullptr);
     998             : }
     999             : 
    1000             : /**
    1001             :  * Get value by key.
    1002             :  * @param  osName Key name.
    1003             :  * @return         Json object.
    1004             :  *
    1005             :  * @since GDAL 2.3
    1006             :  */
    1007      100824 : CPLJSONObject CPLJSONObject::operator[](const std::string &osName) const
    1008             : {
    1009      100824 :     return GetObj(osName);
    1010             : }
    1011             : 
    1012             : /**
    1013             :  * Delete json object by key.
    1014             :  * @param  osName Key name.
    1015             :  *
    1016             :  * @since GDAL 2.3
    1017             :  */
    1018        9625 : void CPLJSONObject::Delete(const std::string &osName)
    1019             : {
    1020       19250 :     std::string objectName;
    1021        9625 :     if (m_osKey == INVALID_OBJ_KEY)
    1022           0 :         m_osKey.clear();
    1023       19250 :     CPLJSONObject object = GetObjectByPath(osName, objectName);
    1024        9625 :     if (object.IsValid())
    1025             :     {
    1026        9625 :         json_object_object_del(TO_JSONOBJ(object.GetInternalHandle()),
    1027             :                                objectName.c_str());
    1028             :     }
    1029        9625 : }
    1030             : 
    1031             : /**
    1032             :  * Delete json object by key (without splitting on /)
    1033             :  * @param  osName Key name.
    1034             :  *
    1035             :  * @since GDAL 3.4
    1036             :  */
    1037         858 : void CPLJSONObject::DeleteNoSplitName(const std::string &osName)
    1038             : {
    1039         858 :     if (m_osKey == INVALID_OBJ_KEY)
    1040           0 :         m_osKey.clear();
    1041         858 :     if (m_poJsonObject)
    1042             :     {
    1043         858 :         json_object_object_del(TO_JSONOBJ(m_poJsonObject), osName.c_str());
    1044             :     }
    1045         858 : }
    1046             : 
    1047             : /**
    1048             :  * Get value by key.
    1049             :  * @param  osName    Key name.
    1050             :  * @param  osDefault Default value.
    1051             :  * @return            String value.
    1052             :  *
    1053             :  * @since GDAL 2.3
    1054             :  */
    1055      237262 : std::string CPLJSONObject::GetString(const std::string &osName,
    1056             :                                      const std::string &osDefault) const
    1057             : {
    1058      474524 :     CPLJSONObject object = GetObj(osName);
    1059      474524 :     return object.ToString(osDefault);
    1060             : }
    1061             : 
    1062             : /**
    1063             :  * Get value.
    1064             :  * @param  osDefault Default value.
    1065             :  * @return            String value.
    1066             :  *
    1067             :  * @since GDAL 2.3
    1068             :  */
    1069      382693 : std::string CPLJSONObject::ToString(const std::string &osDefault) const
    1070             : {
    1071      382693 :     if( m_poJsonObject /*&& json_object_get_type( TO_JSONOBJ(m_poJsonObject) ) ==
    1072             :             json_type_string*/ )
    1073             :     {
    1074             :         const char *pszString =
    1075      369209 :             json_object_get_string(TO_JSONOBJ(m_poJsonObject));
    1076      369209 :         if (nullptr != pszString)
    1077             :         {
    1078      369209 :             return pszString;
    1079             :         }
    1080             :     }
    1081       13484 :     return osDefault;
    1082             : }
    1083             : 
    1084             : /**
    1085             :  * Get value by key.
    1086             :  * @param  osName    Key name.
    1087             :  * @param  dfDefault  Default value.
    1088             :  * @return            Double value.
    1089             :  *
    1090             :  * @since GDAL 2.3
    1091             :  */
    1092        6373 : double CPLJSONObject::GetDouble(const std::string &osName,
    1093             :                                 double dfDefault) const
    1094             : {
    1095       12746 :     CPLJSONObject object = GetObj(osName);
    1096       12746 :     return object.ToDouble(dfDefault);
    1097             : }
    1098             : 
    1099             : /**
    1100             :  * Get value
    1101             :  * @param  dfDefault  Default value.
    1102             :  * @return            Double value.
    1103             :  *
    1104             :  * @since GDAL 2.3
    1105             :  */
    1106       23121 : double CPLJSONObject::ToDouble(double dfDefault) const
    1107             : {
    1108       23121 :     if( m_poJsonObject /*&& json_object_get_type( TO_JSONOBJ(m_poJsonObject) ) ==
    1109             :             json_type_double*/ )
    1110       22465 :         return json_object_get_double(TO_JSONOBJ(m_poJsonObject));
    1111         656 :     return dfDefault;
    1112             : }
    1113             : 
    1114             : /**
    1115             :  * Get value by key.
    1116             :  * @param  osName    Key name.
    1117             :  * @param  nDefault   Default value.
    1118             :  * @return            Integer value.
    1119             :  *
    1120             :  * @since GDAL 2.3
    1121             :  */
    1122       24217 : int CPLJSONObject::GetInteger(const std::string &osName, int nDefault) const
    1123             : {
    1124       48434 :     CPLJSONObject object = GetObj(osName);
    1125       48434 :     return object.ToInteger(nDefault);
    1126             : }
    1127             : 
    1128             : /**
    1129             :  * Get value.
    1130             :  * @param  nDefault   Default value.
    1131             :  * @return            Integer value.
    1132             :  *
    1133             :  * @since GDAL 2.3
    1134             :  */
    1135       25932 : int CPLJSONObject::ToInteger(int nDefault) const
    1136             : {
    1137       25932 :     if( m_poJsonObject /*&& json_object_get_type( TO_JSONOBJ(m_poJsonObject) ) ==
    1138             :             json_type_int*/ )
    1139       25071 :         return json_object_get_int(TO_JSONOBJ(m_poJsonObject));
    1140         861 :     return nDefault;
    1141             : }
    1142             : 
    1143             : /**
    1144             :  * Get value by key.
    1145             :  * @param  osName    Key name.
    1146             :  * @param  nDefault   Default value.
    1147             :  * @return            Long value.
    1148             :  *
    1149             :  * @since GDAL 2.3
    1150             :  */
    1151          91 : GInt64 CPLJSONObject::GetLong(const std::string &osName, GInt64 nDefault) const
    1152             : {
    1153         182 :     CPLJSONObject object = GetObj(osName);
    1154         182 :     return object.ToLong(nDefault);
    1155             : }
    1156             : 
    1157             : /**
    1158             :  * Get value.
    1159             :  * @param  nDefault   Default value.
    1160             :  * @return            Long value.
    1161             :  *
    1162             :  * @since GDAL 2.3
    1163             :  */
    1164        2576 : GInt64 CPLJSONObject::ToLong(GInt64 nDefault) const
    1165             : {
    1166        2576 :     if( m_poJsonObject /*&& json_object_get_type( TO_JSONOBJ(m_poJsonObject) ) ==
    1167             :             json_type_int*/ )
    1168             :         return static_cast<GInt64>(
    1169        2537 :             json_object_get_int64(TO_JSONOBJ(m_poJsonObject)));
    1170          39 :     return nDefault;
    1171             : }
    1172             : 
    1173             : /**
    1174             :  * Get value by key.
    1175             :  * @param  osName    Key name.
    1176             :  * @param  bDefault   Default value.
    1177             :  * @return            Boolean value.
    1178             :  *
    1179             :  * @since GDAL 2.3
    1180             :  */
    1181          70 : bool CPLJSONObject::GetBool(const std::string &osName, bool bDefault) const
    1182             : {
    1183         140 :     CPLJSONObject object = GetObj(osName);
    1184         140 :     return object.ToBool(bDefault);
    1185             : }
    1186             : 
    1187             : /**
    1188             :  * \brief Get json object children.
    1189             :  *
    1190             :  * This function is useful when keys is not know and need to
    1191             :  * iterate over json object items and get keys and values.
    1192             :  *
    1193             :  * @return Array of CPLJSONObject class instance.
    1194             :  *
    1195             :  * @since GDAL 2.3
    1196             :  */
    1197       12200 : std::vector<CPLJSONObject> CPLJSONObject::GetChildren() const
    1198             : {
    1199       12200 :     std::vector<CPLJSONObject> aoChildren;
    1200       24400 :     if (nullptr == m_poJsonObject ||
    1201       12200 :         json_object_get_type(TO_JSONOBJ(m_poJsonObject)) != json_type_object)
    1202             :     {
    1203           1 :         return aoChildren;
    1204             :     }
    1205             : 
    1206             :     json_object_iter it;
    1207       12199 :     it.key = nullptr;
    1208       12199 :     it.val = nullptr;
    1209       12199 :     it.entry = nullptr;
    1210             :     // cppcheck-suppress cstyleCast
    1211       59573 :     json_object_object_foreachC(TO_JSONOBJ(m_poJsonObject), it)
    1212             :     {
    1213       47374 :         aoChildren.push_back(CPLJSONObject(it.key, it.val));
    1214             :     }
    1215             : 
    1216       12199 :     return aoChildren;
    1217             : }
    1218             : 
    1219             : /**
    1220             :  * Get value.
    1221             :  * @param  bDefault   Default value.
    1222             :  * @return            Boolean value.
    1223             :  *
    1224             :  * @since GDAL 2.3
    1225             :  */
    1226          82 : bool CPLJSONObject::ToBool(bool bDefault) const
    1227             : {
    1228          82 :     if( m_poJsonObject /*&& json_object_get_type( TO_JSONOBJ(m_poJsonObject) ) ==
    1229             :             json_type_boolean*/ )
    1230          22 :         return json_object_get_boolean(TO_JSONOBJ(m_poJsonObject)) == 1;
    1231          60 :     return bDefault;
    1232             : }
    1233             : 
    1234             : /**
    1235             :  * Get value.
    1236             :  * @return            Array
    1237             :  *
    1238             :  * @since GDAL 2.3
    1239             :  */
    1240        8115 : CPLJSONArray CPLJSONObject::ToArray() const
    1241             : {
    1242       15301 :     if (m_poJsonObject &&
    1243        7186 :         json_object_get_type(TO_JSONOBJ(m_poJsonObject)) == json_type_array)
    1244        7182 :         return CPLJSONArray("", TO_JSONOBJ(m_poJsonObject));
    1245         933 :     return CPLJSONArray(INVALID_OBJ_KEY, nullptr);
    1246             : }
    1247             : 
    1248             : /**
    1249             :  * Stringify object to json format.
    1250             :  * @param  eFormat Format type,
    1251             :  * @return         A string in JSON format.
    1252             :  *
    1253             :  * @since GDAL 2.3
    1254             :  */
    1255       27988 : std::string CPLJSONObject::Format(PrettyFormat eFormat) const
    1256             : {
    1257       27988 :     if (m_poJsonObject)
    1258             :     {
    1259       27988 :         const char *pszFormatString = nullptr;
    1260       27988 :         switch (eFormat)
    1261             :         {
    1262           1 :             case PrettyFormat::Spaced:
    1263           2 :                 pszFormatString = json_object_to_json_string_ext(
    1264           1 :                     TO_JSONOBJ(m_poJsonObject), JSON_C_TO_STRING_SPACED);
    1265           1 :                 break;
    1266         178 :             case PrettyFormat::Pretty:
    1267         356 :                 pszFormatString = json_object_to_json_string_ext(
    1268         178 :                     TO_JSONOBJ(m_poJsonObject), JSON_C_TO_STRING_PRETTY);
    1269         178 :                 break;
    1270       27809 :             default:
    1271       27809 :                 pszFormatString = json_object_to_json_string_ext(
    1272       27809 :                     TO_JSONOBJ(m_poJsonObject), JSON_C_TO_STRING_PLAIN);
    1273             :         }
    1274       27988 :         if (nullptr != pszFormatString)
    1275             :         {
    1276       27988 :             return pszFormatString;
    1277             :         }
    1278             :     }
    1279           0 :     return "";
    1280             : }
    1281             : 
    1282             : /*! @cond Doxygen_Suppress */
    1283      530663 : CPLJSONObject CPLJSONObject::GetObjectByPath(const std::string &osPath,
    1284             :                                              std::string &osName) const
    1285             : {
    1286      530663 :     json_object *poVal = nullptr;
    1287             : 
    1288             :     // Typically for keys that contain / character
    1289      530663 :     if (json_object_object_get_ex(TO_JSONOBJ(GetInternalHandle()),
    1290      530663 :                                   osPath.c_str(), &poVal))
    1291             :     {
    1292      382169 :         osName = osPath;
    1293      382169 :         return *this;
    1294             :     }
    1295             : 
    1296             :     CPLStringList pathPortions(
    1297      296988 :         CSLTokenizeString2(osPath.c_str(), JSON_PATH_DELIMITER, 0));
    1298      148494 :     int portionsCount = pathPortions.size();
    1299      148494 :     if (portionsCount > 100)
    1300             :     {
    1301           0 :         CPLError(CE_Failure, CPLE_NotSupported, "Too many components in path");
    1302           0 :         return CPLJSONObject(INVALID_OBJ_KEY, nullptr);
    1303             :     }
    1304      148494 :     if (0 == portionsCount)
    1305           0 :         return CPLJSONObject(INVALID_OBJ_KEY, nullptr);
    1306      296988 :     CPLJSONObject object = *this;
    1307      150822 :     for (int i = 0; i < portionsCount - 1; ++i)
    1308             :     {
    1309             :         // TODO: check array index in path - i.e.
    1310             :         // settings/catalog/root/id:1/name if EQUALN(pathPortions[i+1], "id:",
    1311             :         // 3) -> getArray
    1312        2328 :         if (json_object_object_get_ex(TO_JSONOBJ(object.GetInternalHandle()),
    1313        4656 :                                       pathPortions[i], &poVal))
    1314             :         {
    1315        1511 :             object = CPLJSONObject(pathPortions[i], poVal);
    1316             :         }
    1317             :         else
    1318             :         {
    1319         817 :             if (json_object_get_type(TO_JSONOBJ(object.m_poJsonObject)) !=
    1320             :                 json_type_object)
    1321             :             {
    1322           0 :                 return CPLJSONObject(INVALID_OBJ_KEY, nullptr);
    1323             :             }
    1324         817 :             object = CPLJSONObject(pathPortions[i], object);
    1325             :         }
    1326             :     }
    1327             : 
    1328             :     //    // Check if such object already  exists
    1329             :     //    if(json_object_object_get_ex(object.m_jsonObject,
    1330             :     //                                 pathPortions[portionsCount - 1], &poVal))
    1331             :     //        return JSONObject(nullptr);
    1332             :     //
    1333      148494 :     osName = pathPortions[portionsCount - 1];
    1334      148494 :     return object;
    1335             : }
    1336             : 
    1337             : /*! @endcond */
    1338             : 
    1339             : /**
    1340             :  * Get json object type.
    1341             :  * @return Json object type.
    1342             :  *
    1343             :  * @since GDAL 2.3
    1344             :  */
    1345       85593 : CPLJSONObject::Type CPLJSONObject::GetType() const
    1346             : {
    1347       85593 :     if (nullptr == m_poJsonObject)
    1348             :     {
    1349       11936 :         if (m_osKey == INVALID_OBJ_KEY)
    1350        8844 :             return CPLJSONObject::Type::Unknown;
    1351        3092 :         return CPLJSONObject::Type::Null;
    1352             :     }
    1353       73657 :     auto jsonObj(TO_JSONOBJ(m_poJsonObject));
    1354       73657 :     switch (json_object_get_type(jsonObj))
    1355             :     {
    1356          12 :         case json_type_boolean:
    1357          12 :             return CPLJSONObject::Type::Boolean;
    1358        2982 :         case json_type_double:
    1359        2982 :             return CPLJSONObject::Type::Double;
    1360       15373 :         case json_type_int:
    1361             :         {
    1362       15373 :             if (CPL_INT64_FITS_ON_INT32(json_object_get_int64(jsonObj)))
    1363       15261 :                 return CPLJSONObject::Type::Integer;
    1364             :             else
    1365         112 :                 return CPLJSONObject::Type::Long;
    1366             :         }
    1367        9244 :         case json_type_object:
    1368        9244 :             return CPLJSONObject::Type::Object;
    1369        5464 :         case json_type_array:
    1370        5464 :             return CPLJSONObject::Type::Array;
    1371       40582 :         case json_type_string:
    1372       40582 :             return CPLJSONObject::Type::String;
    1373           0 :         default:
    1374           0 :             break;
    1375             :     }
    1376           0 :     return CPLJSONObject::Type::Unknown;
    1377             : }
    1378             : 
    1379             : /**
    1380             :  * Check if json object valid.
    1381             :  * @return true if json object valid.
    1382             :  *
    1383             :  * @since GDAL 2.3
    1384             :  */
    1385      586740 : bool CPLJSONObject::IsValid() const
    1386             : {
    1387      586740 :     return m_osKey != INVALID_OBJ_KEY;
    1388             : }
    1389             : 
    1390             : /**
    1391             :  * Decrement reference counter and make pointer NULL.
    1392             :  * A json object will become invalid.
    1393             :  *
    1394             :  * @since GDAL 2.3
    1395             :  */
    1396        8754 : void CPLJSONObject::Deinit()
    1397             : {
    1398        8754 :     if (m_poJsonObject)
    1399             :     {
    1400        6559 :         json_object_put(TO_JSONOBJ(m_poJsonObject));
    1401        6559 :         m_poJsonObject = nullptr;
    1402             :     }
    1403        8754 :     m_osKey = INVALID_OBJ_KEY;
    1404        8754 : }
    1405             : 
    1406             : //------------------------------------------------------------------------------
    1407             : // JSONArray
    1408             : //------------------------------------------------------------------------------
    1409             : /*! @cond Doxygen_Suppress */
    1410       30292 : CPLJSONArray::CPLJSONArray()
    1411             : {
    1412       30292 :     json_object_put(TO_JSONOBJ(m_poJsonObject));
    1413       30292 :     m_poJsonObject = json_object_new_array();
    1414       30292 : }
    1415             : 
    1416           1 : CPLJSONArray::CPLJSONArray(const std::string &osName)
    1417           1 :     : CPLJSONObject(osName, json_object_new_array())
    1418             : {
    1419           1 :     json_object_put(TO_JSONOBJ(m_poJsonObject));
    1420           1 : }
    1421             : 
    1422       30561 : CPLJSONArray::CPLJSONArray(const std::string &osName, JSONObjectH poJsonObject)
    1423       30561 :     : CPLJSONObject(osName, poJsonObject)
    1424             : {
    1425       30561 : }
    1426             : 
    1427           9 : CPLJSONArray::CPLJSONArray(const CPLJSONObject &other) : CPLJSONObject(other)
    1428             : {
    1429           9 : }
    1430             : 
    1431             : /*! @endcond */
    1432             : 
    1433             : /**
    1434             :  * Get array size.
    1435             :  * @return Array size.
    1436             :  *
    1437             :  * @since GDAL 2.3
    1438             :  */
    1439      275084 : int CPLJSONArray::Size() const
    1440             : {
    1441      275084 :     if (m_poJsonObject)
    1442             :         return static_cast<int>(
    1443      272917 :             json_object_array_length(TO_JSONOBJ(m_poJsonObject)));
    1444        2167 :     return 0;
    1445             : }
    1446             : 
    1447             : /**
    1448             :  * Add null object to array.
    1449             :  *
    1450             :  * @since GDAL 3.8
    1451             :  */
    1452        2363 : void CPLJSONArray::AddNull()
    1453             : {
    1454        2363 :     if (m_poJsonObject)
    1455        2363 :         json_object_array_add(TO_JSONOBJ(m_poJsonObject), nullptr);
    1456        2363 : }
    1457             : 
    1458             : /**
    1459             :  * Add json object to array.
    1460             :  * @param oValue Json array.
    1461             :  *
    1462             :  * @since GDAL 2.3
    1463             :  */
    1464        5673 : void CPLJSONArray::Add(const CPLJSONObject &oValue)
    1465             : {
    1466        5673 :     if (m_poJsonObject && oValue.m_poJsonObject)
    1467       11346 :         json_object_array_add(
    1468        5673 :             TO_JSONOBJ(m_poJsonObject),
    1469        5673 :             json_object_get(TO_JSONOBJ(oValue.m_poJsonObject)));
    1470        5673 : }
    1471             : 
    1472             : /**
    1473             :  * Add value to array
    1474             :  * @param osValue Value to add.
    1475             :  *
    1476             :  * @since GDAL 2.3
    1477             :  */
    1478        7163 : void CPLJSONArray::Add(const std::string &osValue)
    1479             : {
    1480        7163 :     if (m_poJsonObject)
    1481        7163 :         json_object_array_add(TO_JSONOBJ(m_poJsonObject),
    1482             :                               json_object_new_string(osValue.c_str()));
    1483        7163 : }
    1484             : 
    1485             : /**
    1486             :  * Add value to array
    1487             :  * @param pszValue Value to add.
    1488             :  *
    1489             :  * @since GDAL 2.3
    1490             :  */
    1491        2482 : void CPLJSONArray::Add(const char *pszValue)
    1492             : {
    1493        2482 :     if (nullptr == pszValue)
    1494           0 :         return;
    1495        2482 :     if (m_poJsonObject)
    1496        2482 :         json_object_array_add(TO_JSONOBJ(m_poJsonObject),
    1497             :                               json_object_new_string(pszValue));
    1498             : }
    1499             : 
    1500             : /**
    1501             :  * Add value to array
    1502             :  * @param dfValue Value to add.
    1503             :  *
    1504             :  * @since GDAL 2.3
    1505             :  */
    1506        2251 : void CPLJSONArray::Add(double dfValue)
    1507             : {
    1508        2251 :     if (m_poJsonObject)
    1509        2251 :         json_object_array_add(TO_JSONOBJ(m_poJsonObject),
    1510             :                               json_object_new_double(dfValue));
    1511        2251 : }
    1512             : 
    1513             : /**
    1514             :  * Add value to array
    1515             :  * @param nValue Value to add.
    1516             :  *
    1517             :  * @since GDAL 2.3
    1518             :  */
    1519         600 : void CPLJSONArray::Add(int nValue)
    1520             : {
    1521         600 :     if (m_poJsonObject)
    1522         600 :         json_object_array_add(TO_JSONOBJ(m_poJsonObject),
    1523             :                               json_object_new_int(nValue));
    1524         600 : }
    1525             : 
    1526             : /**
    1527             :  * Add value to array
    1528             :  * @param nValue Value to add.
    1529             :  *
    1530             :  * @since GDAL 2.3
    1531             :  */
    1532        1458 : void CPLJSONArray::Add(GInt64 nValue)
    1533             : {
    1534        1458 :     if (m_poJsonObject)
    1535        1458 :         json_object_array_add(TO_JSONOBJ(m_poJsonObject),
    1536             :                               json_object_new_int64(nValue));
    1537        1458 : }
    1538             : 
    1539             : /**
    1540             :  * Add value to array
    1541             :  * @param nValue Value to add.
    1542             :  *
    1543             :  * @since GDAL 3.8
    1544             :  */
    1545          28 : void CPLJSONArray::Add(uint64_t nValue)
    1546             : {
    1547          28 :     if (m_poJsonObject)
    1548             :     {
    1549          28 :         json_object_array_add(TO_JSONOBJ(m_poJsonObject),
    1550             :                               json_object_new_uint64(nValue));
    1551             :     }
    1552          28 : }
    1553             : 
    1554             : /**
    1555             :  * Add value to array
    1556             :  * @param bValue Value to add.
    1557             :  *
    1558             :  * @since GDAL 2.3
    1559             :  */
    1560          18 : void CPLJSONArray::Add(bool bValue)
    1561             : {
    1562          18 :     if (m_poJsonObject)
    1563          18 :         json_object_array_add(TO_JSONOBJ(m_poJsonObject),
    1564             :                               json_object_new_boolean(bValue));
    1565          18 : }
    1566             : 
    1567             : /**
    1568             :  * Get array item by index.
    1569             :  * @param  nIndex Item index.
    1570             :  * @return        Json object.
    1571             :  *
    1572             :  * @since GDAL 2.3
    1573             :  */
    1574        2956 : CPLJSONObject CPLJSONArray::operator[](int nIndex)
    1575             : {
    1576             :     return CPLJSONObject(
    1577             :         CPLSPrintf("id:%d", nIndex),
    1578        2956 :         json_object_array_get_idx(TO_JSONOBJ(m_poJsonObject), nIndex));
    1579             : }
    1580             : 
    1581             : /**
    1582             :  * Get array const item by index.
    1583             :  * @param  nIndex Item index.
    1584             :  * @return        Json object.
    1585             :  *
    1586             :  * @since GDAL 2.3
    1587             :  */
    1588      300085 : const CPLJSONObject CPLJSONArray::operator[](int nIndex) const
    1589             : {
    1590             :     return CPLJSONObject(
    1591             :         CPLSPrintf("id:%d", nIndex),
    1592      300085 :         json_object_array_get_idx(TO_JSONOBJ(m_poJsonObject), nIndex));
    1593             : }
    1594             : 
    1595             : /************************************************************************/
    1596             : /*                      CPLParseKeyValueJson()                          */
    1597             : /************************************************************************/
    1598             : 
    1599             : /** Return a string list of key/value pairs extracted from a JSON doc.
    1600             : 
    1601             :     We are expecting simple documents with key:value pairs, like the
    1602             :     following with no hierarchy or complex structure.
    1603             :     \verbatim
    1604             :     {
    1605             :       "Code" : "Success",
    1606             :       "LastUpdated" : "2017-07-03T16:20:17Z",
    1607             :       "Type" : "AWS-HMAC",
    1608             :       "AccessKeyId" : "bla",
    1609             :       "SecretAccessKey" : "bla",
    1610             :       "Token" : "bla",
    1611             :       "Expiration" : "2017-07-03T22:42:58Z"
    1612             :     }
    1613             :     \endverbatim
    1614             :     @since GDAL 3.7
    1615             :  */
    1616          37 : CPLStringList CPLParseKeyValueJson(const char *pszJson)
    1617             : {
    1618          74 :     CPLJSONDocument oDoc;
    1619          37 :     CPLStringList oNameValue;
    1620          37 :     if (pszJson != nullptr && oDoc.LoadMemory(pszJson))
    1621             :     {
    1622         134 :         for (const auto &obj : oDoc.GetRoot().GetChildren())
    1623             :         {
    1624          97 :             const auto eType = obj.GetType();
    1625          97 :             if (eType == CPLJSONObject::Type::String ||
    1626           0 :                 eType == CPLJSONObject::Type::Integer ||
    1627             :                 eType == CPLJSONObject::Type::Double)
    1628             :             {
    1629         194 :                 oNameValue.SetNameValue(obj.GetName().c_str(),
    1630         291 :                                         obj.ToString().c_str());
    1631             :             }
    1632             :         }
    1633             :     }
    1634          74 :     return oNameValue;
    1635             : }

Generated by: LCOV version 1.14