digital-masters
1/26/2026 - 3:45 PM

lock Event Enricher Tag by Stape – multiply by DM

___INFO___

{
  "type": "TAG",
  "id": "cvt_PHRV8",
  "version": 1,
  "displayName": "Event Enricher Tag by Stape – multiply by DM",
  "categories": [
    "UTILITY"
  ],
  "brand": {
    "id": "github.com_stape-io",
    "displayName": "stape-io \u0026 Digital Masters",
    "thumbnail": "\u003d\u003d"
  },
  "description": "This tag re-runs the container during an event, changes the event name, and enriches the Event Data, useful for using transformed data in triggers and variables.",
  "containerContexts": [
    "SERVER"
  ],
  "securityGroups": []
}


___TEMPLATE_PARAMETERS___

[
  {
    "type": "SIMPLE_TABLE",
    "name": "EventNameTable",
    "displayName": "Add Your Event Duplications",
    "simpleTableColumns": [
      {
        "defaultValue": "",
        "displayName": "",
        "name": "EventName",
        "type": "TEXT"
      }
    ],
    "help": "Event Name Table Each row triggers one new duplicated event with the specified EventName. \n\nAll resulting event names must be different from the incoming event name to avoid loops.  \n\nExample Incoming event: purchase Rows: purchase_copy1, purchase_copy2 → two new events are sent."
  },
  {
    "type": "CHECKBOX",
    "name": "copyCurrentEventData",
    "checkboxText": "Copy Event Data",
    "simpleValueType": true,
    "help": "Enable this option to copy the current Event Data parameters to the new event."
  },
  {
    "type": "GROUP",
    "name": "additionalEventDataParametersGroup",
    "displayName": "Additional Event Data Parameters",
    "groupStyle": "ZIPPY_OPEN_ON_PARAM",
    "subParams": [
      {
        "type": "LABEL",
        "name": "additionalEventDataParametersHelpText",
        "displayName": "Define additional Event Data parameters for the new event. If a parameter already exists in the current Event Data, its value will be replaced."
      },
      {
        "type": "SIMPLE_TABLE",
        "name": "additionalEventDataParameters",
        "simpleTableColumns": [
          {
            "defaultValue": "",
            "displayName": "Parameter name",
            "name": "name",
            "type": "TEXT"
          },
          {
            "defaultValue": "",
            "displayName": "Parameter value",
            "name": "value",
            "type": "TEXT"
          }
        ]
      }
    ]
  }
]


___SANDBOXED_JS_FOR_SERVER___

/// <reference path="./server-gtm-sandboxed-apis.d.ts" />

const getAllEventData = require('getAllEventData');
const makeString = require('makeString');
const runContainer = require('runContainer');
const logToConsole = require('logToConsole');

/*==============================================================================
==============================================================================*/

// Simple table: data.EventNameTable = [ { EventName: 'event_1' }, { EventName: 'event_2' }, ... ]
const eventNameRows = data.EventNameTable && data.EventNameTable.length
  ? data.EventNameTable
  : [];

if (!eventNameRows.length) {
  data.gtmOnSuccess();
  return;
}

// Loop protection: do not execute if any resulting event_name equals the incoming event_name
const incomingEventData = getAllEventData();
const incomingEventName = incomingEventData && incomingEventData.event_name
  ? makeString(incomingEventData.event_name)
  : '';

for (let i = 0; i < eventNameRows.length; i++) {
  if (makeString(eventNameRows[i].EventName) === incomingEventName) {
    logToConsole(
      'Event duplication aborted: resulting event_name matches incoming event_name:',
      incomingEventName
    );
    data.gtmOnSuccess();
    return;
  }
}

const additionalEventDataParameters = data.additionalEventDataParameters;

// Start with a copy of the incoming event data (if enabled), then apply overrides.
const baseEventData = data.copyCurrentEventData
  ? mergeObjects({}, incomingEventData)
  : {};

if (additionalEventDataParameters) {
  additionalEventDataParameters.forEach((d) => {
    // event_name is set per row from the table
    if (d && d.name && d.name !== 'event_name') baseEventData[d.name] = d.value;
  });
}

const eventsToSend = eventNameRows.map((row) => {
  const ev = mergeObjects({}, baseEventData);
  ev.event_name = makeString(row.EventName);
  return ev;
});

runCopiesSequentially(eventsToSend, 0, () => {
  data.gtmOnSuccess();
});

/*==============================================================================
  Helpers
==============================================================================*/

function runCopiesSequentially(events, index, done) {
  if (!events || index >= events.length) return done();
  runContainer(events[index], () => runCopiesSequentially(events, index + 1, done));
}

function mergeObjects(target, source) {
  for (const key in source) {
    if (source.hasOwnProperty(key)) target[key] = source[key];
  }
  return target;
}


