LCOV - code coverage report
Current view: top level - port - cpl_google_cloud.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 373 403 92.6 %
Date: 2025-01-18 12:42:00 Functions: 19 19 100.0 %

          Line data    Source code
       1             : /**********************************************************************
       2             :  * Project:  CPL - Common Portability Library
       3             :  * Purpose:  Google Cloud Storage routines
       4             :  * Author:   Even Rouault <even.rouault at spatialys.com>
       5             :  *
       6             :  **********************************************************************
       7             :  * Copyright (c) 2017, Even Rouault <even.rouault at spatialys.com>
       8             :  *
       9             :  * SPDX-License-Identifier: MIT
      10             :  ****************************************************************************/
      11             : 
      12             : #include "cpl_google_cloud.h"
      13             : #include "cpl_vsi_error.h"
      14             : #include "cpl_sha1.h"
      15             : #include "cpl_sha256.h"
      16             : #include "cpl_time.h"
      17             : #include "cpl_http.h"
      18             : #include "cpl_mem_cache.h"
      19             : #include "cpl_aws.h"
      20             : #include "cpl_json.h"
      21             : 
      22             : #include <mutex>
      23             : #include <utility>
      24             : 
      25             : #ifdef HAVE_CURL
      26             : 
      27             : static bool bFirstTimeForDebugMessage = true;
      28             : 
      29             : struct GOA2ManagerCache
      30             : {
      31             :     struct ManagerWithMutex
      32             :     {
      33             :         std::mutex oMutex{};
      34             :         GOA2Manager oManager{};
      35             : 
      36          10 :         explicit ManagerWithMutex(const GOA2Manager &oManagerIn)
      37          10 :             : oManager(oManagerIn)
      38             :         {
      39          10 :         }
      40             :     };
      41             : 
      42             :     std::mutex oMutexGOA2ManagerCache{};
      43             :     lru11::Cache<std::string, std::shared_ptr<ManagerWithMutex>>
      44             :         oGOA2ManagerCache{};
      45             : 
      46          19 :     std::string GetBearer(const GOA2Manager &oManager)
      47             :     {
      48          38 :         const std::string osKey(oManager.GetKey());
      49          19 :         std::shared_ptr<ManagerWithMutex> poSharedManager;
      50             :         {
      51          38 :             std::lock_guard oLock(oMutexGOA2ManagerCache);
      52          19 :             if (!oGOA2ManagerCache.tryGet(osKey, poSharedManager))
      53             :             {
      54          10 :                 poSharedManager = std::make_shared<ManagerWithMutex>(oManager);
      55          10 :                 oGOA2ManagerCache.insert(osKey, poSharedManager);
      56             :             }
      57             :         }
      58             :         {
      59          19 :             std::lock_guard oLock(poSharedManager->oMutex);
      60          19 :             const char *pszBearer = poSharedManager->oManager.GetBearer();
      61          38 :             return std::string(pszBearer ? pszBearer : "");
      62             :         }
      63             :     }
      64             : 
      65         313 :     void clear()
      66             :     {
      67         626 :         std::lock_guard oLock(oMutexGOA2ManagerCache);
      68         313 :         oGOA2ManagerCache.clear();
      69         313 :     }
      70             : 
      71         332 :     static GOA2ManagerCache &GetSingleton()
      72             :     {
      73         332 :         static GOA2ManagerCache goGOA2ManagerCache;
      74         332 :         return goGOA2ManagerCache;
      75             :     }
      76             : };
      77             : 
      78             : /************************************************************************/
      79             : /*                    CPLIsMachineForSureGCEInstance()                  */
      80             : /************************************************************************/
      81             : 
      82             : /** Returns whether the current machine is surely a Google Compute Engine
      83             :  * instance.
      84             :  *
      85             :  * This does a very quick check without network access.
      86             :  * Note: only works for Linux GCE instances.
      87             :  *
      88             :  * @return true if the current machine is surely a GCE instance.
      89             :  * @since GDAL 2.3
      90             :  */
      91        2550 : bool CPLIsMachineForSureGCEInstance()
      92             : {
      93        2550 :     if (CPLTestBool(CPLGetConfigOption("CPL_MACHINE_IS_GCE", "NO")))
      94             :     {
      95           0 :         return true;
      96             :     }
      97             : #ifdef __linux
      98             :     // If /sys/class/dmi/id/product_name exists, it contains "Google Compute
      99             :     // Engine"
     100        2550 :     bool bIsGCEInstance = false;
     101        2550 :     if (CPLTestBool(CPLGetConfigOption("CPL_GCE_CHECK_LOCAL_FILES", "YES")))
     102             :     {
     103           6 :         static bool bIsGCEInstanceStatic = []()
     104             :         {
     105           6 :             bool bIsGCE = false;
     106           6 :             VSILFILE *fp = VSIFOpenL("/sys/class/dmi/id/product_name", "rb");
     107           6 :             if (fp)
     108             :             {
     109           6 :                 const char *pszLine = CPLReadLineL(fp);
     110           6 :                 bIsGCE =
     111           6 :                     pszLine && STARTS_WITH_CI(pszLine, "Google Compute Engine");
     112           6 :                 VSIFCloseL(fp);
     113             :             }
     114           6 :             return bIsGCE;
     115        2543 :         }();
     116        2543 :         bIsGCEInstance = bIsGCEInstanceStatic;
     117             :     }
     118        2550 :     return bIsGCEInstance;
     119             : #else
     120             :     return false;
     121             : #endif
     122             : }
     123             : 
     124             : /************************************************************************/
     125             : /*                 CPLIsMachinePotentiallyGCEInstance()                 */
     126             : /************************************************************************/
     127             : 
     128             : /** Returns whether the current machine is potentially a Google Compute Engine
     129             :  * instance.
     130             :  *
     131             :  * This does a very quick check without network access. To confirm if the
     132             :  * machine is effectively a GCE instance, metadata.google.internal must be
     133             :  * queried.
     134             :  *
     135             :  * @return true if the current machine is potentially a GCE instance.
     136             :  * @since GDAL 2.3
     137             :  */
     138           6 : bool CPLIsMachinePotentiallyGCEInstance()
     139             : {
     140             : #ifdef __linux
     141           6 :     bool bIsMachinePotentialGCEInstance = true;
     142           6 :     if (CPLTestBool(CPLGetConfigOption("CPL_GCE_CHECK_LOCAL_FILES", "YES")))
     143             :     {
     144           1 :         bIsMachinePotentialGCEInstance = CPLIsMachineForSureGCEInstance();
     145             :     }
     146           6 :     return bIsMachinePotentialGCEInstance;
     147             : #elif defined(_WIN32)
     148             :     // We might add later a way of detecting if we run on GCE using WMI
     149             :     // See https://cloud.google.com/compute/docs/instances/managing-instances
     150             :     // For now, unconditionally try
     151             :     return true;
     152             : #else
     153             :     // At time of writing GCE instances can be only Linux or Windows
     154             :     return false;
     155             : #endif
     156             : }
     157             : 
     158             : //! @cond Doxygen_Suppress
     159             : 
     160             : /************************************************************************/
     161             : /*                            GetGSHeaders()                            */
     162             : /************************************************************************/
     163             : 
     164             : static struct curl_slist *
     165          34 : GetGSHeaders(const std::string &osPathForOption, const std::string &osVerb,
     166             :              const struct curl_slist *psExistingHeaders,
     167             :              const std::string &osCanonicalResource,
     168             :              const std::string &osSecretAccessKey,
     169             :              const std::string &osAccessKeyId, const std::string &osUserProject)
     170             : {
     171          34 :     if (osSecretAccessKey.empty())
     172             :     {
     173             :         // GS_NO_SIGN_REQUEST=YES case
     174           1 :         return nullptr;
     175             :     }
     176             : 
     177             :     std::string osDate = VSIGetPathSpecificOption(osPathForOption.c_str(),
     178          66 :                                                   "CPL_GS_TIMESTAMP", "");
     179          33 :     if (osDate.empty())
     180             :     {
     181          22 :         osDate = IVSIS3LikeHandleHelper::GetRFC822DateTime();
     182             :     }
     183             : 
     184          66 :     std::map<std::string, std::string> oSortedMapHeaders;
     185          33 :     if (!osUserProject.empty())
     186           2 :         oSortedMapHeaders["x-goog-user-project"] = osUserProject;
     187             :     std::string osCanonicalizedHeaders(
     188             :         IVSIS3LikeHandleHelper::BuildCanonicalizedHeaders(
     189          66 :             oSortedMapHeaders, psExistingHeaders, "x-goog-"));
     190             : 
     191             :     // See https://cloud.google.com/storage/docs/migrating
     192          66 :     std::string osStringToSign;
     193          33 :     osStringToSign += osVerb + "\n";
     194             :     osStringToSign +=
     195          33 :         CPLAWSGetHeaderVal(psExistingHeaders, "Content-MD5") + "\n";
     196             :     osStringToSign +=
     197          33 :         CPLAWSGetHeaderVal(psExistingHeaders, "Content-Type") + "\n";
     198          33 :     osStringToSign += osDate + "\n";
     199          33 :     osStringToSign += osCanonicalizedHeaders;
     200          33 :     osStringToSign += osCanonicalResource;
     201             : #ifdef DEBUG_VERBOSE
     202             :     CPLDebug("GS", "osStringToSign = %s", osStringToSign.c_str());
     203             : #endif
     204             : 
     205          33 :     GByte abySignature[CPL_SHA1_HASH_SIZE] = {};
     206          66 :     CPL_HMAC_SHA1(osSecretAccessKey.c_str(), osSecretAccessKey.size(),
     207          33 :                   osStringToSign.c_str(), osStringToSign.size(), abySignature);
     208             : 
     209          33 :     char *pszBase64 = CPLBase64Encode(sizeof(abySignature), abySignature);
     210          33 :     std::string osAuthorization("GOOG1 ");
     211          33 :     osAuthorization += osAccessKeyId;
     212          33 :     osAuthorization += ":";
     213          33 :     osAuthorization += pszBase64;
     214          33 :     CPLFree(pszBase64);
     215             : 
     216          33 :     struct curl_slist *headers = nullptr;
     217             :     headers =
     218          33 :         curl_slist_append(headers, CPLSPrintf("Date: %s", osDate.c_str()));
     219          33 :     headers = curl_slist_append(
     220             :         headers, CPLSPrintf("Authorization: %s", osAuthorization.c_str()));
     221          33 :     if (!osUserProject.empty())
     222             :     {
     223             :         headers =
     224           2 :             curl_slist_append(headers, CPLSPrintf("x-goog-user-project: %s",
     225             :                                                   osUserProject.c_str()));
     226             :     }
     227          33 :     return headers;
     228             : }
     229             : 
     230             : /************************************************************************/
     231             : /*                         VSIGSHandleHelper()                          */
     232             : /************************************************************************/
     233          68 : VSIGSHandleHelper::VSIGSHandleHelper(const std::string &osEndpoint,
     234             :                                      const std::string &osBucketObjectKey,
     235             :                                      const std::string &osSecretAccessKey,
     236             :                                      const std::string &osAccessKeyId,
     237             :                                      bool bUseAuthenticationHeader,
     238             :                                      const GOA2Manager &oManager,
     239          68 :                                      const std::string &osUserProject)
     240          68 :     : m_osURL(osEndpoint + CPLAWSURLEncode(osBucketObjectKey, false)),
     241             :       m_osEndpoint(osEndpoint), m_osBucketObjectKey(osBucketObjectKey),
     242             :       m_osSecretAccessKey(osSecretAccessKey), m_osAccessKeyId(osAccessKeyId),
     243             :       m_bUseAuthenticationHeader(bUseAuthenticationHeader),
     244         136 :       m_oManager(oManager), m_osUserProject(osUserProject)
     245             : {
     246          68 :     if (m_osBucketObjectKey.find('/') == std::string::npos)
     247           8 :         m_osURL += "/";
     248          68 : }
     249             : 
     250             : /************************************************************************/
     251             : /*                        ~VSIGSHandleHelper()                          */
     252             : /************************************************************************/
     253             : 
     254         136 : VSIGSHandleHelper::~VSIGSHandleHelper()
     255             : {
     256         136 : }
     257             : 
     258             : /************************************************************************/
     259             : /*                GetConfigurationFromAWSConfigFiles()                  */
     260             : /************************************************************************/
     261             : 
     262          12 : bool VSIGSHandleHelper::GetConfigurationFromConfigFile(
     263             :     std::string &osSecretAccessKey, std::string &osAccessKeyId,
     264             :     std::string &osOAuth2RefreshToken, std::string &osOAuth2ClientId,
     265             :     std::string &osOAuth2ClientSecret, std::string &osCredentials)
     266             : {
     267             : #ifdef _WIN32
     268             :     const char *pszHome = CPLGetConfigOption("USERPROFILE", nullptr);
     269             :     constexpr char SEP_STRING[] = "\\";
     270             : #else
     271          12 :     const char *pszHome = CPLGetConfigOption("HOME", nullptr);
     272          12 :     constexpr char SEP_STRING[] = "/";
     273             : #endif
     274             : 
     275             :     // GDAL specific config option (mostly for testing purpose, but also
     276             :     // used in production in some cases)
     277             :     const char *pszCredentials =
     278          12 :         CPLGetConfigOption("CPL_GS_CREDENTIALS_FILE", nullptr);
     279          12 :     if (pszCredentials)
     280             :     {
     281          12 :         osCredentials = pszCredentials;
     282             :     }
     283             :     else
     284             :     {
     285           0 :         osCredentials = pszHome ? pszHome : "";
     286           0 :         osCredentials += SEP_STRING;
     287           0 :         osCredentials += ".boto";
     288             :     }
     289             : 
     290          12 :     VSILFILE *fp = VSIFOpenL(osCredentials.c_str(), "rb");
     291          12 :     if (fp != nullptr)
     292             :     {
     293             :         const char *pszLine;
     294           3 :         bool bInCredentials = false;
     295           3 :         bool bInOAuth2 = false;
     296          25 :         while ((pszLine = CPLReadLineL(fp)) != nullptr)
     297             :         {
     298          22 :             if (pszLine[0] == '[')
     299             :             {
     300           7 :                 bInCredentials = false;
     301           7 :                 bInOAuth2 = false;
     302             : 
     303           7 :                 if (std::string(pszLine) == "[Credentials]")
     304           3 :                     bInCredentials = true;
     305           4 :                 else if (std::string(pszLine) == "[OAuth2]")
     306           2 :                     bInOAuth2 = true;
     307             :             }
     308          15 :             else if (bInCredentials)
     309             :             {
     310           4 :                 char *pszKey = nullptr;
     311           4 :                 const char *pszValue = CPLParseNameValue(pszLine, &pszKey);
     312           4 :                 if (pszKey && pszValue)
     313             :                 {
     314           4 :                     if (EQUAL(pszKey, "gs_access_key_id"))
     315           1 :                         osAccessKeyId = CPLString(pszValue).Trim();
     316           3 :                     else if (EQUAL(pszKey, "gs_secret_access_key"))
     317           1 :                         osSecretAccessKey = CPLString(pszValue).Trim();
     318           2 :                     else if (EQUAL(pszKey, "gs_oauth2_refresh_token"))
     319           2 :                         osOAuth2RefreshToken = CPLString(pszValue).Trim();
     320             :                 }
     321           4 :                 CPLFree(pszKey);
     322             :             }
     323          11 :             else if (bInOAuth2)
     324             :             {
     325           4 :                 char *pszKey = nullptr;
     326           4 :                 const char *pszValue = CPLParseNameValue(pszLine, &pszKey);
     327           4 :                 if (pszKey && pszValue)
     328             :                 {
     329           4 :                     if (EQUAL(pszKey, "client_id"))
     330           2 :                         osOAuth2ClientId = CPLString(pszValue).Trim();
     331           2 :                     else if (EQUAL(pszKey, "client_secret"))
     332           2 :                         osOAuth2ClientSecret = CPLString(pszValue).Trim();
     333             :                 }
     334           4 :                 CPLFree(pszKey);
     335             :             }
     336             :         }
     337           3 :         VSIFCloseL(fp);
     338             :     }
     339             : 
     340          23 :     return (!osAccessKeyId.empty() && !osSecretAccessKey.empty()) ||
     341          23 :            !osOAuth2RefreshToken.empty();
     342             : }
     343             : 
     344             : /************************************************************************/
     345             : /*                        GetConfiguration()                            */
     346             : /************************************************************************/
     347             : 
     348          74 : bool VSIGSHandleHelper::GetConfiguration(const std::string &osPathForOption,
     349             :                                          CSLConstList papszOptions,
     350             :                                          std::string &osSecretAccessKey,
     351             :                                          std::string &osAccessKeyId,
     352             :                                          bool &bUseAuthenticationHeader,
     353             :                                          GOA2Manager &oManager)
     354             : {
     355          74 :     osSecretAccessKey.clear();
     356          74 :     osAccessKeyId.clear();
     357          74 :     bUseAuthenticationHeader = false;
     358             : 
     359          74 :     if (CPLTestBool(VSIGetPathSpecificOption(osPathForOption.c_str(),
     360             :                                              "GS_NO_SIGN_REQUEST", "NO")))
     361             :     {
     362           5 :         return true;
     363             :     }
     364             : 
     365             :     osSecretAccessKey = VSIGetPathSpecificOption(osPathForOption.c_str(),
     366          69 :                                                  "GS_SECRET_ACCESS_KEY", "");
     367          69 :     if (!osSecretAccessKey.empty())
     368             :     {
     369             :         osAccessKeyId = VSIGetPathSpecificOption(osPathForOption.c_str(),
     370          45 :                                                  "GS_ACCESS_KEY_ID", "");
     371          45 :         if (osAccessKeyId.empty())
     372             :         {
     373           1 :             VSIError(VSIE_AWSInvalidCredentials,
     374             :                      "GS_ACCESS_KEY_ID configuration option not defined");
     375           1 :             bFirstTimeForDebugMessage = false;
     376           1 :             return false;
     377             :         }
     378             : 
     379          44 :         if (bFirstTimeForDebugMessage)
     380             :         {
     381           8 :             CPLDebug("GS", "Using GS_SECRET_ACCESS_KEY and "
     382             :                            "GS_ACCESS_KEY_ID configuration options");
     383             :         }
     384          44 :         bFirstTimeForDebugMessage = false;
     385          44 :         return true;
     386             :     }
     387             : 
     388             :     const std::string osHeaderFile = VSIGetPathSpecificOption(
     389          48 :         osPathForOption.c_str(), "GDAL_HTTP_HEADER_FILE", "");
     390          24 :     bool bMayWarnDidNotFindAuth = false;
     391          24 :     if (!osHeaderFile.empty())
     392             :     {
     393           2 :         bool bFoundAuth = false;
     394           2 :         VSILFILE *fp = nullptr;
     395             :         // Do not allow /vsicurl/ access from /vsicurl because of
     396             :         // GetCurlHandleFor() e.g. "/vsicurl/,HEADER_FILE=/vsicurl/,url= " would
     397             :         // cause use of memory after free
     398           2 :         if (strstr(osHeaderFile.c_str(), "/vsicurl/") == nullptr &&
     399           2 :             strstr(osHeaderFile.c_str(), "/vsicurl?") == nullptr &&
     400           2 :             strstr(osHeaderFile.c_str(), "/vsis3/") == nullptr &&
     401           2 :             strstr(osHeaderFile.c_str(), "/vsigs/") == nullptr &&
     402           2 :             strstr(osHeaderFile.c_str(), "/vsiaz/") == nullptr &&
     403           6 :             strstr(osHeaderFile.c_str(), "/vsioss/") == nullptr &&
     404           2 :             strstr(osHeaderFile.c_str(), "/vsiswift/") == nullptr)
     405             :         {
     406           2 :             fp = VSIFOpenL(osHeaderFile.c_str(), "rb");
     407             :         }
     408           2 :         if (fp == nullptr)
     409             :         {
     410           1 :             CPLError(CE_Failure, CPLE_FileIO, "Cannot read %s",
     411             :                      osHeaderFile.c_str());
     412             :         }
     413             :         else
     414             :         {
     415           1 :             const char *pszLine = nullptr;
     416        1862 :             while ((pszLine = CPLReadLineL(fp)) != nullptr)
     417             :             {
     418        1861 :                 if (STARTS_WITH_CI(pszLine, "Authorization:"))
     419             :                 {
     420           0 :                     bFoundAuth = true;
     421           0 :                     break;
     422             :                 }
     423             :             }
     424           1 :             VSIFCloseL(fp);
     425           1 :             if (!bFoundAuth)
     426           1 :                 bMayWarnDidNotFindAuth = true;
     427             :         }
     428           2 :         if (bFoundAuth)
     429             :         {
     430           0 :             if (bFirstTimeForDebugMessage)
     431             :             {
     432           0 :                 CPLDebug("GS", "Using GDAL_HTTP_HEADER_FILE=%s",
     433             :                          osHeaderFile.c_str());
     434             :             }
     435           0 :             bFirstTimeForDebugMessage = false;
     436           0 :             bUseAuthenticationHeader = true;
     437           0 :             return true;
     438             :         }
     439             :     }
     440             : 
     441          24 :     const char *pszHeaders = VSIGetPathSpecificOption(
     442             :         osPathForOption.c_str(), "GDAL_HTTP_HEADERS", nullptr);
     443          24 :     if (pszHeaders && strstr(pszHeaders, "Authorization:") != nullptr)
     444             :     {
     445           1 :         bUseAuthenticationHeader = true;
     446           1 :         return true;
     447             :     }
     448             : 
     449             :     std::string osRefreshToken(VSIGetPathSpecificOption(
     450          46 :         osPathForOption.c_str(), "GS_OAUTH2_REFRESH_TOKEN", ""));
     451          23 :     if (!osRefreshToken.empty())
     452             :     {
     453             :         std::string osClientId = VSIGetPathSpecificOption(
     454           8 :             osPathForOption.c_str(), "GS_OAUTH2_CLIENT_ID", "");
     455             :         std::string osClientSecret = VSIGetPathSpecificOption(
     456           8 :             osPathForOption.c_str(), "GS_OAUTH2_CLIENT_SECRET", "");
     457             : 
     458             :         int nCount =
     459           4 :             (!osClientId.empty() ? 1 : 0) + (!osClientSecret.empty() ? 1 : 0);
     460           4 :         if (nCount == 1)
     461             :         {
     462           0 :             CPLError(CE_Failure, CPLE_NotSupported,
     463             :                      "Either both or none of GS_OAUTH2_CLIENT_ID and "
     464             :                      "GS_OAUTH2_CLIENT_SECRET must be set");
     465           0 :             return false;
     466             :         }
     467             : 
     468           4 :         if (bFirstTimeForDebugMessage)
     469             :         {
     470             :             std::string osMsg(
     471           4 :                 "Using GS_OAUTH2_REFRESH_TOKEN configuration option");
     472           2 :             if (osClientId.empty())
     473           1 :                 osMsg += " and GDAL default client_id/client_secret";
     474             :             else
     475           1 :                 osMsg += " and GS_OAUTH2_CLIENT_ID and GS_OAUTH2_CLIENT_SECRET";
     476           2 :             CPLDebug("GS", "%s", osMsg.c_str());
     477             :         }
     478           4 :         bFirstTimeForDebugMessage = false;
     479             : 
     480           4 :         return oManager.SetAuthFromRefreshToken(
     481             :             osRefreshToken.c_str(), osClientId.c_str(), osClientSecret.c_str(),
     482           4 :             nullptr);
     483             :     }
     484             : 
     485             :     std::string osJsonFile(CSLFetchNameValueDef(
     486             :         papszOptions, "GOOGLE_APPLICATION_CREDENTIALS",
     487             :         VSIGetPathSpecificOption(osPathForOption.c_str(),
     488          38 :                                  "GOOGLE_APPLICATION_CREDENTIALS", "")));
     489          19 :     if (!osJsonFile.empty())
     490             :     {
     491          10 :         CPLJSONDocument oDoc;
     492           5 :         if (!oDoc.Load(osJsonFile))
     493             :         {
     494           0 :             return false;
     495             :         }
     496             : 
     497             :         // JSON file can be of type 'service_account' or 'authorized_user'
     498          15 :         std::string osJsonFileType = oDoc.GetRoot().GetString("type");
     499             : 
     500           5 :         if (strcmp(osJsonFileType.c_str(), "service_account") == 0)
     501             :         {
     502           9 :             CPLString osPrivateKey = oDoc.GetRoot().GetString("private_key");
     503           6 :             osPrivateKey.replaceAll("\\n", "\n")
     504           6 :                 .replaceAll("\n\n", "\n")
     505           3 :                 .replaceAll("\r", "");
     506             :             std::string osClientEmail =
     507           9 :                 oDoc.GetRoot().GetString("client_email");
     508           3 :             const char *pszScope = CSLFetchNameValueDef(
     509             :                 papszOptions, "GS_OAUTH2_SCOPE",
     510             :                 VSIGetPathSpecificOption(
     511             :                     osPathForOption.c_str(), "GS_OAUTH2_SCOPE",
     512             :                     "https://www.googleapis.com/auth/devstorage.read_write"));
     513             : 
     514           3 :             return oManager.SetAuthFromServiceAccount(
     515             :                 osPrivateKey.c_str(), osClientEmail.c_str(), pszScope, nullptr,
     516           3 :                 nullptr);
     517             :         }
     518           2 :         else if (strcmp(osJsonFileType.c_str(), "authorized_user") == 0)
     519             :         {
     520           6 :             std::string osClientId = oDoc.GetRoot().GetString("client_id");
     521             :             std::string osClientSecret =
     522           6 :                 oDoc.GetRoot().GetString("client_secret");
     523           2 :             osRefreshToken = oDoc.GetRoot().GetString("refresh_token");
     524             : 
     525           2 :             return oManager.SetAuthFromRefreshToken(
     526             :                 osRefreshToken.c_str(), osClientId.c_str(),
     527           2 :                 osClientSecret.c_str(), nullptr);
     528             :         }
     529           0 :         return false;
     530             :     }
     531             : 
     532             :     CPLString osPrivateKey = CSLFetchNameValueDef(
     533             :         papszOptions, "GS_OAUTH2_PRIVATE_KEY",
     534             :         VSIGetPathSpecificOption(osPathForOption.c_str(),
     535          28 :                                  "GS_OAUTH2_PRIVATE_KEY", ""));
     536             :     std::string osPrivateKeyFile = CSLFetchNameValueDef(
     537             :         papszOptions, "GS_OAUTH2_PRIVATE_KEY_FILE",
     538             :         VSIGetPathSpecificOption(osPathForOption.c_str(),
     539          28 :                                  "GS_OAUTH2_PRIVATE_KEY_FILE", ""));
     540          14 :     if (!osPrivateKey.empty() || !osPrivateKeyFile.empty())
     541             :     {
     542           2 :         if (!osPrivateKeyFile.empty())
     543             :         {
     544           1 :             VSILFILE *fp = VSIFOpenL(osPrivateKeyFile.c_str(), "rb");
     545           1 :             if (fp == nullptr)
     546             :             {
     547           0 :                 CPLError(CE_Failure, CPLE_FileIO, "Cannot open %s",
     548             :                          osPrivateKeyFile.c_str());
     549           0 :                 bFirstTimeForDebugMessage = false;
     550           0 :                 return false;
     551             :             }
     552             :             else
     553             :             {
     554           1 :                 char *pabyBuffer = static_cast<char *>(CPLMalloc(32768));
     555           1 :                 size_t nRead = VSIFReadL(pabyBuffer, 1, 32768, fp);
     556           1 :                 osPrivateKey.assign(pabyBuffer, nRead);
     557           1 :                 VSIFCloseL(fp);
     558           1 :                 CPLFree(pabyBuffer);
     559             :             }
     560             :         }
     561           4 :         osPrivateKey.replaceAll("\\n", "\n")
     562           4 :             .replaceAll("\n\n", "\n")
     563           2 :             .replaceAll("\r", "");
     564             : 
     565             :         std::string osClientEmail = CSLFetchNameValueDef(
     566             :             papszOptions, "GS_OAUTH2_CLIENT_EMAIL",
     567             :             VSIGetPathSpecificOption(osPathForOption.c_str(),
     568           4 :                                      "GS_OAUTH2_CLIENT_EMAIL", ""));
     569           2 :         if (osClientEmail.empty())
     570             :         {
     571           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     572             :                      "GS_OAUTH2_CLIENT_EMAIL not defined");
     573           0 :             bFirstTimeForDebugMessage = false;
     574           0 :             return false;
     575             :         }
     576           2 :         const char *pszScope = CSLFetchNameValueDef(
     577             :             papszOptions, "GS_OAUTH2_SCOPE",
     578             :             VSIGetPathSpecificOption(
     579             :                 osPathForOption.c_str(), "GS_OAUTH2_SCOPE",
     580             :                 "https://www.googleapis.com/auth/devstorage.read_write"));
     581             : 
     582           2 :         if (bFirstTimeForDebugMessage)
     583             :         {
     584           2 :             CPLDebug("GS",
     585             :                      "Using %s, GS_OAUTH2_CLIENT_EMAIL and GS_OAUTH2_SCOPE=%s "
     586             :                      "configuration options",
     587           2 :                      !osPrivateKeyFile.empty() ? "GS_OAUTH2_PRIVATE_KEY_FILE"
     588             :                                                : "GS_OAUTH2_PRIVATE_KEY",
     589             :                      pszScope);
     590             :         }
     591           2 :         bFirstTimeForDebugMessage = false;
     592             : 
     593           2 :         return oManager.SetAuthFromServiceAccount(osPrivateKey.c_str(),
     594             :                                                   osClientEmail.c_str(),
     595           2 :                                                   pszScope, nullptr, nullptr);
     596             :     }
     597             : 
     598             :     // Next try reading from ~/.boto
     599          24 :     std::string osCredentials;
     600          24 :     std::string osOAuth2RefreshToken;
     601          24 :     std::string osOAuth2ClientId;
     602          24 :     std::string osOAuth2ClientSecret;
     603          12 :     if (GetConfigurationFromConfigFile(osSecretAccessKey, osAccessKeyId,
     604             :                                        osOAuth2RefreshToken, osOAuth2ClientId,
     605             :                                        osOAuth2ClientSecret, osCredentials))
     606             :     {
     607           3 :         if (!osOAuth2RefreshToken.empty())
     608             :         {
     609             :             std::string osClientId =
     610           4 :                 CPLGetConfigOption("GS_OAUTH2_CLIENT_ID", "");
     611             :             std::string osClientSecret =
     612           4 :                 CPLGetConfigOption("GS_OAUTH2_CLIENT_SECRET", "");
     613           2 :             bool bClientInfoFromEnv = false;
     614           2 :             bool bClientInfoFromFile = false;
     615             : 
     616           2 :             const int nCountClientIdSecret = (!osClientId.empty() ? 1 : 0) +
     617           2 :                                              (!osClientSecret.empty() ? 1 : 0);
     618           2 :             if (nCountClientIdSecret == 1)
     619             :             {
     620           0 :                 CPLError(CE_Failure, CPLE_NotSupported,
     621             :                          "Either both or none of GS_OAUTH2_CLIENT_ID and "
     622             :                          "GS_OAUTH2_CLIENT_SECRET must be set");
     623           0 :                 return false;
     624             :             }
     625           2 :             else if (nCountClientIdSecret == 2)
     626             :             {
     627           0 :                 bClientInfoFromEnv = true;
     628             :             }
     629           2 :             else if (nCountClientIdSecret == 0)
     630             :             {
     631           2 :                 int nCountOAuth2IdSecret = (!osOAuth2ClientId.empty() ? 1 : 0);
     632           2 :                 nCountOAuth2IdSecret += (!osOAuth2ClientSecret.empty() ? 1 : 0);
     633           2 :                 if (nCountOAuth2IdSecret == 1)
     634             :                 {
     635           0 :                     CPLError(CE_Failure, CPLE_NotSupported,
     636             :                              "Either both or none of client_id and "
     637             :                              "client_secret from %s must be set",
     638             :                              osCredentials.c_str());
     639           0 :                     return false;
     640             :                 }
     641           2 :                 else if (nCountOAuth2IdSecret == 2)
     642             :                 {
     643           2 :                     osClientId = std::move(osOAuth2ClientId);
     644           2 :                     osClientSecret = std::move(osOAuth2ClientSecret);
     645           2 :                     bClientInfoFromFile = true;
     646             :                 }
     647             :             }
     648             : 
     649           2 :             if (bFirstTimeForDebugMessage)
     650             :             {
     651           2 :                 CPLString osMsg;
     652             :                 osMsg.Printf("Using gs_oauth2_refresh_token from %s",
     653           1 :                              osCredentials.c_str());
     654           1 :                 if (bClientInfoFromEnv)
     655             :                     osMsg += " and GS_OAUTH2_CLIENT_ID and "
     656           0 :                              "GS_OAUTH2_CLIENT_SECRET configuration options";
     657           1 :                 else if (bClientInfoFromFile)
     658             :                     osMsg +=
     659             :                         CPLSPrintf(" and client_id and client_secret from %s",
     660           1 :                                    osCredentials.c_str());
     661             :                 else
     662           0 :                     osMsg += " and GDAL default client_id/client_secret";
     663           1 :                 CPLDebug("GS", "%s", osMsg.c_str());
     664             :             }
     665           2 :             bFirstTimeForDebugMessage = false;
     666             : 
     667           2 :             return oManager.SetAuthFromRefreshToken(
     668             :                 osOAuth2RefreshToken.c_str(), osClientId.c_str(),
     669           2 :                 osClientSecret.c_str(), nullptr);
     670             :         }
     671             :         else
     672             :         {
     673           1 :             if (bFirstTimeForDebugMessage)
     674             :             {
     675           1 :                 CPLDebug(
     676             :                     "GS",
     677             :                     "Using gs_access_key_id and gs_secret_access_key from %s",
     678             :                     osCredentials.c_str());
     679             :             }
     680           1 :             bFirstTimeForDebugMessage = false;
     681           1 :             return true;
     682             :         }
     683             :     }
     684             : 
     685             :     // Some Travis-CI workers are GCE machines, and for some tests, we don't
     686             :     // want this code path to be taken. And on AppVeyor/Window, we would also
     687             :     // attempt a network access
     688          13 :     if (!CPLTestBool(CPLGetConfigOption("CPL_GCE_SKIP", "NO")) &&
     689           4 :         CPLIsMachinePotentiallyGCEInstance())
     690             :     {
     691           4 :         oManager.SetAuthFromGCE(nullptr);
     692             : 
     693           4 :         if (!GOA2ManagerCache::GetSingleton().GetBearer(oManager).empty())
     694             :         {
     695           4 :             CPLDebug("GS", "Using GCE inherited permissions");
     696             : 
     697           4 :             bFirstTimeForDebugMessage = false;
     698           4 :             return true;
     699             :         }
     700             :     }
     701             : 
     702           5 :     if (bMayWarnDidNotFindAuth)
     703             :     {
     704           1 :         CPLDebug("GS", "Cannot find Authorization header in %s",
     705             :                  CPLGetConfigOption("GDAL_HTTP_HEADER_FILE", ""));
     706             :     }
     707             : 
     708           5 :     CPLString osMsg;
     709             :     osMsg.Printf(
     710             :         "No valid GCS credentials found. "
     711             :         "For authenticated requests, you need to set "
     712             :         "GS_SECRET_ACCESS_KEY, GS_ACCESS_KEY_ID, GS_OAUTH2_REFRESH_TOKEN, "
     713             :         "GOOGLE_APPLICATION_CREDENTIALS, or other configuration "
     714             :         "options, or create a %s file. Consult "
     715             :         "https://gdal.org/en/stable/user/"
     716             :         "virtual_file_systems.html#vsigs-google-cloud-storage-files "
     717             :         "for more details. "
     718             :         "For unauthenticated requests on public resources, set the "
     719             :         "GS_NO_SIGN_REQUEST configuration option to YES.",
     720           5 :         osCredentials.c_str());
     721             : 
     722           5 :     CPLDebug("GS", "%s", osMsg.c_str());
     723           5 :     VSIError(VSIE_AWSInvalidCredentials, "%s", osMsg.c_str());
     724           5 :     return false;
     725             : }
     726             : 
     727             : /************************************************************************/
     728             : /*                          BuildFromURI()                              */
     729             : /************************************************************************/
     730             : 
     731          74 : VSIGSHandleHelper *VSIGSHandleHelper::BuildFromURI(
     732             :     const char *pszURI, const char * /*pszFSPrefix*/,
     733             :     const char *pszURIForPathSpecificOption, CSLConstList papszOptions)
     734             : {
     735         148 :     std::string osPathForOption("/vsigs/");
     736             :     osPathForOption +=
     737          74 :         pszURIForPathSpecificOption ? pszURIForPathSpecificOption : pszURI;
     738             : 
     739             :     // pszURI == bucket/object
     740         148 :     const std::string osBucketObject(pszURI);
     741             :     std::string osEndpoint(VSIGetPathSpecificOption(osPathForOption.c_str(),
     742         148 :                                                     "CPL_GS_ENDPOINT", ""));
     743          74 :     if (osEndpoint.empty())
     744          15 :         osEndpoint = "https://storage.googleapis.com/";
     745             : 
     746         148 :     std::string osSecretAccessKey;
     747         148 :     std::string osAccessKeyId;
     748             :     bool bUseAuthenticationHeader;
     749         148 :     GOA2Manager oManager;
     750             : 
     751          74 :     if (!GetConfiguration(osPathForOption, papszOptions, osSecretAccessKey,
     752             :                           osAccessKeyId, bUseAuthenticationHeader, oManager))
     753             :     {
     754           6 :         return nullptr;
     755             :     }
     756             : 
     757             :     // https://cloud.google.com/storage/docs/xml-api/reference-headers#xgooguserproject
     758             :     // The Project ID for an existing Google Cloud project to bill for access
     759             :     // charges associated with the request.
     760             :     const std::string osUserProject = VSIGetPathSpecificOption(
     761          68 :         osPathForOption.c_str(), "GS_USER_PROJECT", "");
     762             : 
     763             :     return new VSIGSHandleHelper(osEndpoint, osBucketObject, osSecretAccessKey,
     764             :                                  osAccessKeyId, bUseAuthenticationHeader,
     765          68 :                                  oManager, osUserProject);
     766             : }
     767             : 
     768             : /************************************************************************/
     769             : /*                           RebuildURL()                               */
     770             : /************************************************************************/
     771             : 
     772          54 : void VSIGSHandleHelper::RebuildURL()
     773             : {
     774          54 :     m_osURL = m_osEndpoint + CPLAWSURLEncode(m_osBucketObjectKey, false);
     775         106 :     if (!m_osBucketObjectKey.empty() &&
     776          52 :         m_osBucketObjectKey.find('/') == std::string::npos)
     777          24 :         m_osURL += "/";
     778          54 :     m_osURL += GetQueryString(false);
     779          54 : }
     780             : 
     781             : /************************************************************************/
     782             : /*                           UsesHMACKey()                              */
     783             : /************************************************************************/
     784             : 
     785           3 : bool VSIGSHandleHelper::UsesHMACKey() const
     786             : {
     787           3 :     return m_oManager.GetAuthMethod() == GOA2Manager::NONE;
     788             : }
     789             : 
     790             : /************************************************************************/
     791             : /*                           GetCurlHeaders()                           */
     792             : /************************************************************************/
     793             : 
     794             : struct curl_slist *
     795          50 : VSIGSHandleHelper::GetCurlHeaders(const std::string &osVerb,
     796             :                                   const struct curl_slist *psExistingHeaders,
     797             :                                   const void *, size_t) const
     798             : {
     799          50 :     if (m_bUseAuthenticationHeader)
     800           1 :         return nullptr;
     801             : 
     802          49 :     if (m_oManager.GetAuthMethod() != GOA2Manager::NONE)
     803             :     {
     804             :         const std::string osBearer =
     805          30 :             GOA2ManagerCache::GetSingleton().GetBearer(m_oManager);
     806          15 :         if (osBearer.empty())
     807           0 :             return nullptr;
     808             : 
     809          15 :         struct curl_slist *headers = nullptr;
     810          15 :         headers = curl_slist_append(
     811             :             headers, CPLSPrintf("Authorization: Bearer %s", osBearer.c_str()));
     812             : 
     813          15 :         if (!m_osUserProject.empty())
     814             :         {
     815             :             headers =
     816           1 :                 curl_slist_append(headers, CPLSPrintf("x-goog-user-project: %s",
     817             :                                                       m_osUserProject.c_str()));
     818             :         }
     819          15 :         return headers;
     820             :     }
     821             : 
     822             :     std::string osCanonicalResource(
     823          34 :         "/" + CPLAWSURLEncode(m_osBucketObjectKey, false));
     824          67 :     if (!m_osBucketObjectKey.empty() &&
     825          33 :         m_osBucketObjectKey.find('/') == std::string::npos)
     826           6 :         osCanonicalResource += "/";
     827             :     else
     828             :     {
     829          56 :         const auto osQueryString(GetQueryString(false));
     830          28 :         if (osQueryString == "?uploads" || osQueryString == "?acl")
     831           4 :             osCanonicalResource += osQueryString;
     832             :     }
     833             : 
     834          68 :     return GetGSHeaders("/vsigs/" + m_osBucketObjectKey, osVerb,
     835             :                         psExistingHeaders, osCanonicalResource,
     836          68 :                         m_osSecretAccessKey, m_osAccessKeyId, m_osUserProject);
     837             : }
     838             : 
     839             : /************************************************************************/
     840             : /*                          ClearCache()                                */
     841             : /************************************************************************/
     842             : 
     843         313 : void VSIGSHandleHelper::ClearCache()
     844             : {
     845         313 :     GOA2ManagerCache::GetSingleton().clear();
     846         313 :     bFirstTimeForDebugMessage = true;
     847         313 : }
     848             : 
     849             : /************************************************************************/
     850             : /*                           GetSignedURL()                             */
     851             : /************************************************************************/
     852             : 
     853           5 : std::string VSIGSHandleHelper::GetSignedURL(CSLConstList papszOptions)
     854             : {
     855           8 :     if (!((!m_osAccessKeyId.empty() && !m_osSecretAccessKey.empty()) ||
     856           3 :           m_oManager.GetAuthMethod() == GOA2Manager::SERVICE_ACCOUNT))
     857             :     {
     858           2 :         CPLError(CE_Failure, CPLE_NotSupported,
     859             :                  "Signed URL for Google Cloud Storage is only available with "
     860             :                  "AWS style authentication with "
     861             :                  "GS_ACCESS_KEY_ID+GS_SECRET_ACCESS_KEY, "
     862             :                  "or with service account authentication");
     863           2 :         return std::string();
     864             :     }
     865             : 
     866           3 :     GIntBig nStartDate = static_cast<GIntBig>(time(nullptr));
     867           3 :     const char *pszStartDate = CSLFetchNameValue(papszOptions, "START_DATE");
     868           3 :     if (pszStartDate)
     869             :     {
     870             :         int nYear, nMonth, nDay, nHour, nMin, nSec;
     871           3 :         if (sscanf(pszStartDate, "%04d%02d%02dT%02d%02d%02dZ", &nYear, &nMonth,
     872           3 :                    &nDay, &nHour, &nMin, &nSec) == 6)
     873             :         {
     874             :             struct tm brokendowntime;
     875           3 :             brokendowntime.tm_year = nYear - 1900;
     876           3 :             brokendowntime.tm_mon = nMonth - 1;
     877           3 :             brokendowntime.tm_mday = nDay;
     878           3 :             brokendowntime.tm_hour = nHour;
     879           3 :             brokendowntime.tm_min = nMin;
     880           3 :             brokendowntime.tm_sec = nSec;
     881           3 :             nStartDate = CPLYMDHMSToUnixTime(&brokendowntime);
     882             :         }
     883             :     }
     884             :     GIntBig nExpiresIn =
     885             :         nStartDate +
     886           3 :         atoi(CSLFetchNameValueDef(papszOptions, "EXPIRATION_DELAY", "3600"));
     887             :     std::string osExpires(CSLFetchNameValueDef(
     888           6 :         papszOptions, "EXPIRES", CPLSPrintf(CPL_FRMT_GIB, nExpiresIn)));
     889             : 
     890           6 :     std::string osVerb(CSLFetchNameValueDef(papszOptions, "VERB", "GET"));
     891             : 
     892             :     std::string osCanonicalizedResource(
     893           6 :         "/" + CPLAWSURLEncode(m_osBucketObjectKey, false));
     894             : 
     895           6 :     std::string osStringToSign;
     896           3 :     osStringToSign += osVerb + "\n";
     897           3 :     osStringToSign += "\n";  // Content_MD5
     898           3 :     osStringToSign += "\n";  // Content_Type
     899           3 :     osStringToSign += osExpires + "\n";
     900             :     // osStringToSign += // Canonicalized_Extension_Headers
     901           3 :     osStringToSign += osCanonicalizedResource;
     902             : #ifdef DEBUG_VERBOSE
     903             :     CPLDebug("GS", "osStringToSign = %s", osStringToSign.c_str());
     904             : #endif
     905             : 
     906           3 :     if (!m_osAccessKeyId.empty())
     907             :     {
     908             :         // No longer documented but actually works !
     909           2 :         GByte abySignature[CPL_SHA1_HASH_SIZE] = {};
     910           4 :         CPL_HMAC_SHA1(m_osSecretAccessKey.c_str(), m_osSecretAccessKey.size(),
     911           2 :                       osStringToSign.c_str(), osStringToSign.size(),
     912             :                       abySignature);
     913             : 
     914           2 :         char *pszBase64 = CPLBase64Encode(sizeof(abySignature), abySignature);
     915           2 :         std::string osSignature(pszBase64);
     916           2 :         CPLFree(pszBase64);
     917             : 
     918           2 :         ResetQueryParameters();
     919           2 :         AddQueryParameter("GoogleAccessId", m_osAccessKeyId);
     920           2 :         AddQueryParameter("Expires", osExpires);
     921           2 :         AddQueryParameter("Signature", osSignature);
     922             :     }
     923             :     else
     924             :     {
     925           1 :         unsigned nSignatureLen = 0;
     926           1 :         GByte *pabySignature = CPL_RSA_SHA256_Sign(
     927           1 :             m_oManager.GetPrivateKey().c_str(), osStringToSign.data(),
     928           1 :             static_cast<unsigned>(osStringToSign.size()), &nSignatureLen);
     929           1 :         if (pabySignature == nullptr)
     930           0 :             return std::string();
     931           1 :         char *pszBase64 = CPLBase64Encode(nSignatureLen, pabySignature);
     932           1 :         CPLFree(pabySignature);
     933           1 :         std::string osSignature(pszBase64);
     934           1 :         CPLFree(pszBase64);
     935             : 
     936           1 :         ResetQueryParameters();
     937           1 :         AddQueryParameter("GoogleAccessId", m_oManager.GetClientEmail());
     938           1 :         AddQueryParameter("Expires", osExpires);
     939           1 :         AddQueryParameter("Signature", osSignature);
     940             :     }
     941           3 :     return m_osURL;
     942             : }
     943             : 
     944             : #endif
     945             : 
     946             : //! @endcond

Generated by: LCOV version 1.14