ÃÛ¶¹ÊÓƵ

Invoke OpenAPI-based AEM APIs using OAuth Server-to-Server authentication

Learn how to invoke OpenAPI-based AEM APIs on AEM as a Cloud Service from custom applications using OAuth Server-to-Server authentication.

The OAuth Server-to-Server authentication is ideal for backend services needing API access without user interaction. It uses the OAuth 2.0 client_credentials grant type to authenticate the client application.

AVAILABILITY
OpenAPI-based AEM APIs are available as part of an early access program. If you are interested in accessing them, we encourage you to email aem-apis@adobe.com with a description of your use case.

What you learn what-you-learn

In this tutorial, you learn how to:

  • Configure an ÃÛ¶¹ÊÓƵ Developer Console (ADC) project to access the Assets Author API using OAuth Server-to-Server authentication.

  • Develop a sample NodeJS application that calls the Assets Author API to retrieve metadata for a specific asset.

Before you start, make sure you reviewed the following:

Prerequisites

To complete this tutorial, you need:

  • Modernized AEM as a Cloud Service environment with the following:

    • AEM Release 2024.10.18459.20241031T210302Z or later.
    • New style Product Profiles (if environment was created before November 2024)

    See Set up OpenAPI-based AEM APIs article for more details.

  • The sample project must be deployed on to it.

  • Access to the .

  • Install on your local machine to run the sample NodeJS application.

Development steps

The high-level development steps are:

  1. Configure ADC Project

    1. Add the Assets Author API
    2. Configure its authentication method as OAuth Server-to-Server
    3. Associate Product Profile with the authentication configuration
  2. Configure the AEM instance to enable ADC Project communication

  3. Develop a sample NodeJS application

  4. Verify the end-to-end flow

Configure ADC Project

The configure ADC Project step is repeated from the Setup OpenAPI-based AEM APIs. It is repeated to add the Assets Author API and configure its authentication method as OAuth Server-to-Server.

  1. From the , open the desired project.

  2. To add AEM APIs, click on the Add API button.

    Add API

  3. In the Add API dialog, filter by Experience Cloud and select the AEM Assets Author API card and click Next.

    Add AEM API

  4. Next, in the Configure API dialog, select the Server-to-Server authentication option and click Next. The Server-to-Server authentication is ideal for backend services needing API access without user interaction.

    Select authentication

  5. Rename the credential for easier identification (if needed) and click Next. For demo purposes, the default name is used.

    Rename credential

  6. Select the AEM Assets Collaborator Users - author - Program XXX - Environment XXX Product Profile and click Save. As you can see, only the Product Profile associated with the AEM Assets API Users Service is available for selection.

    Select Product Profile

  7. Review the AEM API and authentication configuration.

    AEM API configuration

    Authentication configuration

Configure AEM instance to enable ADC Project communication

Follow the instructions from the Setup OpenAPI-based AEM APIs article to configure the AEM instance to enable ADC Project communication.

Develop a sample NodeJS application

Let’s develop a sample NodeJS application that calls the Assets Author API.

You can use other programming languages like Java, Python, etc., to develop the application.

For testing purposes, you can use the , , or any other REST client to invoke the AEM APIs.

Review the API

Before developing the application, let’s review endpoint from the Assets Author API. The API syntax is:

GET https://{bucket}.adobeaemcloud.com/adobe/../assets/{assetId}/metadata

To retrieve the metadata of a specific asset, you need the bucket and assetId values. The bucket is the AEM instance name without the ÃÛ¶¹ÊÓƵ domain name (.adobeaemcloud.com), for example, author-p63947-e1420428.

The assetId is the JCR UUID of the asset with the urn:aaid:aem: prefix, for example, urn:aaid:aem:a200faf1-6d12-4abc-bc16-1b9a21f870da. There are multiple ways to get the assetId:

  • Append the AEM asset path .json extension to get the asset metadata. For example, https://author-p63947-e1420429.adobeaemcloud.com/content/dam/wknd-shared/en/adventures/cycling-southern-utah/adobestock-221043703.jpg.json and look for the jcr:uuid property.

  • Alternatively, you can get the assetId by inspecting the asset in the browser’s element inspector. Look for the data-id="urn:aaid:aem:..." attribute.

    Inspect asset

Invoke the API using the browser

