Skip to content

Minimal MarTech stack for AEM that includes the WebSDK, Target/AJO, Analytics, Launch & ACDL

License

Notifications You must be signed in to change notification settings

adobe-rnd/aem-martech

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

53 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

🚧 This is an early access technology and is still heavily in development. Reach out to us over Slack before using it.

AEM Edge Delivery Services Marketing Technology

The AEM Marketing Technology plugin helps you quickly set up a complete MarTech stack for your AEM project. It is currently available to customers in collaboration with AEM Engineering via co-innovation VIP Projects. To implement your use cases, please reach out to the AEM Engineering team in the Slack channel dedicated to your project.

Features

The AEM MarTech plugin is essentially a wrapper around the Adobe Experience Platform WebSDK (v2.19.2) and the Adobe Client Data Layer (v2.0.2), and that can seamlessly integrate your website with:

  • 🎯 Adobe Target or Adobe Journey Optimizer: to personalize your pages
  • 📊 Adobe Analytics: to track customer journey data
  • 🚩 Adobe Experience Platform Tags (a.k.a. Launch): to track your custom events

It's key differentiator are:

  • 🌍 Experience Platform enabled: the library fully integrates with our main Adobe Experience Platform and all the services of our ecosystem
  • 🚀 extremely fast: the library is optimized to reduce load delay, TBT and CLS, and has minimal impact on your Core Web Vitals
  • 👤 privacy-first: the library does not track end users by default, and can easily be integrated with your preferred consent management system to open up more advanced use cases

Prerequisites

You need to have access to:

  • Adobe Experience Platform (AEP)
  • Adobe Analytics
  • Adobe Target or Adobe Journey Optimizer

And you need to have preconfigured:

  • a datastream in AEP with Adobe Analytics, and Adobe Target or Adobe Journey Optimizer configured
  • an Adobe Experience Platform Tag (Launch) container with the Adobe Client Data Layer extensions at a minimum

We also recommend using a proper consent management system. If not, make sure to default the consent to in so you don't block out personalization use cases.

Installation

Add the plugin to your AEM project by running:

git subtree add --squash --prefix plugins/martech [email protected]:adobe-rnd/aem-martech.git main

If you later want to pull the latest changes and update your local copy of the plugin

git subtree pull --squash --prefix plugins/martech [email protected]:adobe-rnd/aem-martech.git main

If you prefer using https links you'd replace [email protected]:adobe-rnd/aem-martech.git in the above commands by https://github.com/adobe-rnd/aem-martech.git.

If the subtree pull command is failing with an error like:

fatal: can't squash-merge: 'plugins/martech' was never added

you can just delete the folder and re-add the plugin via the git subtree add command above.

If you use some ELint at the project level (or equivalent), make sure to update ignore minified files in your .eslintignore:

*.min.js

Project instrumentation

