🚧 This is an early access technology and is still heavily in development. Reach out to us over Slack before using it.
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.
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
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.
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
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:
- 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 -->
- 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';
- Configure the plugin at the top of the
loadEager
method:Note that:/** * 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, }, ); … }
- the WebSDK
context
flag will, by default, track theweb
,device
andenvironment
details - the WebSDK
debugEnabled
flag will, by default, be set totrue
on localhost and any.page
URL - the WebSDK
defaultConsent
is set topending
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.
- the WebSDK
- 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), ]); } }
- Add a reference to the lazy logic just above the
sampleRUM('lazy');
call in yourloadLazy
method:async function loadLazy(doc) { … await martechLazy(); sampleRUM('lazy'); … }
- 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); }
- 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, })
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);
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);
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
.
onBeforeEventSend
hook does not directly support those, so you have to handle those in the outer call to one of the methods above.
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:
- The personalization is not applied at all
- 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:
- Set up personalization following the SPA approach and leverage views for the dynamic parts
- 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';
- Fetch the personalization once for the view when it is initially rendered, or whenever you change the page in a paginated component:
We recommend directly including this in the
await getPersonalizationForView('my-view');
loadEager
logic so the default view immediately get the Target propositions:You can for instance map the page template to a given view name, or have the view name directly set in the page metadata./** * 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'); } } }
- Apply the personalization every time there is a meaningful DOM update done by your block or component:
applyPersonalization('my-view');
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
},
);
The library also exposes a few helper methods to let you quickly integrate default RUM events with your Adobe Analytics solution.
- 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
- Load the
rum-to-analytics.js
file before callingmartechLazy()
in yourloadLazy
method:async function loadLazy(doc) { … await import('./rum-to-analytics.js'); await martechLazy(); sampleRUM('lazy'); … }
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.
This can indeed solve the issue in some cases, but comes with some drawbacks:
- personalization use cases will be delayed as well, so you'll introduce content flickering when the personalization kicks in
- 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
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
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