___SERVER_PERMISSIONS___

[
  {
    "instance": {
      "key": {
        "publicId": "read_event_data",
        "versionId": "1"
      },
      "param": [
        {
          "key": "eventDataAccess",
          "value": {
            "type": 1,
            "string": "any"
          }
        }
      ]
    },
    "clientAnnotations": {
      "isEditedByUser": true
    },
    "isRequired": true
  },
  {
    "instance": {
      "key": {
        "publicId": "run_container",
        "versionId": "1"
      },
      "param": []
    },
    "isRequired": true
  },
  {
    "instance": {
      "key": {
        "publicId": "logging",
        "versionId": "1"
      },
      "param": [
        {
          "key": "environments",
          "value": {
            "type": 1,
            "string": "debug"
          }
        }
      ]
    },
    "isRequired": true
  }
]


___TESTS___

scenarios:
- name: New event name is passed to runContainer
  code: |-
    mock('runContainer', (eventData, onComplete, onStart) => {
      assertThat(eventData).isEqualTo({ event_name: expectedValue });
      assertThat(onComplete).isFunction();
      onComplete();
    });

    runCode(mockData);

    assertApi('gtmOnSuccess').wasCalled();
- name: Copied Event Data is passed to runContainer
  code: |-
    mockData.copyCurrentEventData = true;

    const expectedEventData = mergeObjects(expectedGetAllEventData, {
      event_name: expectedValue
    });

    mock('runContainer', (eventData, onComplete, onStart) => {
      assertThat(eventData).isEqualTo(expectedEventData);
      assertThat(onComplete).isFunction();
      onComplete();
    });

    runCode(mockData);

    assertApi('gtmOnSuccess').wasCalled();
- name: Additional Event Data Parameters are passed to runContainer
  code: |-
    // Must be set in the Event Data.
    mockData.additionalEventDataParameters = additionalEventDataParameters;

    // Expected that 'event_name' from expectedAdditionalEventDataParameters doesn't overwrite the event_name from input field.
    const expectedEventData = mergeObjects(
      { event_name: expectedValue },
      expectedAdditionalEventDataParameters
    );

    mock('runContainer', (eventData, onComplete, onStart) => {
      assertThat(eventData).isEqualTo(expectedEventData);
      assertThat(onComplete).isFunction();
      onComplete();
    });

    runCode(mockData);

    assertApi('gtmOnSuccess').wasCalled();
- name: Additional Event Data Parameters overwrite Copied Event Data and are passed
    to runContainer
  code: |-
    mockData.copyCurrentEventData = true;

    // Must be set in the Event Data and overwrite the Copied Event Data.
    mockData.additionalEventDataParameters = additionalEventDataParameters;

    const expectedEventData = mergeObjects(
      expectedGetAllEventData,
      mergeObjects(
        { event_name: expectedValue },
        expectedAdditionalEventDataParameters
      )
    );

    mock('runContainer', (eventData, onComplete, onStart) => {
      assertThat(eventData).isEqualTo(expectedEventData);
      assertThat(onComplete).isFunction();
      onComplete();
    });

    runCode(mockData);

    assertApi('gtmOnSuccess').wasCalled();
setup: |-
  function mergeObjects(target, source) {
    for (const key in source) {
      if (source.hasOwnProperty(key)) target[key] = source[key];
    }
    return target;
  }

  const expectedValue = 'test';
  const expectedClientHints = {
    architecture: 'arm',
    bitness: '64',
    full_version_list: [
      { brand: 'Chromium', version: '134.0.6998.89' },
      { brand: 'Not:A-Brand', version: '24.0.0.0' },
      { brand: 'Google Chrome', version: '134.0.6998.89' }
    ],
    mobile: false,
    model: '',
    platform: 'macOS',
    platform_version: '15.2.0',
    wow64: false,
    brands: [
      { brand: 'Chromium', version: '134' },
      { brand: 'Not:A-Brand', version: '24' },
      { brand: 'Google Chrome', version: '134' }
    ]
  };

  const expectedGetAllEventData = {
    page_location: expectedValue,
    page_referrer: expectedValue,
    client_hints: expectedClientHints
  };
  mock('getAllEventData', expectedGetAllEventData);

  const additionalEventDataParameters = [
    { name: 'event_name', value: 'must_not_be_set_as_event_name' },
    { name: 'page_location', value: expectedValue },
    { name: 'foo', value: expectedValue },
    { name: 'another_obj', value: { foo: expectedValue } }
  ];
  const expectedAdditionalEventDataParameters = {
    page_location: expectedValue,
    foo: expectedValue,
    another_obj: { foo: expectedValue }
  };

  const mockData = {
    newEventName: expectedValue
  };


___NOTES___

Created on 3/21/2025, 10:18:42 AM