Sample code for 30+ languages & platforms
PureBasic

Refresh Access Token on 401 Unauthorized and Retry (Service Account)

See more Google Cloud Storage Examples

Demonstrates how to handle an expired access token error, refresh the token, and retry the request. (In this case, the request is to download an object from Google Cloud Storage.)

Chilkat PureBasic Downloads

PureBasic
IncludeFile "CkBinData.pb"
IncludeFile "CkHttp.pb"
IncludeFile "CkFileAccess.pb"
IncludeFile "CkSocket.pb"
IncludeFile "CkStringBuilder.pb"
IncludeFile "CkAuthGoogle.pb"

Procedure ChilkatExample()

    success.i = 0

    ; This example requires the Chilkat API to have been previously unlocked.
    ; See Global Unlock Sample for sample code.

    ; This example uses a previously obtained access token having permission for the 
    ; scope "https://www.googleapis.com/auth/cloud-platform"

    ; In this example, Get Google Cloud Storage OAuth2 Access Token, 
    ; the service account access token was saved to a text file.  This example fetches the access token from the file..
    sbToken.i = CkStringBuilder::ckCreate()
    If sbToken.i = 0
        Debug "Failed to create object."
        ProcedureReturn
    EndIf

    success = CkStringBuilder::ckLoadFile(sbToken,"qa_data/tokens/googleCloudStorageAccessToken.txt","utf-8")
    If success = 0
        Debug "Failed to load access token."
        CkStringBuilder::ckDispose(sbToken)
        ProcedureReturn
    EndIf

    http.i = CkHttp::ckCreate()
    If http.i = 0
        Debug "Failed to create object."
        ProcedureReturn
    EndIf

    CkHttp::setCkAuthToken(http, CkStringBuilder::ckGetAsString(sbToken))

    ; Construct a URL to download an object named "starfish.jpg" from the "chilkat-ocean" bucket.
    CkHttp::ckSetUrlVar(http,"bucket_name","chilkat-ocean")
    CkHttp::ckSetUrlVar(http,"object_name","starfish.jpg")
    url.s = "https://www.googleapis.com/storage/v1/b/{$bucket_name}/o/{$object_name}?alt=media"

    ; If there is an error response, then we didn't actually download the file data,
    ; but instead we downloaded an error response..
    fileData.i = CkBinData::ckCreate()
    If fileData.i = 0
        Debug "Failed to create object."
        ProcedureReturn
    EndIf

    success = CkHttp::ckDownloadBd(http,url,fileData)
    responseCode.i = CkHttp::ckLastStatus(http)
    If (success = 0) AND (responseCode <> 401)
        Debug CkHttp::ckLastErrorText(http)
        CkStringBuilder::ckDispose(sbToken)
        CkHttp::ckDispose(http)
        CkBinData::ckDispose(fileData)
        ProcedureReturn
    EndIf

    If responseCode = 401
        Debug "Received 401 Unauthorized response. Attempting to refresh the access token..."

        ; May be that the access token expired.
        ; Load our JSON key and request a new access token, then retry the original request.
        fac.i = CkFileAccess::ckCreate()
        If fac.i = 0
            Debug "Failed to create object."
            ProcedureReturn
        EndIf

        jsonKey.s = CkFileAccess::ckReadEntireTextFile(fac,"qa_data/googleApi/ChilkatCloud-13a07a2e8b3f.json","utf-8")
        If CkFileAccess::ckLastMethodSuccess(fac) <> 1
            Debug CkFileAccess::ckLastErrorText(fac)
            CkStringBuilder::ckDispose(sbToken)
            CkHttp::ckDispose(http)
            CkBinData::ckDispose(fileData)
            CkFileAccess::ckDispose(fac)
            ProcedureReturn
        EndIf

        gAuth.i = CkAuthGoogle::ckCreate()
        If gAuth.i = 0
            Debug "Failed to create object."
            ProcedureReturn
        EndIf

        CkAuthGoogle::setCkJsonKey(gAuth, jsonKey)
        CkAuthGoogle::setCkScope(gAuth, "https://www.googleapis.com/auth/cloud-platform")
        CkAuthGoogle::setCkExpireNumSeconds(gAuth, 3600)
        CkAuthGoogle::setCkSubEmailAddress(gAuth, "")

        tlsSock.i = CkSocket::ckCreate()
        If tlsSock.i = 0
            Debug "Failed to create object."
            ProcedureReturn
        EndIf

        success = CkSocket::ckConnect(tlsSock,"www.googleapis.com",443,1,5000)
        If success <> 1
            Debug CkSocket::ckLastErrorText(tlsSock)
            CkStringBuilder::ckDispose(sbToken)
            CkHttp::ckDispose(http)
            CkBinData::ckDispose(fileData)
            CkFileAccess::ckDispose(fac)
            CkAuthGoogle::ckDispose(gAuth)
            CkSocket::ckDispose(tlsSock)
            ProcedureReturn
        EndIf

        success = CkAuthGoogle::ckObtainAccessToken(gAuth,tlsSock)
        If success <> 1
            Debug CkAuthGoogle::ckLastErrorText(gAuth)
            CkStringBuilder::ckDispose(sbToken)
            CkHttp::ckDispose(http)
            CkBinData::ckDispose(fileData)
            CkFileAccess::ckDispose(fac)
            CkAuthGoogle::ckDispose(gAuth)
            CkSocket::ckDispose(tlsSock)
            ProcedureReturn
        EndIf

        CkFileAccess::ckWriteEntireTextFile(fac,"qa_data/tokens/googleCloudStorageAccessToken.txt",CkAuthGoogle::ckAccessToken(gAuth),"utf-8",0)

        ; Retry the original request.
        CkHttp::setCkAuthToken(http, CkAuthGoogle::ckAccessToken(gAuth))
        success = CkHttp::ckDownloadBd(http,url,fileData)
        If success = 0
            Debug CkHttp::ckLastErrorText(http)
            CkStringBuilder::ckDispose(sbToken)
            CkHttp::ckDispose(http)
            CkBinData::ckDispose(fileData)
            CkFileAccess::ckDispose(fac)
            CkAuthGoogle::ckDispose(gAuth)
            CkSocket::ckDispose(tlsSock)
            ProcedureReturn
        EndIf

    EndIf

    If responseCode <> 200
        ; Get the error response
        sbErrorResponse.i = CkStringBuilder::ckCreate()
        If sbErrorResponse.i = 0
            Debug "Failed to create object."
            ProcedureReturn
        EndIf

        CkStringBuilder::ckAppendBd(sbErrorResponse,fileData,"utf-8",0,0)
        Debug "Error response code = " + Str(responseCode)
        Debug "Error:"
        Debug CkStringBuilder::ckGetAsString(sbErrorResponse)
        CkStringBuilder::ckDispose(sbToken)
        CkHttp::ckDispose(http)
        CkBinData::ckDispose(fileData)
        CkFileAccess::ckDispose(fac)
        CkAuthGoogle::ckDispose(gAuth)
        CkSocket::ckDispose(tlsSock)
        CkStringBuilder::ckDispose(sbErrorResponse)
        ProcedureReturn
    EndIf

    Debug "Success."

    ; Save the downloaded data to a file.
    success = CkBinData::ckWriteFile(fileData,"qa_output/starfish.jpg")


    CkStringBuilder::ckDispose(sbToken)
    CkHttp::ckDispose(http)
    CkBinData::ckDispose(fileData)
    CkFileAccess::ckDispose(fac)
    CkAuthGoogle::ckDispose(gAuth)
    CkSocket::ckDispose(tlsSock)
    CkStringBuilder::ckDispose(sbErrorResponse)


    ProcedureReturn
EndProcedure