LCOV - code coverage report
Current view: top level - port - cplstring.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 148 156 94.9 %
Date: 2025-01-18 12:42:00 Functions: 18 19 94.7 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  GDAL
       4             :  * Purpose:  CPLString implementation.
       5             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2005, Frank Warmerdam <warmerdam@pobox.com>
       9             :  * Copyright (c) 2011, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "cpl_port.h"
      15             : #include "cpl_string.h"
      16             : 
      17             : #include <cctype>
      18             : #include <cstdarg>
      19             : #include <cstddef>
      20             : #include <cstring>
      21             : #include <string>
      22             : 
      23             : #include "cpl_config.h"
      24             : #include "cpl_conv.h"
      25             : 
      26             : #if !defined(va_copy) && defined(__va_copy)
      27             : #define va_copy __va_copy
      28             : #endif
      29             : 
      30             : /*
      31             :  * The CPLString class is derived from std::string, so the vast majority
      32             :  * of the implementation comes from that.  This module is just the extensions
      33             :  * we add.
      34             :  */
      35             : 
      36             : /************************************************************************/
      37             : /*                               Printf()                               */
      38             : /************************************************************************/
      39             : 
      40             : /** Assign the content of the string using sprintf() */
      41      480085 : CPLString &CPLString::Printf(CPL_FORMAT_STRING(const char *pszFormat), ...)
      42             : 
      43             : {
      44             :     va_list args;
      45             : 
      46      480085 :     va_start(args, pszFormat);
      47      480085 :     vPrintf(pszFormat, args);
      48      480042 :     va_end(args);
      49             : 
      50      480042 :     return *this;
      51             : }
      52             : 
      53             : /************************************************************************/
      54             : /*                              vPrintf()                               */
      55             : /************************************************************************/
      56             : 
      57             : /** Assign the content of the string using vsprintf() */
      58      601139 : CPLString &CPLString::vPrintf(CPL_FORMAT_STRING(const char *pszFormat),
      59             :                               va_list args)
      60             : 
      61             : {
      62             :     /* -------------------------------------------------------------------- */
      63             :     /*      This implementation for platforms without vsnprintf() will      */
      64             :     /*      just plain fail if the formatted contents are too large.        */
      65             :     /* -------------------------------------------------------------------- */
      66             : 
      67             : #if !defined(HAVE_VSNPRINTF)
      68             :     char *pszBuffer = static_cast<char *>(CPLMalloc(30000));
      69             :     if (CPLvsnprintf(pszBuffer, 30000, pszFormat, args) > 29998)
      70             :     {
      71             :         CPLError(CE_Fatal, CPLE_AppDefined,
      72             :                  "CPLString::vPrintf() ... buffer overrun.");
      73             :     }
      74             :     *this = pszBuffer;
      75             :     CPLFree(pszBuffer);
      76             : 
      77             : /* -------------------------------------------------------------------- */
      78             : /*      This should grow a big enough buffer to hold any formatted      */
      79             : /*      result.                                                         */
      80             : /* -------------------------------------------------------------------- */
      81             : #else
      82             :     va_list wrk_args;
      83             : 
      84             : #ifdef va_copy
      85      601139 :     va_copy(wrk_args, args);
      86             : #else
      87             :     wrk_args = args;
      88             : #endif
      89             : 
      90      601139 :     char szModestBuffer[500] = {};
      91      601139 :     szModestBuffer[0] = '\0';
      92      601139 :     int nPR = CPLvsnprintf(szModestBuffer, sizeof(szModestBuffer), pszFormat,
      93             :                            wrk_args);
      94      601097 :     if (nPR == -1 || nPR >= static_cast<int>(sizeof(szModestBuffer)) - 1)
      95             :     {
      96        1166 :         int nWorkBufferSize = 2000;
      97        1166 :         char *pszWorkBuffer = static_cast<char *>(CPLMalloc(nWorkBufferSize));
      98             : 
      99             : #ifdef va_copy
     100        1176 :         va_end(wrk_args);
     101        1176 :         va_copy(wrk_args, args);
     102             : #else
     103             :         wrk_args = args;
     104             : #endif
     105        1342 :         while ((nPR = CPLvsnprintf(pszWorkBuffer, nWorkBufferSize, pszFormat,
     106        1342 :                                    wrk_args)) >= nWorkBufferSize - 1 ||
     107             :                nPR == -1)
     108             :         {
     109         166 :             nWorkBufferSize *= 4;
     110             :             pszWorkBuffer =
     111         166 :                 static_cast<char *>(CPLRealloc(pszWorkBuffer, nWorkBufferSize));
     112             : #ifdef va_copy
     113         166 :             va_end(wrk_args);
     114         166 :             va_copy(wrk_args, args);
     115             : #else
     116             :             wrk_args = args;
     117             : #endif
     118             :         }
     119        1176 :         *this = pszWorkBuffer;
     120        1176 :         CPLFree(pszWorkBuffer);
     121             :     }
     122             :     else
     123             :     {
     124      599931 :         *this = szModestBuffer;
     125             :     }
     126             : #ifdef va_copy
     127      601110 :     va_end(wrk_args);
     128             : #endif
     129             : 
     130             : #endif /* !defined(HAVE_VSNPRINTF) */
     131             : 
     132      601110 :     return *this;
     133             : }
     134             : 
     135             : /************************************************************************/
     136             : /*                              FormatC()                               */
     137             : /************************************************************************/
     138             : 
     139             : /**
     140             :  * Format double in C locale.
     141             :  *
     142             :  * The passed value is formatted using the C locale (period as decimal
     143             :  * separator) and appended to the target CPLString.
     144             :  *
     145             :  * @param dfValue the value to format.
     146             :  * @param pszFormat the sprintf() style format to use or omit for default.
     147             :  * Note that this format string should only include one substitution argument
     148             :  * and it must be for a double (%f or %g).
     149             :  *
     150             :  * @return a reference to the CPLString.
     151             :  */
     152             : 
     153        4122 : CPLString &CPLString::FormatC(double dfValue, const char *pszFormat)
     154             : 
     155             : {
     156        4122 :     if (pszFormat == nullptr)
     157        3508 :         pszFormat = "%g";
     158             : 
     159             :     // presumably long enough for any number.
     160        4122 :     const size_t buf_size = 512;
     161        4122 :     char szWork[buf_size] = {};
     162             : 
     163        4122 :     CPLsnprintf(szWork, buf_size, pszFormat, dfValue);
     164             : 
     165        4122 :     *this += szWork;
     166             : 
     167        4122 :     return *this;
     168             : }
     169             : 
     170             : /************************************************************************/
     171             : /*                                Trim()                                */
     172             : /************************************************************************/
     173             : 
     174             : /**
     175             :  * Trim white space.
     176             :  *
     177             :  * Trims white space off the let and right of the string.  White space
     178             :  * is any of a space, a tab, a newline ('\\n') or a carriage control ('\\r').
     179             :  *
     180             :  * @return a reference to the CPLString.
     181             :  */
     182             : 
     183      245289 : CPLString &CPLString::Trim()
     184             : 
     185             : {
     186      245289 :     constexpr char szWhitespace[] = " \t\r\n";
     187             : 
     188      245289 :     const size_t iLeft = find_first_not_of(szWhitespace);
     189      245289 :     const size_t iRight = find_last_not_of(szWhitespace);
     190             : 
     191      245289 :     if (iLeft == std::string::npos)
     192             :     {
     193       12668 :         erase();
     194       12668 :         return *this;
     195             :     }
     196             : 
     197      232621 :     assign(substr(iLeft, iRight - iLeft + 1));
     198             : 
     199      232621 :     return *this;
     200             : }
     201             : 
     202             : /************************************************************************/
     203             : /*                               Recode()                               */
     204             : /************************************************************************/
     205             : 
     206             : /** Recode the string */
     207        3488 : CPLString &CPLString::Recode(const char *pszSrcEncoding,
     208             :                              const char *pszDstEncoding)
     209             : 
     210             : {
     211        3488 :     if (pszSrcEncoding == nullptr)
     212           0 :         pszSrcEncoding = CPL_ENC_UTF8;
     213        3488 :     if (pszDstEncoding == nullptr)
     214           0 :         pszDstEncoding = CPL_ENC_UTF8;
     215             : 
     216        3488 :     if (strcmp(pszSrcEncoding, pszDstEncoding) == 0)
     217          40 :         return *this;
     218             : 
     219        3448 :     char *pszRecoded = CPLRecode(c_str(), pszSrcEncoding, pszDstEncoding);
     220             : 
     221        3448 :     if (pszRecoded == nullptr)
     222           0 :         return *this;
     223             : 
     224        3448 :     assign(pszRecoded);
     225        3448 :     CPLFree(pszRecoded);
     226             : 
     227        3448 :     return *this;
     228             : }
     229             : 
     230             : /************************************************************************/
     231             : /*                               ifind()                                */
     232             : /************************************************************************/
     233             : 
     234             : /**
     235             :  * Case insensitive find() alternative.
     236             :  *
     237             :  * @param str substring to find.
     238             :  * @param pos offset in the string at which the search starts.
     239             :  * @return the position of substring in the string or std::string::npos if not
     240             :  * found.
     241             :  * @since GDAL 1.9.0
     242             :  */
     243             : 
     244        7105 : size_t CPLString::ifind(const std::string &str, size_t pos) const
     245             : 
     246             : {
     247        7105 :     return ifind(str.c_str(), pos);
     248             : }
     249             : 
     250             : /**
     251             :  * Case insensitive find() alternative.
     252             :  *
     253             :  * @param s substring to find.
     254             :  * @param nPos offset in the string at which the search starts.
     255             :  * @return the position of the substring in the string or std::string::npos if
     256             :  * not found.
     257             :  * @since GDAL 1.9.0
     258             :  */
     259             : 
     260      213621 : size_t CPLString::ifind(const char *s, size_t nPos) const
     261             : 
     262             : {
     263      213621 :     const char *pszHaystack = c_str();
     264             :     const char chFirst =
     265      213612 :         static_cast<char>(CPLTolower(static_cast<unsigned char>(s[0])));
     266      213614 :     const size_t nTargetLen = strlen(s);
     267             : 
     268      213614 :     if (nPos > size())
     269           0 :         nPos = size();
     270             : 
     271      213605 :     pszHaystack += nPos;
     272             : 
     273     7863130 :     while (*pszHaystack != '\0')
     274             :     {
     275     7658670 :         if (chFirst == CPLTolower(static_cast<unsigned char>(*pszHaystack)))
     276             :         {
     277      331095 :             if (EQUALN(pszHaystack, s, nTargetLen))
     278        9140 :                 return nPos;
     279             :         }
     280             : 
     281     7649530 :         nPos++;
     282     7649530 :         pszHaystack++;
     283             :     }
     284             : 
     285      204461 :     return std::string::npos;
     286             : }
     287             : 
     288             : /************************************************************************/
     289             : /*                              toupper()                               */
     290             : /************************************************************************/
     291             : 
     292             : /**
     293             :  * Convert to upper case in place.
     294             :  */
     295             : 
     296     2910810 : CPLString &CPLString::toupper()
     297             : 
     298             : {
     299    25302400 :     for (size_t i = 0; i < size(); i++)
     300    22391600 :         (*this)[i] = static_cast<char>(CPLToupper((*this)[i]));
     301             : 
     302     2910810 :     return *this;
     303             : }
     304             : 
     305             : /************************************************************************/
     306             : /*                              tolower()                               */
     307             : /************************************************************************/
     308             : 
     309             : /**
     310             :  * Convert to lower case in place.
     311             :  */
     312             : 
     313       92992 : CPLString &CPLString::tolower()
     314             : 
     315             : {
     316     1607760 :     for (size_t i = 0; i < size(); i++)
     317     1514780 :         (*this)[i] = static_cast<char>(CPLTolower((*this)[i]));
     318             : 
     319       92992 :     return *this;
     320             : }
     321             : 
     322             : /************************************************************************/
     323             : /*                             replaceAll()                             */
     324             : /************************************************************************/
     325             : 
     326             : /**
     327             :  * Replace all occurrences of osBefore with osAfter.
     328             :  */
     329     1048730 : CPLString &CPLString::replaceAll(const std::string &osBefore,
     330             :                                  const std::string &osAfter)
     331             : {
     332     1048730 :     const size_t nBeforeSize = osBefore.size();
     333     1048720 :     const size_t nAfterSize = osAfter.size();
     334     1048710 :     if (nBeforeSize)
     335             :     {
     336     1048710 :         size_t nStartPos = 0;
     337     1131420 :         while ((nStartPos = find(osBefore, nStartPos)) != std::string::npos)
     338             :         {
     339       82714 :             replace(nStartPos, nBeforeSize, osAfter);
     340       82710 :             nStartPos += nAfterSize;
     341             :         }
     342             :     }
     343     1048720 :     return *this;
     344             : }
     345             : 
     346             : /**
     347             :  * Replace all occurrences of chBefore with osAfter.
     348             :  */
     349        4672 : CPLString &CPLString::replaceAll(char chBefore, const std::string &osAfter)
     350             : {
     351        4672 :     return replaceAll(std::string(&chBefore, 1), osAfter);
     352             : }
     353             : 
     354             : /**
     355             :  * Replace all occurrences of osBefore with chAfter.
     356             :  */
     357     1019450 : CPLString &CPLString::replaceAll(const std::string &osBefore, char chAfter)
     358             : {
     359     1019450 :     return replaceAll(osBefore, std::string(&chAfter, 1));
     360             : }
     361             : 
     362             : /**
     363             :  * Replace all occurrences of chBefore with chAfter.
     364             :  */
     365        3748 : CPLString &CPLString::replaceAll(char chBefore, char chAfter)
     366             : {
     367        3748 :     return replaceAll(std::string(&chBefore, 1), std::string(&chAfter, 1));
     368             : }
     369             : 
     370             : /************************************************************************/
     371             : /*                             endsWith()                              */
     372             : /************************************************************************/
     373             : 
     374             : /**
     375             :  * Returns whether the string ends with another string
     376             :  * @param osStr other string.
     377             :  * @return true if the string ends with osStr.
     378             :  */
     379       20574 : bool CPLString::endsWith(const std::string &osStr) const
     380             : {
     381       20574 :     if (size() < osStr.size())
     382         150 :         return false;
     383       20424 :     return substr(size() - osStr.size()) == osStr;
     384             : }
     385             : 
     386             : /************************************************************************/
     387             : /*                         CPLURLGetValue()                             */
     388             : /************************************************************************/
     389             : 
     390             : /**
     391             :  * Return the value matching a key from a key=value pair in a URL.
     392             :  *
     393             :  * @param pszURL the URL.
     394             :  * @param pszKey the key to find.
     395             :  * @return the value of empty string if not found.
     396             :  * @since GDAL 1.9.0
     397             :  */
     398        1628 : CPLString CPLURLGetValue(const char *pszURL, const char *pszKey)
     399             : {
     400        3256 :     CPLString osKey(pszKey);
     401        1628 :     osKey += "=";
     402        1628 :     size_t nKeyPos = CPLString(pszURL).ifind(osKey);
     403        1628 :     if (nKeyPos != std::string::npos && nKeyPos > 0 &&
     404         328 :         (pszURL[nKeyPos - 1] == '?' || pszURL[nKeyPos - 1] == '&'))
     405             :     {
     406         656 :         CPLString osValue(pszURL + nKeyPos + osKey.size());
     407         328 :         const char *pszValue = osValue.c_str();
     408         328 :         const char *pszSep = strchr(pszValue, '&');
     409         328 :         if (pszSep)
     410             :         {
     411          95 :             osValue.resize(pszSep - pszValue);
     412             :         }
     413         328 :         return osValue;
     414             :     }
     415        1300 :     return "";
     416             : }
     417             : 
     418             : /************************************************************************/
     419             : /*                          CPLURLAddKVP()                              */
     420             : /************************************************************************/
     421             : 
     422             : /**
     423             :  * Return a new URL with a new key=value pair.
     424             :  *
     425             :  * @param pszURL the URL.
     426             :  * @param pszKey the key to find.
     427             :  * @param pszValue the value of the key (may be NULL to unset an existing KVP).
     428             :  * @return the modified URL.
     429             :  * @since GDAL 1.9.0
     430             :  */
     431        5112 : CPLString CPLURLAddKVP(const char *pszURL, const char *pszKey,
     432             :                        const char *pszValue)
     433             : {
     434        5112 :     const CPLString osURL(strchr(pszURL, '?') == nullptr
     435       14684 :                               ? CPLString(pszURL).append("?")
     436       10224 :                               : pszURL);
     437             : 
     438       10224 :     CPLString osKey(pszKey);
     439        5112 :     osKey += "=";
     440        5112 :     size_t nKeyPos = osURL.ifind(osKey);
     441        5191 :     if (nKeyPos != std::string::npos && nKeyPos > 0 &&
     442          79 :         (osURL[nKeyPos - 1] == '?' || osURL[nKeyPos - 1] == '&'))
     443             :     {
     444         150 :         CPLString osNewURL(osURL);
     445          75 :         osNewURL.resize(nKeyPos);
     446          75 :         if (pszValue)
     447             :         {
     448          44 :             osNewURL += osKey;
     449          44 :             osNewURL += pszValue;
     450             :         }
     451          75 :         const char *pszNext = strchr(osURL.c_str() + nKeyPos, '&');
     452          75 :         if (pszNext)
     453             :         {
     454          72 :             if (osNewURL.back() == '&' || osNewURL.back() == '?')
     455          30 :                 osNewURL += pszNext + 1;
     456             :             else
     457          42 :                 osNewURL += pszNext;
     458             :         }
     459          75 :         return osNewURL;
     460             :     }
     461             :     else
     462             :     {
     463       10074 :         CPLString osNewURL(osURL);
     464        5037 :         if (pszValue)
     465             :         {
     466        3293 :             if (osNewURL.back() != '&' && osNewURL.back() != '?')
     467        2408 :                 osNewURL += '&';
     468        3293 :             osNewURL += osKey;
     469        3293 :             osNewURL += pszValue;
     470             :         }
     471        5037 :         return osNewURL;
     472             :     }
     473             : }
     474             : 
     475             : /************************************************************************/
     476             : /*                            CPLOPrintf()                              */
     477             : /************************************************************************/
     478             : 
     479             : /** Return a CPLString with the content of sprintf() */
     480       17415 : CPLString CPLOPrintf(CPL_FORMAT_STRING(const char *pszFormat), ...)
     481             : 
     482             : {
     483             :     va_list args;
     484       17415 :     va_start(args, pszFormat);
     485             : 
     486       17415 :     CPLString osTarget;
     487       17415 :     osTarget.vPrintf(pszFormat, args);
     488             : 
     489       17415 :     va_end(args);
     490             : 
     491       34830 :     return osTarget;
     492             : }
     493             : 
     494             : /************************************************************************/
     495             : /*                            CPLOvPrintf()                             */
     496             : /************************************************************************/
     497             : 
     498             : /** Return a CPLString with the content of vsprintf() */
     499           0 : CPLString CPLOvPrintf(CPL_FORMAT_STRING(const char *pszFormat), va_list args)
     500             : 
     501             : {
     502           0 :     CPLString osTarget;
     503           0 :     osTarget.vPrintf(pszFormat, args);
     504           0 :     return osTarget;
     505             : }
     506             : 
     507             : /************************************************************************/
     508             : /*                            CPLQuotedSQLIdentifer()                   */
     509             : /************************************************************************/
     510             : 
     511             : /** Return a CPLString of the SQL quoted identifier */
     512           6 : CPLString CPLQuotedSQLIdentifier(const char *pszIdent)
     513             : 
     514             : {
     515           6 :     CPLString osIdent;
     516             : 
     517           6 :     if (pszIdent)
     518             :     {
     519           6 :         char *pszQuotedIdent = CPLEscapeString(pszIdent, -1, CPLES_SQLI);
     520           6 :         osIdent.Printf("\"%s\"", pszQuotedIdent);
     521           6 :         CPLFree(pszQuotedIdent);
     522             :     }
     523             : 
     524           6 :     return osIdent;
     525             : }

Generated by: LCOV version 1.14