Before developing the application, let’s invoke the API using the Try it feature in the .

  1. Open the in the browser.

  2. Expand the Metadata section and click on the Delivers the specified asset’s metadata option.

  3. In the right pane, click on the Try it button.
    API documentation

  4. Enter the following values:

    table 0-row-3 1-row-3 2-row-3 3-row-3 4-row-3 5-row-3
    Section Parameter Value
    bucket The AEM instance name without the ÃÛ¶¹ÊÓƵ domain name (.adobeaemcloud.com), for example, author-p63947-e1420428.
    Security Bearer Token Use the access token from the ADC Project’s OAuth Server-to-Server credential.
    Security X-Api-Key Use the ClientID value from the ADC Project’s OAuth Server-to-Server credential.
    Parameters assetId The unique identifier for the asset in AEM, for example, urn:aaid:aem:a200faf1-6d12-4abc-bc16-1b9a21f870da
    Parameters X-ÃÛ¶¹ÊÓƵ-Accept-Experimental 1

    Invoke API - access token

    Invoke API - input values

  5. Click Send to invoke the API, and review the response in the Response tab.

    Invoke API - response

The above steps confirm the modernization of the AEM as a Cloud Service environment, enabling AEM APIs access. It also confirms the successful configuration of the ADC Project, and the OAuth Server-to-Server credential ClientID communication with the AEM Author instance.

Sample NodeJS application

Let’s develop a sample NodeJS application.

To develop the application, you can either use the Run-the-sample-application or the Step-by-step-development instructions.

Run-the-sample-application
  1. Download the sample demo-nodejs-app-to-invoke-aem-openapi application zip file and extract it.

  2. Navigate to the extracted folder and install the dependencies.

    code language-bash
    $ npm install
    
  3. Replace the placeholders in the .env file with the actual values from the ADC Project’s OAuth Server-to-Server credential.

  4. Replace the <BUCKETNAME> and <ASSETID> in the src/index.js file with the actual values.

  5. Run the NodeJS application.

    code language-bash
    $ node src/index.js
    
Step-by-step-development
  1. Create a new NodeJS project.

    code language-bash
    $ mkdir demo-nodejs-app-to-invoke-aem-openapi
    $ cd demo-nodejs-app-to-invoke-aem-openapi
    $ npm init -y
    
  2. Install the fetch and dotenv library to make HTTP requests and read the environment variables respectively.

    code language-bash
    $ npm install node-fetch
    $ npm install dotenv
    
  3. Open the project in your favorite code editor and update the package.json file to add the type to module.

    code language-json
    {
        ...
        "version": "1.0.0",
        "type": "module",
        "main": "index.js",
        ...
    }
    
  4. Create .env file and add the following configuration. Replace the placeholders with the actual values from the ADC Project’s OAuth Server-to-Server credential.

    code language-properties
    CLIENT_ID=<ADC Project OAuth Server-to-Server credential ClientID>
    CLIENT_SECRET=<ADC Project OAuth Server-to-Server credential Client Secret>
    SCOPES=<ADC Project OAuth Server-to-Server credential Scopes>
    
  5. Create src/index.js file and add the following code, and replace the <BUCKETNAME> and <ASSETID> with the actual values.

    code language-javascript
    // Import the dotenv configuration to load environment variables from the .env file
    import "dotenv/config";
    
    // Import the fetch function to make HTTP requests
    import fetch from "node-fetch";
    
    // REPLACE THE FOLLOWING VALUES WITH YOUR OWN
    const bucket = "<BUCKETNAME>"; // Bucket name is the AEM instance name (e.g. author-p63947-e1420428)
    const assetId = "<ASSETID>"; // Asset ID is the unique identifier for the asset in AEM (e.g. urn:aaid:aem:a200faf1-6d12-4abc-bc16-1b9a21f870da). You can get it by inspecting the asset in browser's element inspector, look for data-id="urn:aaid:aem:..."
    
    // Load environment variables for authentication
    const clientId = process.env.CLIENT_ID; // ÃÛ¶¹ÊÓƵ IMS client ID
    const clientSecret = process.env.CLIENT_SECRET; // ÃÛ¶¹ÊÓƵ IMS client secret
    const scopes = process.env.SCOPES; // Scope for the API access
    
    // ÃÛ¶¹ÊÓƵ IMS endpoint for obtaining an access token
    const adobeIMSV3TokenEndpointURL =
    "https://ims-na1.adobelogin.com/ims/token/v3";
    
    // Function to obtain an access token from ÃÛ¶¹ÊÓƵ IMS
    const getAccessToken = async () => {
        console.log("Getting access token from IMS"); // Log process initiation
        //console.log("Client ID: " + clientId); // Display client ID for debugging purposes
    
        // Configure the HTTP POST request to fetch the access token
        const options = {
            method: "POST",
            headers: {
            "Content-Type": "application/x-www-form-urlencoded", // Specify form data content type
            },
            // Send client ID, client secret, and scopes as the request body
            body: `grant_type=client_credentials&client_id=${clientId}&client_secret=${clientSecret}&scope=${scopes}`,
        };
    
        // Make the HTTP request to fetch the access token
        const response = await fetch(adobeIMSV3TokenEndpointURL, options);
    
        //console.log("Response status: " + response.status); // Log the HTTP status for debugging
    
        const responseJSON = await response.json(); // Parse the JSON response
    
        console.log("Access token received"); // Log success message
    
        // Return the access token
        return responseJSON.access_token;
    };
    
    // Function to retrieve metadata for a specific asset from AEM
    const getAssetMetadat = async () => {
        // Fetch the access token using the getAccessToken function
        const accessToken = await getAccessToken();
    
        console.log("Getting asset metadata from AEM");
    
        // Invoke the Assets Author API to retrieve metadata for a specific asset
        const resp = await fetch(
            `https://${bucket}.adobeaemcloud.com/adobe/../assets/${assetId}/metadata`, // Construct the URL with bucket and asset ID
            {
            method: "GET",
            headers: {
                "If-None-Match": "string", // Header to handle caching (not critical for this tutorial)
                "X-ÃÛ¶¹ÊÓƵ-Accept-Experimental": "1", // Header to enable experimental ÃÛ¶¹ÊÓƵ API features
                Authorization: "Bearer " + accessToken, // Provide the access token for authorization
                "X-Api-Key": clientId, // Include the OAuth S2S ClientId for identification
            },
            }
        );
    
        const data = await resp.json(); // Parse the JSON response
    
        console.log("Asset metadata received"); // Log success message
        console.log(data); // Display the retrieved metadata
    };
    
    // Call the getAssets function to start the process
    getAssetMetadat();
    
  6. Run the NodeJS application.

    code language-bash
    $ node src/index.js
    

