Akamai CDN Setup - Client Side

Overview

The purpose of the Akamai CDN client side implementation is to inject the Arkose client side scripts into the HTML on a page such as Login or Registration. Once an Arkose challenge is triggered and completed, a POST request is made which will trigger the Akamai EdgeWorker to proxy the Arkose Verify request; found here Akamai CDN Setup - Server Side.

Setup

Prerequisite: An Akamai Property needs to be setup. The details on how to do so can also be found in Akamai CDN Setup - Server Side.

Once an Akamai Property is setup, the HTML for a Login or Registration page will need to be hosted and served via the Akamai Property Origin Server. This can be via an AWS S3 bucket, GCP Bucket, or a custom server. Here is an example HTML file with Arkose client side scripts injected:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Akamai CDN Integration - Verify Proxy Example</title>
    <!-- Javascript to inject in the page header -->
    <script 
        src="https://client-api.arkoselabs.com/v2/<YOUR PUBLIC KEY>/api.js"
        data-callback="setupEnforcement"
        async defer
    ></script>
    <!---->
    <style>
        form { width: 10em; margin: 1em auto; }
        form .formField { margin: 0.5em 0 }
    </style>
  </head>
  <body>
    <form action="" method="post" id="submitForm">
        <h3>Login</h3>
        <div class="formField">
            <input type="text" id="username" placeholder="Username" />
        </div>
        <div>
            <input type="password" id="password" placeholder="Password" />
        </div>
        <div class="formField">
            <input type="submit" value="Submit" id="submitButton" />
        </div>
        <p>Select Token Method</p>
          <input type="radio" id="header-token-method" name="token-method" value="header" checked="checked">
          <label for="header-token-method">Header</label><br>
          <input type="radio" id="cookie-token-method" name="token-method" value="cookie">
          <label for="cookie-token-method">Cookie</label><br>
    </form>
    <!-- Javascript to inject before the closing body tag -->
    <script>
        var arkoseReady = false; // Flag for checking if CAPI has called onReady
        var arkoseComplete = false; // Flag to signify the challenge being completed
        var arkoseResetting = false; // Flag to signify when the challenge is being reset
        var arkose; // Arkose Enforcement Object

        // The following variables can be changed
        var arkoseCookieName = 'arkoseToken'; // The name of the cookie that the Arkose token will be stored in
        var arkoseErrorCookieName = 'arkoseError'; // The name of the cookie that an Arkose error will be stored in 
        var arkoseCookieLife = 5 * 60 * 1000;  // The length of time that the cookie should be active for, 5 mins is the default
        var buttonSelector = '#submitButton'; // The querySelector string used for selecting the required button to protect

        var button = document.querySelector(buttonSelector);
        setupButton();

        /**
        * Sets up the events for hijacking the button click
        */
        function setupButton() {
            if (button) {
                arkoseComplete = false;
                if (!arkoseReady) {
                    button.setAttribute('disabled', true);
                }
                button.addEventListener('click', function (event) {
                    if (!arkoseReady) {
                        event.preventDefault();
                        return;
                    }
                    if (!arkoseComplete) {
                        event.preventDefault();
                        arkose.run();
                        return;
                    }
                })
            }
        }

        /**
        * Sends a request to an endpoint which triggers the Akamai EdgeWorker to make a proxy call to Arkose Verify.
        */
        function proxyVerify(token) {
            var tokenMethod = document.querySelector('input[name="token-method"]:checked').value;
            var request = new XMLHttpRequest();

            // Route to run the Proxy EdgeWorker to call Arkose Verify
            request.open("POST", "/proxy-verify");

            request.setRequestHeader('Content-type', 'application/json; charset=utf-8');

            if (tokenMethod === "header") {
                request.setRequestHeader('arkose-token', token);
            }

            if (tokenMethod === "cookie") {
                document.cookie = "arkose-token" + '=' + token + ';expires=' + new Date(Date.now() + 10000).toUTCString() + '; path=/;';
            }

            var body = JSON.stringify({
                username: document.getElementById("username").value,
                password: document.getElementById("password").value
            });

            // Sending the request
            request.send(body);

            request.onload = function () {
                if (request.readyState == 4 && request.status == 200) {
                    const data = request.response;
                    console.log('Successfully Verified', data);
                }
            }
        }

        /**
        * In the case of an error reset the arkose cookie and setup a new cookie that tracks the error
        * @param  {string} error The error string to set in the error cookie
        */
        function handleError(error) {
            arkoseComplete = true;
            document.cookie = arkoseCookieName + '=;expires=' + new Date(Date.now() + arkoseCookieLife).toUTCString() + '; path=/;';
            document.cookie = arkoseErrorCookieName + '=' + error + ';expires=' + new Date(Date.now() + arkoseCookieLife).toUTCString() + '; path=/;';                               
        }  

        /**
        * The callback thats called after the Arkose Labs script has been loaded
        * @param  {object} myEnforcement The Arkose Labs enforcement object
        */
        function setupEnforcement(myEnforcement) {
            arkose = myEnforcement;
            arkose.setConfig({
                onReady: function() {
                    arkoseReady = true;
                    if (button) {
                        button.removeAttribute('disabled');
                    }
                    if (arkoseResetting) {
                        arkoseResetting = false;
                        arkose.run();
                    }
                    document.cookie = arkoseCookieName + '=' + '=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;';
                    document.cookie = arkoseErrorCookieName + '=' + '=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;';
                },
                onCompleted: function(response) {
                    arkoseComplete = true;
                    if (response.token){
                        proxyVerify(response.token);
                    } else {
                        handleError('TOKEN_MISSING');
                    }
                },
                onError: function(response) {
                    handleError(response.error ? response.error.error : 'error');
                    button.removeAttribute('disabled');
                    button.click();
                }
            });
        }
    </script>
    <!---->
  </body>
</html>

To view our example in Github please visit: Arkose Examples Akamai Client Side.

This example HTML page mimics a Login page and contains two fields: username and password. The Token Method is another input that is used to choose how the Arkose Token is sent to the Akamai EdgeWorker.

Once a User completes an Arkose Challenge, the onCompleted callback is triggered, where the proxyVerify function is called with the Arkose Token. This function makes a POST request to /proxy-verify, the route in which triggers the EdgeWorker to proxy the verify call runs. This is tied to the Akamai Property rule here: Rule that sets up proxy call. Depending on which Token Method was selected, the Arkose Token is sent via a Header arkose-token or a Cookie.

This route /proxy-verify can be anything, as long as the Akamai Property contains a rule to run.

📘

The implementation will have to match between the how the Client Side code (this example) and the Server Side code (Akamai CDN Setup - Server Side) is configured. For example, if the Server Side is configured to read in the Arkose Token via the Header, the Client Side will need to send via Header.

Inject EdgeWorker

The following section details on how to leverage Akamai’s EdgeWorkers to be able to Inject Arkose scripts into a HTML file. The example code can be found here: Inject code example.

View this documentation: Arkose Verify EdgeWorker on how to add EdgeWorkers to an Akamai Property.