Chilkat Examples

ChilkatHOMEAndroid™Classic ASPCC++C#Mono C#.NET Core C#C# UWP/WinRTDataFlexDelphi ActiveXDelphi DLLVisual FoxProJavaLianjaMFCObjective-CPerlPHP ActiveXPHP ExtensionPowerBuilderPowerShellPureBasicCkPythonChilkat2-PythonRubySQL ServerSwift 2Swift 3/4TclUnicode CUnicode C++Visual Basic 6.0VB.NETVB.NET UWP/WinRTVBScriptXojo PluginNode.jsExcelGo

C Web API Examples

Primary Categories

CardConnect
Facebook
GeoOp
Jira
Microsoft Calendar
PayPal
Peoplevox

QuickBooks
Shopify
Stripe
SugarCRM
Twitter
VoiceBase
Walmart
Xero
eBay
effectconnect

 

 

 

(C) Quickbooks OAuth1 Authorization (3-legged)

Demonstrates 3-legged OAuth1 authorization for Quickbooks.

Chilkat C/C++ Library Downloads

MS Visual C/C++

Linux/CentOS C/C++

Alpine Linux C/C++

MAC OS X C/C++

armhf/aarch64 C/C++

C++ Builder

iOS C/C++

Android C/C++

Win Mobile 5.0/Pocket PC 2003

Solaris C/C++

FreeBSD C/C++

OpenBSD C/C++

MinGW C/C++

#include <C_CkHttp.h>
#include <C_CkHttpRequest.h>
#include <C_CkHttpResponse.h>
#include <C_CkHashtable.h>
#include <C_CkStringBuilder.h>
#include <C_CkSocket.h>
#include <C_CkTask.h>
#include <C_CkJsonObject.h>
#include <C_CkFileAccess.h>