API response

Upon successful execution, the API response is displayed in the console. The response contains the metadata of the specified asset.

{
  "assetId": "urn:aaid:aem:9c09ff70-9ee8-4b14-a5fa-ec37baa0d1b3",
  "assetMetadata": {
    ...
    "dc:title": "A Young Mountain Biking Couple Takes A Minute To Take In The Scenery",
    "xmp:CreatorTool": "ÃÛ¶¹ÊÓƵ Photoshop Lightroom Classic 7.5 (Macintosh)",
    ...
  },
  "repositoryMetadata": {
    ...
    "repo:name": "adobestock-221043703.jpg",
    "repo:path": "/content/dam/wknd-shared/en/adventures/cycling-southern-utah/adobestock-221043703.jpg",
    "repo:state": "ACTIVE",
    ...
  }
}

Congratulations! You have successfully invoked the OpenAPI-based AEM APIs from your custom application using OAuth Server-to-Server authentication.

Review the application code

The key callouts from the sample NodeJS application code are:

  1. IMS Authentication: Fetches an access token using OAuth Server-to-Server credentials setup in the ADC Project.

    code language-javascript
    
    // Function to obtain an access token from ÃÛ¶¹ÊÓƵ IMS
    const getAccessToken = async () => {
    
        // Configure the HTTP POST request to fetch the access token
        const options = {
            method: "POST",
            headers: {
            "Content-Type": "application/x-www-form-urlencoded", // Specify form data content type
            },
            // Send client ID, client secret, and scopes as the request body
            body: `grant_type=client_credentials&client_id=${clientId}&client_secret=${clientSecret}&scope=${scopes}`,
        };
    
        // Make the HTTP request to fetch the access token from ÃÛ¶¹ÊÓƵ IMS token endpoint https://ims-na1.adobelogin.com/ims/token/v3
        const response = await fetch(adobeIMSV3TokenEndpointURL, options);
    
        const responseJSON = await response.json(); // Parse the JSON response
    
        // Return the access token
        return responseJSON.access_token;
    };
    ...
    
  2. API Invocation: Invokes the Assets Author API to retrieve metadata for a specific asset by providing the access token for authorization.

    code language-javascript
    // Function to retrieve metadata for a specific asset from AEM
    const getAssetMetadat = async () => {
        // Fetch the access token using the getAccessToken function
        const accessToken = await getAccessToken();
    
        console.log("Getting asset metadata from AEM");
    
        // Invoke the Assets Author API to retrieve metadata for a specific asset
        const resp = await fetch(
            `https://${bucket}.adobeaemcloud.com/adobe/../assets/${assetId}/metadata`, // Construct the URL with bucket and asset ID
            {
            method: "GET",
            headers: {
                "If-None-Match": "string", // Header to handle caching (not critical for this tutorial)
                "X-ÃÛ¶¹ÊÓƵ-Accept-Experimental": "1", // Header to enable experimental ÃÛ¶¹ÊÓƵ API features
                Authorization: "Bearer " + accessToken, // Provide the access token for authorization
                "X-Api-Key": clientId, // Include the OAuth S2S ClientId for identification
            },
            }
        );
    
        const data = await resp.json(); // Parse the JSON response
    
        console.log("Asset metadata received"); // Log success message
        console.log(data); // Display the retrieved metadata
    };
    ...
    