To properly connect and configure the plugin for your project, you'll need to edit both the head.html and scripts.js in your AEM project and add the following:

  1. Add preload hints for the dependencies we need to speed up the page load at the end of your head.html:
    <link rel="preload" as="script" crossorigin="anonymous" href="/plugins/martech/src/index.js"/>
    <link rel="preload" as="script" crossorigin="anonymous" href="/plugins/martech/src/alloy.min.js"/>
    <link rel="preconnect" href="https://edge.adobedc.net"/>
    <!-- change to adobedc.demdex.net if you enable third party cookies -->
  2. Import the various plugin methods at the top of your scripts.js file:
    import {
      initMartech,
      updateUserConsent,
      martechEager,
      martechLazy,
      martechDelayed,
    } from '../plugins/martech/src/index.js';
  3. Configure the plugin at the top of the loadEager method:
    /**
     * loads everything needed to get to LCP.
    */
    async function loadEager(doc) {
      const isConsentGiven = /* hook in your consent check here to make sure you can run personalization use cases. */;
      const martechLoadedPromise = initMartech(
        // The WebSDK config
        // Documentation: https://experienceleague.adobe.com/en/docs/experience-platform/web-sdk/commands/configure/overview#configure-js
        {
          datastreamId: /* your datastream id here, formally edgeConfigId */,
          orgId: /* your ims org id here */,
          onBeforeEventSend: (payload) => {
            // set custom Target params 
            // see doc at https://experienceleague.adobe.com/en/docs/platform-learn/migrate-target-to-websdk/send-parameters#parameter-mapping-summary
            payload.data.__adobe.target ||= {};
    
            // set custom Analytics params
            // see doc at https://experienceleague.adobe.com/en/docs/analytics/implementation/aep-edge/data-var-mapping
            payload.data.__adobe.analytics ||= {};
          },
    
          // set custom datastream overrides
          // see doc at:
          // - https://experienceleague.adobe.com/en/docs/experience-platform/web-sdk/commands/datastream-overrides
          // - https://experienceleague.adobe.com/en/docs/experience-platform/datastreams/overrides
          edgeConfigOverrides: {
            // Override the datastream id
            // datastreamId: '...'
    
            // Override AEP event datasets
            // com_adobe_experience_platform: {
            //   datasets: {
            //     event: {
            //       datasetId: '...'
            //     }
            //   }
            // },
    
            // Override the Analytics report suites
            // com_adobe_analytics: {
            //   reportSuites: ['...']
            // },
    
            // Override the Target property token
            // com_adobe_target: {
            //   propertyToken: '...'
            // }
          },
        },
        // The library config
        {
          launchUrls: [/* your Launch script URLs here */],
          personalization: !!getMetadata('target') && isConsentGiven,
        },
      );
      
    }
    Note that:
    • the WebSDK context flag will, by default, track the web, device and environment details
    • the WebSDK debugEnabled flag will, by default, be set to true on localhost and any .page URL
    • the WebSDK defaultConsent is set to pending to avoid tracking any sensitive information by default. This will also prevent personalization to properly run unless consent is explicitly given
    • we recommend enabling personalization only if needed to limit the performance impact, and only if consent has been given by the user to be compliant with privacy laws. We typically recommend using a page metadata flag for the former, and integrating with your preferred consent management system APIs for the latter.
  4. Adjust your loadEager method so it waits for the martech to load and personalize the page:
    /**
     * loads everything needed to get to LCP.
    */
    async function loadEager(doc) {
      
      if (main) {
        decorateMain(main);
        await Promise.all([
          martechLoadedPromise.then(martechEager),
          waitForLCP(LCP_BLOCKS),
        ]);
      }
    }
  5. Add a reference to the lazy logic just above the sampleRUM('lazy'); call in your loadLazy method:
    async function loadLazy(doc) {
      
      await martechLazy();
      sampleRUM('lazy');
      
    }
  6. Add a reference to the delayed logic in the loadDelayed method:
    function loadDelayed() {
      // eslint-disable-next-line import/no-cycle
      window.setTimeout(() => {
        martechDelayed();
        return import('./delayed.js');
      }, 3000);
    }
  7. Connect your consent management system to track when user consent is explicitly given. Typically call the updateUserConsent with a set of categories & booleans pairs once your consent management sends the event. The marketing option supports granular control as in the official documentation:
    updateUserConsent({
      collect: true,
      marketing: {
        preferred: 'email',
        any: false,
        email: true,
        push: false,
        sms: true,
      },
      personalize: true,
      share: true,
    })

⚠️ Note that the integration will be specific to the vendor you chose. See some examples below.

Integrating with consent management solutions

AEM Consent Banner Block

Here is an example for the consent banner block in AEM Block Collection:

function consentEventHandler(ev) {
  const collect = ev.detail.categories.includes('CC_ANALYTICS');
  const marketing = ev.detail.categories.includes('CC_MARKETING');
  const personalize = ev.detail.categories.includes('CC_TARGETING');
  const share = ev.detail.categories.includes('CC_SHARING');
  updateUserConsent({ collect, marketing, personalize, share });
}
window.addEventListener('consent', consentEventHandler);
window.addEventListener('consent-updated', consentEventHandler);

Integrating with OneTrust

Here is an example for OneTrust:

function consentEventHandler(ev) {
 const groups = ev.detail;
 const collect = groups.includes('C0002'); // Performance Cookies
 const personalize = groups.includes('C0003'); // Functional Cookies
 const share = groups.includes('C0008'); // Targeted Advertising and Selling/Sharing of Personal Information
 updateUserConsent({ collect, personalize, share });
}
window.addEventListener('consent.onetrust', consentEventHandler);

Automatic forwarding of events

The library is built to automatically foward all events that are pushed to the datalayer directly to Analyics/CJS. So all you have to do is:

window.adobeDatalayer.push({
  xdm: { ... }, // the XDM schema to push
  data: { ... }, // The data mappings to use
  configOverrides: { ... }, // The possible edge config overrides, like datastreamId overrides
})

you can also just use our helper methods for this, that simplifies the input a bit:

pushEventToDataLayer('my-event', xdm, data, configOverrides);

or directly leverage:

pushToDataLayer({
  xdm: { ... }, // the XDM schema to push
  data: { ... }, // The data mappings to use
  configOverrides: { ... }, // The possible edge config overrides, like datastreamId overrides
})

which is just a proxy for window.adobeDatalayer.push.

⚠️ When working with overrides, the onBeforeEventSend hook does not directly support those, so you have to handle those in the outer call to one of the methods above.

Working with SPA and dynamic content

If your page is built dynamically with content that renders either asynchronously or updates based on user input, it is likely that the default personalization applied will not be enough and you can end up with 2 possible edge cases:

  1. The personalization is not applied at all
  2. The personalization is incorrectly applied

This is typical of a concurrency issue between personalization being applied and the content of the page not being ready yet and still updating. Some examples are:

  • blocks with lose promises that resolve after block decoration is finished
  • frontend frameworks like (p)react that update the DOM based on internal state changes to a virtual DOM

For those cases, we typically recommend:

  1. Set up personalization following the SPA approach and leverage views for the dynamic parts
  2. Import the 2 helper methods from our plugin in the blocks that represent these views:
    import {
      isPersonalizationEnabled,
      getPersonalizationForView,
      applyPersonalization,
    } from '../plugins/martech/src/index.js';
  3. Fetch the personalization once for the view when it is initially rendered, or whenever you change the page in a paginated component:
    await getPersonalizationForView('my-view');
    We recommend directly including this in the loadEager logic so the default view immediately get the Target propositions:
    /**
     * loads everything needed to get to LCP.
    */
    async function loadEager(doc) {
      
      if (main) {
        decorateMain(main);
        await Promise.all([
          martechLoadedPromise.then(martechEager),
          waitForLCP(LCP_BLOCKS),
        ]);
        if (isPersonalizationEnabled()) {
          getPersonalizationForView('my-view');
        }
      }
    }
    You can for instance map the page template to a given view name, or have the view name directly set in the page metadata.
  4. Apply the personalization every time there is a meaningful DOM update done by your block or component:
    applyPersonalization('my-view');

Custom plugin options

There are various aspects of the plugin that you can configure via options you can pass to the initMartech method above. Here is the full list we support:

initMartech(
  // Documentation: https://experienceleague.adobe.com/en/docs/experience-platform/web-sdk/commands/configure/overview#configure-js
  {
    datastreamId: '...', // the Datastream ID you want to report to
    orgId: '...', // your IMS organisation ID,
    // See https://experienceleague.adobe.com/en/docs/experience-platform/web-sdk/commands/configure/overview for other options
  }
  // The library config
  {
    analytics: true, // whether to track data in Adobe Analytics (AA)
    alloyInstanceName: 'alloy', // the name of the global WebSDK instance
    dataLayer: true, // whether to use the Adobe Client Data Layer (ACDL)
    dataLayerInstanceName: 'adobeDataLayer', // the name of the global ACDL instance
    includeDataLayerState: true, // whether to include the whole data layer state on every event sent
    launchUrls: [], // the list of Launch scripts to load
    personalization: true, // whether to apply page personalization from Adobe Target (AT) or Adobe Journey Optimizer (AJO)
    performanceOptimized: true, // whether to use the agressive performance optimized approach or more traditional
    personalizationTimeout: 1000, // the amount of time to wait (in ms) before bailing out and continuing page rendering
  },
);

