Chilkat HOME Android™ AutoIt C C# C++ Chilkat2-Python CkPython Classic ASP DataFlex Delphi DLL Go Java Node.js Objective-C PHP Extension Perl PowerBuilder PowerShell PureBasic Ruby SQL Server Swift Tcl Unicode C Unicode C++ VB.NET VBScript Visual Basic 6.0 Visual FoxPro Xojo Plugin
(PureBasic) Dynamics CRM OAuth2 Access TokenDemonstrates how to get a Microsoft Dynamics CRM OAuth2 access token from a desktop (installed) application or script.
IncludeFile "CkJsonObject.pb" IncludeFile "CkDateTime.pb" IncludeFile "CkHttp.pb" IncludeFile "CkFileAccess.pb" IncludeFile "CkStringBuilder.pb" IncludeFile "CkOAuth2.pb" Procedure ChilkatExample() ; This example requires the Chilkat API to have been previously unlocked. ; See Global Unlock Sample for sample code. oauth2.i = CkOAuth2::ckCreate() If oauth2.i = 0 Debug "Failed to create object." ProcedureReturn EndIf success.i ; This should be the port in the localhost redirect URL for your app. ; Important: Make sure the redirect URI in your Azure app registration ; is exactly "http://localhost:3017/". Don't forget the ending "/" char. ; The redirect URL would look like "http://localhost:3017/" if the port number is 3017. CkOAuth2::setCkListenPort(oauth2, 3017) ; Get the endpoints from the Azure Portal in the "endpoints" section of App Registrations.. ; The DYNAMICS-ENDPOINT-GUID is a Tenant dynamics-endpoint-guid such as 1b54e7ee-d61b-4f12-a621-a6b2805b98c2 CkOAuth2::setCkAuthorizationEndpoint(oauth2, "https://login.microsoftonline.com/DYNAMICS-ENDPOINT-GUID/oauth2/authorize") CkOAuth2::setCkTokenEndpoint(oauth2, "https://login.microsoftonline.com/DYNAMICS-ENDPOINT-GUID/oauth2/token") ; Replace these with actual values. CkOAuth2::setCkClientId(oauth2, "DYNAMICS-CRM-CLIENT-ID") CkOAuth2::setCkClientSecret(oauth2, "DYNAMICS-CRM-SECRET-KEY") CkOAuth2::setCkResource(oauth2, "https://mydynamicsdomain.api.crm.dynamics.com") CkOAuth2::setCkRedirectAllowHtml(oauth2, "<html><body><p>Thank you for granting access.</p></bod></html>") CkOAuth2::setCkRedirectDenyHtml(oauth2, "<html><body><p>Maybe next time...</p></bod></html>") ; Begin the OAuth2 three-legged flow. This returns a URL that should be loaded in a browser. url.s = CkOAuth2::ckStartAuth(oauth2) If CkOAuth2::ckLastMethodSuccess(oauth2) <> 1 Debug CkOAuth2::ckLastErrorText(oauth2) CkOAuth2::ckDispose(oauth2) ProcedureReturn EndIf ; We can pre-check the URL by sending a GET to catch any up-front errors. ; If we receive JSON with an error in response, then we don't proceed any farther. ; If we receive an HTML response, and the response status = 200, then all should be OK. ; We can proceed by displaying an interactive browser loaded with the url. http.i = CkHttp::ckCreate() If http.i = 0 Debug "Failed to create object." ProcedureReturn EndIf sbPreCheck.i = CkStringBuilder::ckCreate() If sbPreCheck.i = 0 Debug "Failed to create object." ProcedureReturn EndIf success = CkHttp::ckQuickGetSb(http,url,sbPreCheck) If (CkHttp::ckLastStatus(http) <> 200) OR (CkStringBuilder::ckStartsWith(sbPreCheck,"{",1) <> 1) Debug "JSON Error Response:" Debug CkStringBuilder::ckGetAsString(sbPreCheck) CkOAuth2::ckDispose(oauth2) CkHttp::ckDispose(http) CkStringBuilder::ckDispose(sbPreCheck) ProcedureReturn EndIf ; If the HTTP request itself failed, then examine the error. If success = 0 Debug CkHttp::ckLastErrorText(http) CkOAuth2::ckDispose(oauth2) CkHttp::ckDispose(http) CkStringBuilder::ckDispose(sbPreCheck) ProcedureReturn EndIf ; OK.. the pre-flight check seems OK, go ahead with getting the interactive user authorization.. ; At this point, your application should load the URL in a browser. ; For example, ; in C#: System.Diagnostics.Process.Start(url); ; in Java: Desktop.getDesktop().browse(new URI(url)); ; in VBScript: Set wsh=WScript.CreateObject("WScript.Shell") ; wsh.Run url ; in Xojo: ShowURL(url) (see http://docs.xojo.com/index.php/ShowURL) ; in Dataflex: Runprogram Background "c:\Program Files\Internet Explorer\iexplore.exe" sUrl ; The GitHub account owner would interactively accept or deny the authorization request. ; Add the code to load the url in a web browser here... ; Add the code to load the url in a web browser here... ; Add the code to load the url in a web browser here... ; Now wait for the authorization. ; We'll wait for a max of 30 seconds. numMsWaited.i = 0 While (numMsWaited < 30000) AND (CkOAuth2::ckAuthFlowState(oauth2) < 3) CkOAuth2::ckSleepMs(oauth2,100) numMsWaited = numMsWaited + 100 Wend ; If there was no response from the browser within 30 seconds, then ; the AuthFlowState will be equal to 1 or 2. ; 1: Waiting for Redirect. The OAuth2 background thread is waiting to receive the redirect HTTP request from the browser. ; 2: Waiting for Final Response. The OAuth2 background thread is waiting for the final access token response. ; In that case, cancel the background task started in the call to StartAuth. If CkOAuth2::ckAuthFlowState(oauth2) < 3 CkOAuth2::ckCancel(oauth2) Debug "No response from the browser!" CkOAuth2::ckDispose(oauth2) CkHttp::ckDispose(http) CkStringBuilder::ckDispose(sbPreCheck) ProcedureReturn EndIf ; Check the AuthFlowState to see if authorization was granted, denied, or if some error occurred ; The possible AuthFlowState values are: ; 3: Completed with Success. The OAuth2 flow has completed, the background thread exited, and the successful JSON response is available in AccessTokenResponse property. ; 4: Completed with Access Denied. The OAuth2 flow has completed, the background thread exited, and the error JSON is available in AccessTokenResponse property. ; 5: Failed Prior to Completion. The OAuth2 flow failed to complete, the background thread exited, and the error information is available in the FailureInfo property. If CkOAuth2::ckAuthFlowState(oauth2) = 5 Debug "OAuth2 failed to complete." Debug CkOAuth2::ckFailureInfo(oauth2) CkOAuth2::ckDispose(oauth2) CkHttp::ckDispose(http) CkStringBuilder::ckDispose(sbPreCheck) ProcedureReturn EndIf If CkOAuth2::ckAuthFlowState(oauth2) = 4 Debug "OAuth2 authorization was denied." Debug CkOAuth2::ckAccessTokenResponse(oauth2) CkOAuth2::ckDispose(oauth2) CkHttp::ckDispose(http) CkStringBuilder::ckDispose(sbPreCheck) ProcedureReturn EndIf If CkOAuth2::ckAuthFlowState(oauth2) <> 3 Debug "Unexpected AuthFlowState:" + Str(CkOAuth2::ckAuthFlowState(oauth2)) CkOAuth2::ckDispose(oauth2) CkHttp::ckDispose(http) CkStringBuilder::ckDispose(sbPreCheck) ProcedureReturn EndIf Debug "OAuth2 authorization granted!" Debug "Access Token = " + CkOAuth2::ckAccessToken(oauth2) ; Get the full JSON response: json.i = CkJsonObject::ckCreate() If json.i = 0 Debug "Failed to create object." ProcedureReturn EndIf CkJsonObject::ckLoad(json,CkOAuth2::ckAccessTokenResponse(oauth2)) CkJsonObject::setCkEmitCompact(json, 0) ; The JSON response looks like this: ; { ; "token_type": "Bearer", ; "scope": "user_impersonation", ; "expires_in": "3599", ; "ext_expires_in": "0", ; "expires_on": "1524783438", ; "not_before": "1524779538", ; "resource": "https://mydomain.api.crm.dynamics.com", ; "access_token": "...", ; "refresh_token": "...", ; "id_token": "..." ; } ; If an "expires_on" member does not exist, then add the JSON member by ; getting the current system date/time and adding the "expires_in" seconds. ; This way we'll know when the token expires. If CkJsonObject::ckHasMember(json,"expires_on") <> 1 dtExpire.i = CkDateTime::ckCreate() If dtExpire.i = 0 Debug "Failed to create object." ProcedureReturn EndIf CkDateTime::ckSetFromCurrentSystemTime(dtExpire) CkDateTime::ckAddSeconds(dtExpire,CkJsonObject::ckIntOf(json,"expires_in")) CkJsonObject::ckAppendString(json,"expires_on",CkDateTime::ckGetAsUnixTimeStr(dtExpire,0)) EndIf Debug CkJsonObject::ckEmit(json) ; Save the JSON to a file for future requests. fac.i = CkFileAccess::ckCreate() If fac.i = 0 Debug "Failed to create object." ProcedureReturn EndIf CkFileAccess::ckWriteEntireTextFile(fac,"qa_data/tokens/dynamicsCrm.json",CkJsonObject::ckEmit(json),"utf-8",0) CkOAuth2::ckDispose(oauth2) CkHttp::ckDispose(http) CkStringBuilder::ckDispose(sbPreCheck) CkJsonObject::ckDispose(json) CkDateTime::ckDispose(dtExpire) CkFileAccess::ckDispose(fac) ProcedureReturn EndProcedure |
© 2000-2025 Chilkat Software, Inc. All Rights Reserved.