Under the hood

Upon successful API invocation, a user representing the ADC Project’s OAuth Server-to-Server credential is created in the AEM Author service, along with the user groups that match the Product Profile and Services configuration. The technical account user is associated with the Product Profile and Services user group, which has the necessary permissions to READ the asset metadata.

To verify the technical account user and user group creation, follow these steps:

  • In the ADC Project, navigate to the OAuth Server-to-Server credential configuration. Note the Technical Account Email value.

    Technical Account Email

  • In the AEM Author service, navigate to the Tools > Security > Users and search for the Technical Account Email value.

    Technical Account User

  • Click on the technical account user to view the user details, like Groups membership. As shown below, the technical account user is associated with the AEM Assets Collaborator Users - author - Program XXX - Environment XXX and AEM Assets Collaborator Users - Service user groups.

    Technical Account User Membership

  • Note that the technical account user is associated with the AEM Assets Collaborator Users - author - Program XXX - Environment XXX Product Profile. The Product Profile is associated with the AEM Assets API Users and AEM Assets Collaborator Users Services.

    Technical Account User Product Profile

  • The Product Profile and technical account user association can be verified in the Product Profiles 's API credentials tab.

    Product Profile API credentials

403 error for non-GET requests

To READ the asset metadata, the technical account user created for the OAuth Server-to-Server credential has the necessary permissions via the Services user group (for example, AEM Assets Collaborator Users - Service).

However, to Create, Update, Delete (CUD) the asset metadata, the technical account user requires additional permissions. You can verify it by invoking the API with a non-GET request (for example, PATCH, DELETE) and observe the 403 error response.

Let’s invoke the PATCH request to update the asset metadata and observe the 403 error response.

  • Open the in the browser.

  • Enter the following values:

    table 0-row-3 1-row-3 2-row-3 3-row-3 4-row-3 5-row-3 6-row-3 7-row-3
    Section Parameter Value
    Bucket The AEM instance name without the ÃÛ¶¹ÊÓƵ domain name (.adobeaemcloud.com), for example, author-p63947-e1420428.
    Security Bearer Token Use the access token from the ADC Project’s OAuth Server-to-Server credential.
    Security X-Api-Key Use the ClientID value from the ADC Project’s OAuth Server-to-Server credential.
    Body [{ "op": "add", "path": "foo","value": "bar"}]
    Parameters assetId The unique identifier for the asset in AEM, for example, urn:aaid:aem:a200faf1-6d12-4abc-bc16-1b9a21f870da
    Parameters X-ÃÛ¶¹ÊÓƵ-Accept-Experimental *
    Parameters X-ÃÛ¶¹ÊÓƵ-Accept-Experimental 1
  • Click Send to invoke the PATCH request and observe the 403 error response.

    Invoke API - PATCH request

To fix the 403 error, you have two options:

  • In ADC Project, update the OAuth Server-to-Server credential’s associated Product Profile with an appropriate Product Profile that has the necessary permissions to Create, Update, Delete (CUD) the asset metadata, for example, AEM Administrators - author - Program XXX - Environment XXX. For more information, see How To - API’s connected credentials and Product Profile management article.

  • Using AEM Project, update the associated AEM Service user group’s (for example, AEM Assets Collaborator Users - Service) permissions in AEM Author to allow Create, Update, Delete (CUD) of the asset metadata. For more information, see How To - AEM Service user group permission management article.

Summary

In this tutorial, you learned how to invoke OpenAPI-based AEM APIs from custom applications. You enabled AEM APIs access, created and configured an ÃÛ¶¹ÊÓƵ Developer Console (ADC) project.
In the ADC Project, you added the AEM APIs, configured its authentication type, and associated the Product Profile. You also configured the AEM instance to enable ADC Project communication and developed a sample NodeJS application that calls the Assets Author API.

Additional resources

recommendation-more-help
4859a77c-7971-4ac9-8f5c-4260823c6f69