void ChilkatSample(void)
    {
    const char *consumerKey;
    const char *consumerSecret;
    const char *requestTokenUrl;
    const char *authorizeUrl;
    const char *accessTokenUrl;
    const char *callbackUrl;
    int callbackLocalPort;
    HCkHttp http;
    BOOL success;
    HCkHttpRequest req;
    HCkHttpResponse resp;
    HCkHashtable hashTab1;
    const char *requestToken;
    const char *requestTokenSecret;
    HCkStringBuilder sbUrlForBrowser;
    const char *urlForBrowser;
    HCkSocket listenSock;
    int backLog;
    int maxWaitMs;
    HCkTask task;
    HCkSocket sock;
    const char *startLine;
    const char *requestHeader;
    HCkStringBuilder sbResponseHtml;
    HCkStringBuilder sbResponse;
    HCkStringBuilder sbStartLine;
    int numReplacements;
    const char *authVerifier;
    HCkHashtable hashTab2;
    const char *accessToken;
    const char *accessTokenSecret;
    HCkJsonObject json;
    const char *realmId;
    const char *dataSource;
    HCkFileAccess fac;

    consumerKey = "QUICKBOOKS_CONSUMER_KEY";
    consumerSecret = "QUICKBOOKS_CONSUMER_SECRET";

    requestTokenUrl = "https://oauth.intuit.com/oauth/v1/get_request_token";
    authorizeUrl = "https://appcenter.intuit.com/Connect/Begin";
    accessTokenUrl = "https://oauth.intuit.com/oauth/v1/get_access_token";

    // The port number is picked at random. It's some unused port that won't likely conflict with anything else..
    callbackUrl = "http://localhost:3017/";
    callbackLocalPort = 3017;

    // The 1st step in 3-legged OAuth1.0a is to send a POST to the request token URL to obtain an OAuth Request Token
    http = CkHttp_Create();

    CkHttp_putOAuth1(http,TRUE);
    CkHttp_putOAuthConsumerKey(http,consumerKey);
    CkHttp_putOAuthConsumerSecret(http,consumerSecret);
    CkHttp_putOAuthCallback(http,callbackUrl);

    req = CkHttpRequest_Create();
    resp = CkHttp_PostUrlEncoded(http,requestTokenUrl,req);
    if (CkHttp_getLastMethodSuccess(http) != TRUE) {
        printf("%s\n",CkHttp_lastErrorText(http));
        CkHttp_Dispose(http);
        CkHttpRequest_Dispose(req);
        return;
    }

    if (CkHttpResponse_getStatusCode(resp) >= 400) {
        printf("Error response status code = %d\n",CkHttpResponse_getStatusCode(resp));
        printf("%s\n",CkHttpResponse_bodyStr(resp));
        CkHttp_Dispose(http);
        CkHttpRequest_Dispose(req);
        return;
    }

    // If successful, the resp.BodyStr contains this:  
    // oauth_token=-Wa_KwAAAAAAxfEPAAABV8Qar4Q&oauth_token_secret=OfHY4tZBX2HK4f7yIw76WYdvnl99MVGB&oauth_callback_confirmed=true
    printf("%s\n",CkHttpResponse_bodyStr(resp));

    hashTab1 = CkHashtable_Create();
    CkHashtable_AddQueryParams(hashTab1,CkHttpResponse_bodyStr(resp));

    requestToken = CkHashtable_lookupStr(hashTab1,"oauth_token");
    requestTokenSecret = CkHashtable_lookupStr(hashTab1,"oauth_token_secret");
    CkHttp_putOAuthTokenSecret(http,requestTokenSecret);

    CkHttpResponse_Dispose(resp);

    printf("oauth_token = %s\n",requestToken);
    printf("oauth_token_secret = %s\n",requestTokenSecret);

    // ---------------------------------------------------------------------------
    // The next step is to form a URL to send to the AuthorizeUrl
    // This is an HTTP GET that we load into a popup browser.
    sbUrlForBrowser = CkStringBuilder_Create();
    CkStringBuilder_Append(sbUrlForBrowser,authorizeUrl);
    CkStringBuilder_Append(sbUrlForBrowser,"?oauth_token=");
    CkStringBuilder_Append(sbUrlForBrowser,requestToken);
    urlForBrowser = CkStringBuilder_getAsString(sbUrlForBrowser);

    // When the urlForBrowser is loaded into a browser, the response from Quickbooks will redirect back to localhost:3017
    // We'll need to start a socket that is listening on port 3017 for the callback from the browser.
    listenSock = CkSocket_Create();

    backLog = 5;
    success = CkSocket_BindAndListen(listenSock,callbackLocalPort,backLog);
    if (success != TRUE) {
        printf("%s\n",CkSocket_lastErrorText(listenSock));
        CkHttp_Dispose(http);
        CkHttpRequest_Dispose(req);
        CkHashtable_Dispose(hashTab1);
        CkStringBuilder_Dispose(sbUrlForBrowser);
        CkSocket_Dispose(listenSock);
        return;
    }

    // Wait for the browser's connection in a background thread.
    // (We'll send load the URL into the browser following this..)
    // Wait a max of 60 seconds before giving up.
    maxWaitMs = 60000;
    task = CkSocket_AcceptNextConnectionAsync(listenSock,maxWaitMs);
    CkTask_Run(task);

    //  At this point, your application should load the URL in a browser.
    //  For example,
    //  in C#:  System.Diagnostics.Process.Start(urlForBrowser);
    //  in Java: Desktop.getDesktop().browse(new URI(urlForBrowser));
    //  in VBScript: Set wsh=WScript.CreateObject("WScript.Shell")
    //               wsh.Run urlForBrowser
    //  The Quickbooks 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...
    // System.Diagnostics.Process.Start(urlForBrowser);

    // Wait for the listenSock's task to complete.
    success = CkTask_Wait(task,maxWaitMs);
    if (!success || (CkTask_getStatusInt(task) != 7) || (CkTask_getTaskSuccess(task) != TRUE)) {
        if (!success) {
            // The task.LastErrorText applies to the Wait method call.
            printf("%s\n",CkTask_lastErrorText(task));
        }
        else {
            // The ResultErrorText applies to the underlying task method call (i.e. the AcceptNextConnection)
            printf("%s\n",CkTask_status(task));
            printf("%s\n",CkTask_resultErrorText(task));
        }

        CkTask_Dispose(task);
        CkHttp_Dispose(http);
        CkHttpRequest_Dispose(req);
        CkHashtable_Dispose(hashTab1);
        CkStringBuilder_Dispose(sbUrlForBrowser);
        CkSocket_Dispose(listenSock);
        return;
    }

    // If we get to this point, the connection from the browser arrived and was accepted.

    // We no longer need the listen socket...
    // Close it so that it's no longer listening on port 3017.
    CkSocket_Close(listenSock,10);

    // The first thing to do is to get the connected socket.
    sock = CkSocket_Create();
    CkSocket_LoadTaskResult(sock,task);
    CkTask_Dispose(task);

    // Read the start line of the request..
    startLine = CkSocket_receiveUntilMatch(sock,"\r\n");
    if (CkSocket_getLastMethodSuccess(sock) != TRUE) {
        printf("%s\n",CkSocket_lastErrorText(sock));
        CkHttp_Dispose(http);
        CkHttpRequest_Dispose(req);
        CkHashtable_Dispose(hashTab1);
        CkStringBuilder_Dispose(sbUrlForBrowser);
        CkSocket_Dispose(listenSock);
        CkSocket_Dispose(sock);
        return;
    }

    // Read the request header.
    requestHeader = CkSocket_receiveUntilMatch(sock,"\r\n\r\n");
    if (CkSocket_getLastMethodSuccess(sock) != TRUE) {
        printf("%s\n",CkSocket_lastErrorText(sock));
        CkHttp_Dispose(http);
        CkHttpRequest_Dispose(req);
        CkHashtable_Dispose(hashTab1);
        CkStringBuilder_Dispose(sbUrlForBrowser);
        CkSocket_Dispose(listenSock);
        CkSocket_Dispose(sock);
        return;
    }

    // The browser SHOULD be sending us a GET request, and therefore there is no body to the request.
    // Once the request header is received, we have all of it.
    // We can now send our HTTP response.
    sbResponseHtml = CkStringBuilder_Create();
    CkStringBuilder_Append(sbResponseHtml,"<html><body><p>Chilkat thanks you!</b></body</html>");

    sbResponse = CkStringBuilder_Create();
    CkStringBuilder_Append(sbResponse,"HTTP/1.1 200 OK\r\n");
    CkStringBuilder_Append(sbResponse,"Content-Length: ");
    CkStringBuilder_AppendInt(sbResponse,CkStringBuilder_getLength(sbResponseHtml));
    CkStringBuilder_Append(sbResponse,"\r\n");
    CkStringBuilder_Append(sbResponse,"Content-Type: text/html\r\n");
    CkStringBuilder_Append(sbResponse,"\r\n");
    CkStringBuilder_AppendSb(sbResponse,sbResponseHtml);

    CkSocket_SendString(sock,CkStringBuilder_getAsString(sbResponse));
    CkSocket_Close(sock,50);

    // The information we need is in the startLine.
    // For example, the startLine will look like this:
    //  GET /?oauth_token=abcdRQAAZZAAxfBBAAABVabcd_k&oauth_verifier=9rdOq5abcdCe6cn8M3jabcdj3Eabcd HTTP/1.1
    sbStartLine = CkStringBuilder_Create();
    CkStringBuilder_Append(sbStartLine,startLine);
    numReplacements = CkStringBuilder_Replace(sbStartLine,"GET /?","");
    numReplacements = CkStringBuilder_Replace(sbStartLine," HTTP/1.1","");
    CkStringBuilder_Trim(sbStartLine);

    // oauth_token=qyprdP04IrTDIXtP1HRZz0geQdjXHVlGDxXPexlXZsjZNRcY&oauth_verifier=arx5pj5&realmId=193514465596199&dataSource=QBO
    printf("startline: %s\n",CkStringBuilder_getAsString(sbStartLine));

    CkHashtable_Clear(hashTab1);
    CkHashtable_AddQueryParams(hashTab1,CkStringBuilder_getAsString(sbStartLine));

    requestToken = CkHashtable_lookupStr(hashTab1,"oauth_token");
    authVerifier = CkHashtable_lookupStr(hashTab1,"oauth_verifier");

    // ------------------------------------------------------------------------------
    // Finally , we must exchange the OAuth Request Token for an OAuth Access Token.

    CkHttp_putOAuthToken(http,requestToken);
    CkHttp_putOAuthVerifier(http,authVerifier);
    resp = CkHttp_PostUrlEncoded(http,accessTokenUrl,req);
    if (CkHttp_getLastMethodSuccess(http) != TRUE) {
        printf("%s\n",CkHttp_lastErrorText(http));
        CkHttp_Dispose(http);
        CkHttpRequest_Dispose(req);
        CkHashtable_Dispose(hashTab1);
        CkStringBuilder_Dispose(sbUrlForBrowser);
        CkSocket_Dispose(listenSock);
        CkSocket_Dispose(sock);
        CkStringBuilder_Dispose(sbResponseHtml);
        CkStringBuilder_Dispose(sbResponse);
        CkStringBuilder_Dispose(sbStartLine);
        return;
    }

    // Make sure a successful response was received.
    if (CkHttpResponse_getStatusCode(resp) != 200) {
        printf("%s\n",CkHttpResponse_statusLine(resp));
        printf("%s\n",CkHttpResponse_header(resp));
        printf("%s\n",CkHttpResponse_bodyStr(resp));
        CkHttp_Dispose(http);
        CkHttpRequest_Dispose(req);
        CkHashtable_Dispose(hashTab1);
        CkStringBuilder_Dispose(sbUrlForBrowser);
        CkSocket_Dispose(listenSock);
        CkSocket_Dispose(sock);
        CkStringBuilder_Dispose(sbResponseHtml);
        CkStringBuilder_Dispose(sbResponse);
        CkStringBuilder_Dispose(sbStartLine);
        return;
    }

    // If successful, the resp.BodyStr contains something like this:
    // oauth_token=12347455-ffffrrlaBdCjbdGfyjZabcdb5APNtuTPNabcdEpp&oauth_token_secret=RxxxxJ8mTzUhwES4xxxxuJyFWDN8ZfHmrabcddh88LmWE
    printf("%s\n",CkHttpResponse_bodyStr(resp));

    hashTab2 = CkHashtable_Create();
    CkHashtable_AddQueryParams(hashTab2,CkHttpResponse_bodyStr(resp));

    accessToken = CkHashtable_lookupStr(hashTab2,"oauth_token");
    accessTokenSecret = CkHashtable_lookupStr(hashTab2,"oauth_token_secret");

    CkHttpResponse_Dispose(resp);

    // The access token + secret is what should be saved and used for
    // subsequent REST API calls.
    printf("Access Token = %s\n",accessToken);
    printf("Access Token Secret = %s\n",accessTokenSecret);

    // Save this access token for future calls.
    json = CkJsonObject_Create();
    CkJsonObject_AppendString(json,"oauth_token",accessToken);
    CkJsonObject_AppendString(json,"oauth_token_secret",accessTokenSecret);

    // Also save the realmId and dataSource from hashTab1.
    realmId = CkHashtable_lookupStr(hashTab1,"realmId");
    printf("realmId = %s\n",realmId);
    dataSource = CkHashtable_lookupStr(hashTab1,"dataSource");
    printf("dataSource = %s\n",dataSource);

    CkJsonObject_AppendString(json,"realmId",realmId);
    CkJsonObject_AppendString(json,"dataSource",dataSource);

    fac = CkFileAccess_Create();
    CkFileAccess_WriteEntireTextFile(fac,"qa_data/tokens/quickbooks.json",CkJsonObject_emit(json),"utf-8",FALSE);

    printf("Success.\n");


    CkHttp_Dispose(http);
    CkHttpRequest_Dispose(req);
    CkHashtable_Dispose(hashTab1);
    CkStringBuilder_Dispose(sbUrlForBrowser);
    CkSocket_Dispose(listenSock);
    CkSocket_Dispose(sock);
    CkStringBuilder_Dispose(sbResponseHtml);
    CkStringBuilder_Dispose(sbResponse);
    CkStringBuilder_Dispose(sbStartLine);
    CkHashtable_Dispose(hashTab2);
    CkJsonObject_Dispose(json);
    CkFileAccess_Dispose(fac);

    }

 

© 2000-2019 Chilkat Software, Inc. All Rights Reserved.