Integrating Adobe EDS RUM events with Adobe Analytics

The library also exposes a few helper methods to let you quickly integrate default RUM events with your Adobe Analytics solution.

  1. Create a new rum-to-analytics.js file:
    import { sampleRUM } from './aem.js';
    import { initRumTracking, pushEventToDataLayer } from '../plugins/martech/src/index.js';
    
    // Define RUM tracking function
    const track = initRumTracking(sampleRUM, { withRumEnhancer: true });
    
    // Track page views when the page is fully rendered
    // The data will be automatically enriched with applied propositions for personalization use cases
    track('lazy', () => {
      pushEventToDataLayer(
        'web.webpagedetails.pageViews', {
          web: {
            webPageDetails: {
              pageViews: { value: 1 },
              isHomePage: window.location.pathname === '/',
            },
          },
        },
        {
          __adobe: {
            analytics: {
              // see documentation at https://experienceleague.adobe.com/en/docs/analytics/implementation/aep-edge/data-var-mapping
            },
          },
        });
    });
    
    track('click', ({source, target}) => {
      pushEventToDataLayer('web.webinteraction.linkClicks', {
        web: {
          webInteraction: {
            URL: target,
            name: source,
            linkClicks: { value: 1 },
            type: target && new URL(target).origin !== window.location.origin
              ? 'exit'
              : 'other',
          },
        },
      });
    });
    
    // see documentation at https://www.aem.live/developer/rum#checkpoints for other events you can track
  2. Load the rum-to-analytics.js file before calling martechLazy() in your loadLazy method:
    async function loadLazy(doc) {
      
      await import('./rum-to-analytics.js');
      await martechLazy();
      sampleRUM('lazy');
      
    }

FAQ

Why shouldn't I use the default Adobe Tag/Launch approach to do all of this?

Typical instrumentations based on a centralized approach using Adobe Tag/Launch that is loaded early in the page life-cycle essentially impacts the user experience negatively for the benefit of marketing metrics. Core Web Vitals are noticeably impacted, and Google PageSpeed reports typically show a drop of 20~40 points in the performance category.

But can't I just defer the launch script to solve this?

This can indeed solve the issue in some cases, but comes with some drawbacks:

  1. personalization use cases will be delayed as well, so you'll introduce content flickering when the personalization kicks in
  2. analytics metrics gathering will be delayed as well, so if your website has a large percentage of early bounces, your analytics reports won't be able to capture those

What guarantee do I have that your approach will not break …some feature name…?

Adobe Tags/Launch will typically wrap the Adobe Experience Platform SDK and Adobe Client Data Layer, and configure your data streams to connect to Adobe Target and Adobe Analytics.

Our approach just extracts those key elements from the Launch container so we can instrument those selectively at the right time in the page load for optimal performance, but we still leverage the official documented APIs and configurations as Adobe Launch would.

We are basically building on top of:

  • Top and bottom of page events so we can enable personalization early in the page load, and wait for the page to fully render to report metrics
  • Data object variable mapping so we can gather key page metadata for your page in Adobe Analytics
  • Adobe Launch to trigger additional rules based on data elements in a delayed manner so we still support marketing use cases you'd expect to cover via Adobe Launch alone

On top of this, we also fine-tuned the code to:

  • avoid content flicker as the DOM is dynamically rendered to support AEM EDS and/or SPA use cases
  • dynamically load personalization and data layer dependencies only when needed

So what's the catch?

Since we essentially just split up the Adobe Launch container to execute it in a more controlled way, not all features can be controlled from the Launch UI, and some of the logic moves to the project code.

Also, some default Adobe Launch extensions won't work directly with such a setup. We recommend the following baseline:

  • Core Extension
  • Adobe Client Data Layer: so you can react to data layer events
  • AA via AEP Web SDK: so you can have rules setting variables, product strings and send beacons
  • (optionally) Mapping Table: so you can remap selected values in your data elements

About

Minimal MarTech stack for AEM that includes the WebSDK, Target/AJO, Analytics, Launch & ACDL

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published