___TERMS_OF_SERVICE___ By creating or modifying this file you agree to Google Tag Manager's Community Template Gallery Developer Terms of Service available at https://developers.google.com/tag-manager/gallery-tos (or such other URL as Google may provide), as modified from time to time. ___INFO___ { "type": "TAG", "id": "cvt_temp_public_id", "version": 1, "securityGroups": [], "displayName": "MoEngage", "categories": ["MARKETING"], "brand": { "id": "brand_dummy", "displayName": "", "thumbnail": "\u003d\u003d" }, "description": "Tag that sends events to MoEngage from Snowplow, GAv4 or other clients.", "containerContexts": [ "SERVER" ] } ___TEMPLATE_PARAMETERS___ [ { "type": "SELECT", "name": "dataCenter", "displayName": "Select data center", "macrosInSelect": false, "selectItems": [ { "value": 1, "displayValue": "Dashboard-01 (DC-01)" }, { "value": 2, "displayValue": "Dashboard-02 (DC-02)" }, { "value": 3, "displayValue": "Dashboard-03 (DC-03)" }, { "value": 4, "displayValue": "Dashboard-04 (DC-04)" } ], "simpleValueType": true, "alwaysInSummary": true, "help": "You can find more details about data centers here: https://help.moengage.com/hc/en-us/articles/360057030512-Data-Centers-in-MoEngage", "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "POSITIVE_NUMBER" } ] }, { "type": "TEXT", "name": "apiID", "displayName": "API ID", "simpleValueType": true, "alwaysInSummary": true, "help": "The API ID for your MoEngage account is available on the MoEngage Dashboard \u003e\u003e Settings \u003e\u003e APIs \u003e\u003e General Settings \u003e\u003e APP ID", "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "type": "TEXT", "name": "apiKey", "displayName": "API Key", "simpleValueType": true, "alwaysInSummary": true, "help": "The API Key for your MoEngage account is available on the MoEngage Dashboard \u003e\u003e Settings \u003e\u003e APIs \u003e\u003e General Settings \u003e\u003e APP Key", "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "type": "GROUP", "name": "userIdentifier", "displayName": "User Identifier", "groupStyle": "ZIPPY_OPEN", "subParams": [ { "type": "SELECT", "name": "userIdentifierType", "displayName": "", "macrosInSelect": false, "selectItems": [ { "value": "userIdentifierVariable", "displayValue": "From Variable" }, { "value": "userIdentifierEvent", "displayValue": "From Event Property" } ], "simpleValueType": true, "defaultValue": "userIdentifierVariable" }, { "type": "TEXT", "name": "userID", "displayName": "", "simpleValueType": true, "alwaysInSummary": true, "valueValidators": [ { "type": "NON_EMPTY" } ] } ], "help": "Select the identifier to map users in MoEngage" }, { "type": "GROUP", "name": "snowplowEventMapping", "displayName": "Snowplow Event Mapping Options", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "GROUP", "name": "atomicProp", "displayName": "Snowplow atomic properties", "groupStyle": "ZIPPY_OPEN", "subParams": [ { "type": "CHECKBOX", "name": "includeAllAtomicProperties", "checkboxText": "Include all atomic properties", "simpleValueType": true, "defaultValue": true, "alwaysInSummary": true } ] }, { "type": "GROUP", "name": "selfDescribingEvent", "displayName": "Snowplow Self Describing Event", "groupStyle": "ZIPPY_OPEN", "subParams": [ { "type": "CHECKBOX", "name": "includeSelfDescribingEvent", "checkboxText": "Include Self Describing event", "simpleValueType": true, "help": "Indicates if a Snowplow Self Describing event should be in the event object.", "defaultValue": true, "alwaysInSummary": true } ] }, { "type": "GROUP", "name": "entityRules", "displayName": "Snowplow Event Context Rules", "groupStyle": "ZIPPY_OPEN", "subParams": [ { "type": "SELECT", "name": "includeEntities", "displayName": "Include Snowplow Entities in event object", "macrosInSelect": false, "selectItems": [ { "value": "all", "displayValue": "All" }, { "value": "none", "displayValue": "None" } ], "simpleValueType": true, "defaultValue": "none" }, { "type": "SIMPLE_TABLE", "name": "entityMappingRules", "displayName": "Snowplow Entities to Add/Edit mapping", "simpleTableColumns": [ { "defaultValue": "", "displayName": "Entity Name", "name": "key", "type": "TEXT", "valueHint": "The entity name from the original event (e.g. `web_page` from `iglu:com.snowplowanalytics.snowplow/web_page/jsonschema/1-0-0`)", "isUnique": true, "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "defaultValue": "event_object", "displayName": "Include in event object or user attributes object", "name": "propertiesObjectToPopulate", "type": "SELECT", "selectItems": [ { "value": "event_object", "displayValue": "event_object" }, { "value": "user_attributes_object", "displayValue": "user_attributes_object" } ], "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "defaultValue": "control", "displayName": "Apply to all versions", "name": "version", "type": "SELECT", "selectItems": [ { "value": "control", "displayValue": "False" }, { "value": "free", "displayValue": "True" } ], "valueValidators": [ { "type": "NON_EMPTY" } ] } ], "alwaysInSummary": false, "help": "Specify the Entity name and append `x-sp-` in all the entities before mapping." }, { "type": "SIMPLE_TABLE", "name": "entityExclusionRules", "displayName": "Snowplow Entities to Exclude", "simpleTableColumns": [ { "defaultValue": "", "displayName": "Entity Name", "name": "key", "type": "TEXT", "isUnique": true, "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "defaultValue": "control", "displayName": "Apply to all versions", "name": "version", "type": "SELECT", "selectItems": [ { "value": "control", "displayValue": "False" }, { "value": "free", "displayValue": "True" } ], "valueValidators": [ { "type": "NON_EMPTY" } ] } ], "enablingConditions": [ { "paramName": "includeEntities", "paramValue": "all", "type": "EQUALS" } ], "help": "Specify the Entity name you want to exclude from the MoEngage event. Additionally, you can also set whether the exclusion applies to all versions of the entity." } ] } ] }, { "type": "GROUP", "name": "otherEventMapping", "displayName": "Additional Event Mapping Options", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "GROUP", "name": "commonEventPropertyRules", "displayName": "Event Property Rules", "groupStyle": "ZIPPY_OPEN", "subParams": [ { "type": "CHECKBOX", "name": "includeCommonEventProperties", "checkboxText": "Include common event properties", "simpleValueType": true, "alwaysInSummary": true, "defaultValue": true, "help": "Include the event properties from the common event definition in MoEngage event object." }, { "type": "SIMPLE_TABLE", "name": "eventMappingRules", "displayName": "Additional Event Property Mapping Rules", "simpleTableColumns": [ { "defaultValue": "", "displayName": "Event Property Key", "name": "key", "type": "TEXT" }, { "defaultValue": "", "displayName": "MoEngage Mapped Key (optional)", "name": "mappedKey", "type": "TEXT" } ], "alwaysInSummary": false, "help": "Specify the Property Key from the GTM Event, and then key you could like to map it to or leave the mapped key blank to keep the same name. These keys will populate the MoEngage event object." } ] }, { "type": "GROUP", "name": "commonUserPropertyRules", "displayName": "User Property Rules", "groupStyle": "ZIPPY_OPEN", "subParams": [ { "type": "CHECKBOX", "name": "includeCommonUserProperties", "checkboxText": "Include common user properties", "simpleValueType": true, "alwaysInSummary": true, "defaultValue": true, "help": "Include the user_data properties from the common event definition as MoEngage user attributes." }, { "type": "SIMPLE_TABLE", "name": "userMappingRules", "displayName": "Additional User Property Mapping Rules", "simpleTableColumns": [ { "defaultValue": "", "displayName": "User Property Key", "name": "key", "type": "TEXT" }, { "defaultValue": "", "displayName": "MoEngage Mapped Key (optional)", "name": "mappedKey", "type": "TEXT" } ], "alwaysInSummary": false, "help": "Specify the Property Key from the GTM Event, and the key you could like to map it to or leave the mapped key blank to keep the same name. These keys will populate the MoEngage user attributes object." } ] } ] }, { "type": "GROUP", "name": "additionalUnmappedEventData", "displayName": "Additional Event Data", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "SIMPLE_TABLE", "name": "additionalEventKeyValue", "displayName": "Additional event properties mapping", "simpleTableColumns": [ { "defaultValue": "", "displayName": "Key", "name": "key", "type": "TEXT" }, { "defaultValue": "", "displayName": "Value", "name": "value", "type": "TEXT" } ], "help": "Directly send these key value pairs in the payload using GTM variables" } ], "help": "Directly send these key value pairs in the payload using GTM variables" }, { "type": "GROUP", "name": "additionalUnmappedUserData", "displayName": "Additional User Data", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "SIMPLE_TABLE", "name": "additionalUserKeyValue", "displayName": "Additional user attributes mapping", "simpleTableColumns": [ { "defaultValue": "", "displayName": "Key", "name": "key", "type": "TEXT" }, { "defaultValue": "", "displayName": "Value", "name": "value", "type": "TEXT" } ], "help": "Directly send these key value pairs in the payload using GTM variables" } ], "help": "Directly send these key value pairs in the payload using GTM variables" }, { "type": "GROUP", "name": "advancedProperties", "displayName": "Advanced Event Settings", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "TEXT", "name": "eventNameOverride", "displayName": "Event Name Override", "simpleValueType": true, "help": "Use this option to override the name of the MoEngage event object. Leave blank to inherit from common event properties." }, { "type": "TEXT", "name": "eventTime", "displayName": "Event time property", "simpleValueType": true, "help": "Specify the client event property to populate the event time (in ISO-8601 format) or leave it empty to use the current time." } ] }, { "type": "GROUP", "name": "logsGroup", "displayName": "Log settings", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "RADIO", "name": "logType", "displayName": "", "radioItems": [ { "value": "no", "displayValue": "Do not log" }, { "value": "debug", "displayValue": "Log to console during debug and preview" }, { "value": "always", "displayValue": "Always log to console" } ], "simpleValueType": true, "defaultValue": "debug" } ] } ] ___SANDBOXED_JS_FOR_SERVER___ const getAllEventData = require('getAllEventData'); const getEventData = require('getEventData'); const getContainerVersion = require('getContainerVersion'); const getRequestHeader = require('getRequestHeader'); const toBase64 = require('toBase64'); const makeTableMap = require('makeTableMap'); const getType = require('getType'); const getTimestampMillis = require('getTimestampMillis'); const sendHttpRequest = require('sendHttpRequest'); const logToConsole = require('logToConsole'); const JSON = require('JSON'); const Object = require('Object'); const Math = require('Math'); const createRegex = require('createRegex'); // **************************** Constants Start **************************** const tagName = 'MoEngage'; // **************************** Constants End **************************** // **************************** Snowplow Helpers Start **************************** /** * Returns whether a string is upper case. * * @param {string} value - The string to check * @returns {boolean} */ const isUpper = (value) => { return value === value.toUpperCase() && value !== value.toLowerCase(); }; /** * Converts a string to snake case. * * @param {string} value - The string to convert * @returns {string} The converted string */ const toSnakeCase = (value) => { let result = ''; let previousChar; for (var i = 0; i < value.length; i++) { let currentChar = value.charAt(i); if (isUpper(currentChar) && i > 0 && previousChar !== '_') { result = result + '_' + currentChar; } else { result = result + currentChar; } previousChar = currentChar; } return result; }; /** * Merges objects. * * @param {Object[]} args - The array of objects to merge * @returns {Object} The resulting object */ const mergeObjects = (args) => { let target = {}; const addToTarget = (obj) => { for (let prop in obj) { if (obj.hasOwnProperty(prop)) { target[prop] = obj[prop]; } } }; for (let i = 0; i < args.length; i++) { addToTarget(args[i]); } return target; }; /** * Returns object keys in an array * * @param Object - Object * @returns Array - List of keys */ const getKeysOf = (obj) => { const keys = []; for (let prop in obj) { if (obj.hasOwnProperty(prop)) { keys.push(prop); } } return keys; }; /** * Flattens a nested object to root level * For same key last occurence is given as final value * @example * // returns { a: [1,2,3] } * flattenObject({ a: [1,2,3] }); * @example * // returns { d:3,e:2,b:4 } * flattenObject({a:{b:{c:{d:1,e:2}, d: 3}},b: 4}); * @example * // returns { d:[1,2,3,4],e:2,b:4 } * flattenObject({a:{b:{c:{d:[1,2,3,4],e:2}}},b: 4}); * @param Object - Object * @returns Object - Flattened object */ const flattenObject = (obj) => { if (obj) { return getKeysOf(obj).reduce((acc, k) => { if ( getType(obj[k]) === 'object' || ( getType(obj[k]) === 'array' && getType(obj[k][0]) !== 'number' && getType(obj[k][0]) !== 'string' ) ) { acc = mergeObjects([acc, flattenObject(obj[k], k)]); } else { acc[k] = obj[k]; } return acc; }, {}); } return {}; }; /** * Filters out invalid rules to avoid unintended behavior. * (e.g. version control being ignored if version num is not included in name) * Assumes that a rule contains 'key' and 'version' properties. * * @param {Object[]} rules - The array of rules * @returns {Object[]} The valid rules */ const cleanRules = (rules) => { const lastNumRexp = createRegex('[0-9]$'); return rules.filter((row) => { if (row.version === 'control') { return !!row.key.match(lastNumRexp); } return true; }); }; /** * Parses a Snowplow schema to the expected major version format, * also prefixed so as to match the contexts' output of the Snowplow Client. * * @param schema {string} - The input schema * @returns {string} The expected output client event property */ const parseSchemaToMajorKeyValue = (schema) => { if (schema.indexOf('x-sp-contexts_') === 0) return schema; if (schema.indexOf('contexts_') === 0) return 'x-sp-' + schema; if (schema.indexOf('iglu:') === 0) { const rexp = createRegex('[./]', 'g'); let fixed = schema .replace('iglu:', '') .replace('jsonschema/', '') .replace(rexp, '_'); for (let i = 0; i < 2; i++) { fixed = fixed.substring(0, fixed.lastIndexOf('-')); } return 'x-sp-contexts_' + toSnakeCase(fixed).toLowerCase(); } return schema; }; /** * Removes the major version part from a schema reference if exists. * @example * // returns 'com_acme_test' * mkVersionFree('com_acme_test_1') * @example * // returns 'com_acme_test' * mkVersionFree('com_acme_test') * * @param {string} schemaRef - The schema * @returns {string} */ const mkVersionFree = (schemaRef) => { const versionRexp = createRegex('_[0-9]+$'); return schemaRef.replace(versionRexp, ''); }; /** * Cleans a name from the GTM-SS Snowplow prefix ('x-sp-'). * * @param {string} prop - The property name * @returns {string} The property name with the GTM-SS Snowplow prefix removed. */ const cleanPropertyName = (prop) => prop.replace('x-sp-', ''); /** * Parses the entity inclusion rules from the tag configuration. * * @param {Object} tagConfig - The tag configuration * @returns {Object[]} */ const parseEntityRules = (tagConfig) => { const rules = tagConfig.entityMappingRules; if (rules) { const validRules = cleanRules(rules); const parsedRules = validRules.map((row) => { const parsedKey = parseSchemaToMajorKeyValue(row.key); return { ref: row.version === 'control' ? parsedKey : mkVersionFree(parsedKey), parsedKey: parsedKey, mappedKey: cleanPropertyName(parsedKey), target: row.propertiesObjectToPopulate, version: row.version, }; }); return parsedRules; } return []; }; /** * Parses the entity exclusion rules from the tag configuration. * * @param {Object} tagConfig - The tag configuration * @returns {Object[]} */ const parseEntityExclusionRules = (tagConfig) => { const rules = tagConfig.entityExclusionRules; if (rules) { const validRules = cleanRules(rules); const excludedEntities = validRules.map((row) => { const entityRef = parseSchemaToMajorKeyValue(row.key); return { ref: row.version === 'control' ? entityRef : mkVersionFree(entityRef), version: row.version, }; }); return excludedEntities; } return []; }; /** * Given a list of entity references and an entity name, * returns the index of a matching reference. * Matching reference means whether the entity name starts with ref. * @example * // returns 0 * getReferenceIdx('com_test_test_1', ['com_test_test_1']); * @example * // returns 0 * getReferenceIdx('com_test_test_1', ['com_test_test']); * @example * // returns -1 * getReferenceIdx('com_test_test_1', ['com_test_test_2']); * @example * // returns -1 * getReferenceIdx('com_test_test', ['com_test_test_fail']); * @example * // returns -1 * getReferenceIdx('com_test_test_fail', ['com_test_test']); * * @param {string} entity - The entity name to match * @param {string[]} refsList - An array of references * @returns {integer} */ const getReferenceIdx = (entity, refsList) => { const versionFreeEntity = mkVersionFree(entity); for (let i = 0; i < refsList.length; i++) { const okControl = entity.indexOf(refsList[i]) === 0; const okFree = versionFreeEntity === mkVersionFree(refsList[i]); if (okControl && okFree) { return i; } } return -1; }; /** * Given the inclusion rules and the excluded entity references, * returns the final entity mapping rules. * * @param {Object} tagConfig - The tag configuration * @returns {Object[]} */ const finalizeEntityRules = (inclusionRules, excludedRefs) => { const finalEntities = inclusionRules.filter((row) => { const refIdx = getReferenceIdx(row.ref, excludedRefs); return refIdx < 0; }); return finalEntities; }; /** * Returns whether a property name is a Snowplow self-describing event property. * * @param {string} prop - The property name * @returns {boolean} */ const isSpSelfDescProp = (prop) => { return prop.indexOf('x-sp-self_describing_event_') === 0; }; /** * Returns whether a property name is a Snowplow context/entity property. * * @param {string} prop - The property name * @returns {boolean} */ const isSpContextsProp = (prop) => { return prop.indexOf('x-sp-contexts_') === 0; }; /** * Returns whether a property name is a Snowplow event property. * * @param {string} prop - The property name * @returns {boolean} */ const isSpEventProp = (prop) => { const excludeKeys = [ 'x-sp-tp2', 'x-sp-contexts', 'x-sp-self_describing_event', ]; return prop.indexOf('x-sp-') === 0 && excludeKeys.indexOf(prop) < 0; }; /** * Returns whether a property name is a Snowplow atomic property. * * @param {string} prop - The property name * @returns {boolean} */ const isSpAtomicProp = (prop) => { return ( isSpEventProp(prop) && !isSpSelfDescProp(prop) && !isSpContextsProp(prop) ); }; /** * Modifies the respective objects to populate according to Snowplow Event Context Rules. * * @param {Object} eventData - The client event object * @param {Object} tagConfig - The tag configuration object * @param {Object} eventProperties - The object to populate for `event_object` * @param {Object} userAttributes - The object to populate for `user_attributes_object` * @returns {undefined} */ const parseCustomEventAndEntities = ( eventData, tagConfig, eventProperties, userAttributes ) => { const inclusionRules = parseEntityRules(tagConfig); const exclusionRules = parseEntityExclusionRules(tagConfig); const excludedRefs = exclusionRules.map((r) => r.ref); const finalEntityRules = finalizeEntityRules(inclusionRules, excludedRefs); const finalEntityRefs = finalEntityRules.map((r) => r.ref); for (let prop in eventData) { if (eventData.hasOwnProperty(prop)) { const cleanPropName = cleanPropertyName(prop); if (isSpSelfDescProp(prop) && tagConfig.includeSelfDescribingEvent) { if (!isEmpty(eventData[prop])) { for (let key in eventData[prop]) { if (eventData[prop].hasOwnProperty(key)) { eventProperties[key] = eventData[prop][key]; } } } continue; } if (isSpContextsProp(prop)) { if (getReferenceIdx(prop, excludedRefs) >= 0) { continue; } const ctxVal = eventData[prop]; const refIdx = getReferenceIdx(prop, finalEntityRefs); if (refIdx >= 0) { const rule = finalEntityRules[refIdx]; const target = rule.target === 'event_object' ? eventProperties : userAttributes; target[rule.mappedKey] = ctxVal; } else { if (tagConfig.includeEntities === 'none') { continue; } // here includedEntities is 'all' and prop is not excluded eventProperties[cleanPropName] = ctxVal; } } if (isSpAtomicProp(prop) && tagConfig.includeAllAtomicProperties) { eventProperties[cleanPropName] = eventData[prop]; } } } }; // **************************** Snowplow Helpers End **************************** // **************************** Helpers Start **************************** /** * Assumes logType argument is string. * Determines if logging is enabled. * * @param {string} logType - The logType set ('no', 'debug', 'always') * @returns {boolean} Whether logging is enabled */ const determineIsLoggingEnabled = (logType) => { const containerVersion = getContainerVersion(); const isDebugMode = !!( containerVersion && (containerVersion.debugMode || containerVersion.previewMode) ); if (!logType) { return isDebugMode; } if (data.logType === 'no') { return false; } if (data.logType === 'debug') { return isDebugMode; } return data.logType === 'always'; }; /** * Creates the log message and logs it to console. * * @param {string} typeName - The type of log ('Message', 'Request', 'Response') * @param {Object} stdInfo - The standard info for all logs (Name, Type, TraceId, EventName) * @param {Object} logInfo - An object including information for the specific log type * @returns {undefined} */ const doLogging = (typeName, stdInfo, logInfo) => { const logMessage = { Name: stdInfo.tagName, Type: typeName, TraceId: stdInfo.traceId, EventName: stdInfo.eventName, }; switch (typeName) { case 'Message': logMessage.Message = logInfo.msg; break; case 'Request': logMessage.RequestMethod = logInfo.requestMethod; logMessage.RequestUrl = logInfo.requestUrl; logMessage.RequestHeaders = logInfo.requestHeaders; logMessage.RequestBody = logInfo.requestBody; break; case 'Response': logMessage.ResponseStatusCode = logInfo.responseStatusCode; logMessage.ResponseHeaders = logInfo.responseHeaders; logMessage.ResponseBody = logInfo.responseBody; break; default: // do nothing return; } logToConsole(JSON.stringify(logMessage)); }; /** * Left-pads an integer with zeros. Assumes a non-negative integer. * * @param {integer} x - The integer to pad * @param {integer} totalLen - The desired total length * @returns {string} The resulting string after padding with zeros */ const pad = (x, totalLen) => { const s = x.toString(); if (s.length < totalLen) { return pad('0' + s, totalLen); } return s; }; /** * Given the time components, returns the ISO-8601 time code. * * @param {integer} yr - Year * @param {integer} mon - Month * @param {integer} day - Day * @param {integer} hrs - Hours * @param {integer} mins - Minutes * @param {integer} secs - Seconds * @param {integer} msecs - Milliseconds * @returns {string} */ const formatISO = (yr, mon, day, hrs, mins, secs, msecs) => { const isoDate = [yr.toString(), pad(mon, 2), pad(day, 2)].join('-'); const isoTime = [pad(hrs, 2), pad(mins, 2), pad(secs, 2)].join(':'); return isoDate + 'T' + isoTime + '.' + pad(msecs, 3) + 'Z'; }; /** * Transforms a unix timestamp in milliseconds to ISO-8601 time code. * Ref: https://howardhinnant.github.io/date_algorithms.html#civil_from_days * * @param {integer} unixMillis - The unix time in millis * @returns {string} The ISO time code */ const unixMillisToISO = (unixMillis) => { const totalSecs = Math.floor(unixMillis / 1000); const z = Math.floor(totalSecs / 86400) + 719468; const era = Math.floor((z >= 0 ? z : z - 146096) / 146097); const doe = z - era * 146097; const yoe = Math.floor( (doe - (Math.floor(doe / 1460) + Math.floor(doe / 36524) - Math.floor(doe / 146096))) / 365 ); const y = yoe + era * 400; const doy = doe - (365 * yoe + Math.floor(yoe / 4) - Math.floor(yoe / 100)); const mp = Math.floor((5 * doy + 2) / 153); const prevDay = doy - Math.floor((153 * mp + 2) / 5); const day = prevDay + 1; const month = mp < 10 ? mp + 3 : mp - 9; const year = month <= 2 ? y + 1 : y; const daySecs = totalSecs % 86400; const dayMins = Math.floor(daySecs / 60); const hour = Math.floor(dayMins / 60); const minute = dayMins - hour * 60; const second = daySecs - dayMins * 60; const millis = unixMillis % 1000; return formatISO(year, month, day, hour, minute, second, millis); }; /** * Utility function to check if an object has own properties. * Assumes its input is an object in standard JavaScript. * * @param {Object} obj - The object to check * @returns {boolean} */ const isEmpty = (obj) => { for (let prop in obj) { if (obj.hasOwnProperty(prop)) { return false; } } return true; }; /** * Removes equal to null properties from given object. * * @param {Object} obj - The object to clean * @returns {Object} */ const cleanObject = (obj) => { let target = {}; for (let prop in obj) { if (obj.hasOwnProperty(prop) && obj[prop] != null) { target[prop] = obj[prop]; } } return target; }; /** * Given the tag configuration, * returns the value to be used as the time property of a MoEngage event. * * @param {Object} tagConfig - The tag configuration * @returns {*} */ const getEventTime = (tagConfig) => { if (tagConfig.eventTime) { return getEventData(tagConfig.eventTime); } return unixMillisToISO(getTimestampMillis()); }; /** * Utility function that creates an object according to Event Property Rules. * * @param {Object[]} configProps - The event property rules * @returns {Object} */ const getEventDataByKeys = (configProps) => { const props = {}; configProps.forEach((p) => { let eventProperty = getEventData(p.key); if (eventProperty !== undefined) { props[p.mappedKey || p.key] = eventProperty; } }); return props; }; const generateUserAttributesObject = (eventData, tagConfig) => { const attributes = { language: eventData.language, time_zone: eventData['x-sp-geo_timezone'], }; if (eventData['x-sp-geo_longitude'] && eventData['x-sp-geo_latitude']) { attributes.current_location = { longitude: eventData['x-sp-geo_longitude'], latitude: eventData['x-sp-geo_latitude'], }; } if (tagConfig.includeCommonUserProperties && eventData.user_data) { attributes.email = eventData.user_data.email_address; attributes.phone = eventData.user_data.phone_number; if (eventData.user_data.address) { const address = eventData.user_data.address.street + ' ' + eventData.user_data.address.city + ' ' + eventData.user_data.address.region; attributes.address = address; attributes.country = eventData.user_data.address.country; attributes.postal_code = eventData.user_data.address.postal_code; } } if (tagConfig.userMappingRules && tagConfig.userMappingRules.length > 0) { return mergeObjects([attributes, getEventDataByKeys(tagConfig.userMappingRules)]); } return attributes; }; const generateEventProperties = (eventData, tagConfig) => { const properties = {}; if (tagConfig.includeCommonEventProperties) { properties.page_location = eventData.page_location; properties.page_encoding = eventData.page_encoding; properties.page_referrer = eventData.page_referrer; properties.page_title = eventData.page_title; properties.screen_resolution = eventData.screen_resolution; properties.viewport_size = eventData.viewport_size; } if (tagConfig.eventMappingRules && tagConfig.eventMappingRules.length > 0) { return mergeObjects([properties, getEventDataByKeys(tagConfig.eventMappingRules)]); } return properties; }; const generateEventObject = (eventData, tagConfig) => { const properties = generateEventProperties(eventData, tagConfig); return { attributes: properties, }; }; const getCustomerID = (eventData, tagConfig) => { if (tagConfig.userIdentifierType === 'userIdentifierVariable') { return tagConfig.userID; } return getEventData(tagConfig.userID); }; const generateAdditionalKeyValue = (dataType, tagConfig) => { const dataToLoop = (dataType === 'event' ? tagConfig.additionalEventKeyValue : tagConfig.additionalUserKeyValue) || []; return makeTableMap(dataToLoop, 'key', 'value'); }; /** * Given the tag configuration and event data, * generates payload */ const generatePayload = (eventData, tagConfig) => { const userAttrs = generateUserAttributesObject(eventData, tagConfig); const eventObject = generateEventObject(eventData, tagConfig); parseCustomEventAndEntities( eventData, tagConfig, eventObject.attributes, userAttrs ); const additionalEventData = generateAdditionalKeyValue('event', tagConfig); const additionalUserData = generateAdditionalKeyValue('user', tagConfig); const requestPayload = { type: "transition", elements: [ { type: "event", customer_id: getCustomerID(eventData, tagConfig), actions: [ { action: tagConfig.eventNameOverride || eventData.event_name, current_time: getEventTime(tagConfig), attributes: flattenObject(mergeObjects([eventObject.attributes, additionalEventData])), } ] } ] }; if (!isEmpty(additionalUserData)) { requestPayload.elements.push( { type: "customer", customer_id: getCustomerID(eventData, tagConfig), attributes: flattenObject(mergeObjects([userAttrs, additionalUserData])) } ); } else if (!isEmpty(userAttrs)) { requestPayload.elements.push( { type: "customer", customer_id: getCustomerID(eventData, tagConfig), attributes: flattenObject(userAttrs) } ); } return cleanObject(requestPayload); }; // Generates options for api call const generateRequestOptions = (tagConfig, redact) => { const authToken = tagConfig.apiID + ':' + tagConfig.apiKey; const authTokenHashed = toBase64(authToken); return { method: 'POST', headers: { 'Content-Type': 'application/json', Authorization: redact ? 'redacted' : 'Basic ' + authTokenHashed, // Don't get confused here. Below is correct 'MOE-APPKEY': redact ? 'redacted' : tagConfig.apiID, }, }; }; // **************************** Helpers End **************************** // **************************** Main **************************** const url = 'https://api-0' + data.dataCenter + '.moengage.com/v1/transition/' + data.apiID; const eventData = getAllEventData(); const loggingEnabled = determineIsLoggingEnabled(data.logType); const traceIdHeader = loggingEnabled ? getRequestHeader('trace-id') : undefined; const stdLogInfo = { tagName: tagName, traceId: traceIdHeader, eventName: eventData.event_name, }; doLogging('Message', stdLogInfo, { msg: eventData }); const requestOptions = generateRequestOptions(data, false); const requestPayload = generatePayload(eventData, data); if (loggingEnabled) { const logRequestOptions = generateRequestOptions(data, true); doLogging('Request', stdLogInfo, { requestMethod: logRequestOptions.method, requestUrl: url, requestHeaders: logRequestOptions.headers, requestBody: requestPayload, }); } sendHttpRequest(url, requestOptions, JSON.stringify(requestPayload)) .then((response) => { if (loggingEnabled) { doLogging('Response', stdLogInfo, { responseStatusCode: response.statusCode, responseHeaders: response.headers, responseBody: response.body, }); } if (response.statusCode >= 200 && response.statusCode < 300) { data.gtmOnSuccess(); return; } data.gtmOnFailure(); }, (errorData) => { doLogging('Message', stdLogInfo, { msg: 'HTTP Request promise reject reason: ' + errorData.error }); }); ___SERVER_PERMISSIONS___ [ { "instance": { "key": { "publicId": "read_request", "versionId": "1" }, "param": [ { "key": "headerWhitelist", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "headerName" } ], "mapValue": [ { "type": 1, "string": "trace-id" } ] } ] } }, { "key": "headersAllowed", "value": { "type": 8, "boolean": true } }, { "key": "requestAccess", "value": { "type": 1, "string": "specific" } }, { "key": "headerAccess", "value": { "type": 1, "string": "specific" } }, { "key": "queryParameterAccess", "value": { "type": 1, "string": "any" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "logging", "versionId": "1" }, "param": [ { "key": "environments", "value": { "type": 1, "string": "all" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "read_container_data", "versionId": "1" }, "param": [] }, "isRequired": true }, { "instance": { "key": { "publicId": "read_event_data", "versionId": "1" }, "param": [ { "key": "eventDataAccess", "value": { "type": 1, "string": "any" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "send_http", "versionId": "1" }, "param": [ { "key": "allowedUrls", "value": { "type": 1, "string": "any" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true } ] ___TESTS___ scenarios: - name: Test defaults code: | const mockData = { dataCenter: 2, apiID: 'testApiID', apiKey: 'testApiKey', userID: 'testUserID', userIdentifierType: 'userIdentifierVariable', includeSelfDescribingEvent: true, extractFromArray: true, includeEntities: 'all', includeCommonEventProperties: true, includeCommonUserProperties: true, includeAllAtomicProperties: true, logType: 'debug', }; const testEvent = mockEventObjectSelfDesc; // to assert on let argUrl, argOptions, argBody; // Mocks const httpPromiseResolve = promise.create((resolve, reject) => { const resolvedValue = { statusCode: 200, headers: { foo: 'bar' }, body: 'ok', }; resolve(resolvedValue); }); mock('sendHttpRequest', function () { argUrl = arguments[0]; argOptions = arguments[1]; argBody = arguments[2]; return httpPromiseResolve; }); mock('getContainerVersion', function () { let containerVersion = { debugMode: true, previewMode: false, }; return containerVersion; }); mock('getAllEventData', function () { return testEvent; }); mock('getEventData', function (x) { return getFromPath(x, testEvent); }); // Expectations const expectedUrl = 'https://api-0' + mockData.dataCenter + '.moengage.com/v1/transition/' + mockData.apiID; const expectedOptions = { method: 'POST', headers: { 'Content-Type': 'application/json', Authorization: 'Basic ' + toB64(mockData.apiID + ':' + mockData.apiKey), 'MOE-APPKEY': mockData.apiID, }, }; const expectedBody = { "type": "transition", "elements": [ { "type": "event", "customer_id": "testUserID", "actions": [ { "action": "media_player_event", "current_time": "2022-09-05T18:39:56.056Z", "attributes": { "page_location": "http://localhost:8000/", "page_encoding": "windows-1252", "screen_resolution": "1920x1080", "viewport_size": "1044x975", "app_id": "media-test", "platform": "web", "dvce_created_tstamp": "1658567928426", "event_id": "c2084e30-5e4f-4d9c-86b2-e0bc3781509a", "name_tracker": "spTest", "v_tracker": "js-3.5.0", "domain_sessionid": "1ab28b79-bfdd-4855-9bf1-5199ce15beac", "domain_sessionidx": 1, "br_cookies": "1", "br_colordepth": "24", "br_viewwidth": 1044, "br_viewheight": 975, "dvce_screenwidth": 1920, "dvce_screenheight": 1080, "doc_charset": "windows-1252", "doc_width": 1044, "doc_height": 975, "dvce_sent_tstamp": "1658567928427", "type": "play", "osType": "myOsType", "osVersion": "myOsVersion", "deviceManufacturer": "myDevMan", "deviceModel": "myDevModel", "autoPlay": false, "avaliablePlaybackRates": [ 0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2 ], "buffering": false, "controls": true, "cued": false, "loaded": 3, "playbackQuality": "medium", "playerId": "youtube-song", "unstarted": false, "url": "https://www.youtube.com/watch?v=foobarbaz", "avaliableQualityLevels": [ "hd1080", "hd720", "large", "medium", "small", "tiny", "auto" ], "currentTime": 0.015303093460083008, "duration": 190.301, "ended": false, "loop": false, "muted": false, "paused": false, "playbackRate": 1, "volume": 100, "id": "68027aa2-34b1-4018-95e3-7176c62dbc84", "email_address": "foo@test.io", "userId": "fd0e5288-e89b-45df-aad5-6d0c6eda6198", "sessionId": "1ab28b79-bfdd-4855-9bf1-5199ce15beac", "eventIndex": 24, "sessionIndex": 1, "previousSessionId": null, "storageMechanism": "COOKIE_1", "firstEventId": "40fbdb30-1b99-42a3-99f7-850dacf5be43", "firstEventTimestamp": "2022-07-23T09:08:04.451Z" } } ] }, { "type": "customer", "customer_id": "testUserID", "attributes": { "language": "en-US", "email": "foo@test.io" } } ] }; // Call runCode to run the template's code runCode(mockData); // Assertions assertApi('sendHttpRequest').wasCalled(); assertThat(argUrl).isStrictlyEqualTo(expectedUrl); assertThat(argOptions).isEqualTo(expectedOptions); const body = jsonApi.parse(argBody); assertThat(body).isEqualTo(expectedBody); assertApi('logToConsole').wasCalled(); - name: Test no atomic properties code: | const mockData = { dataCenter: 2, apiID: 'testApiID', apiKey: 'testApiKey', userID: 'testUserID', userIdentifierType: 'userIdentifierVariable', includeSelfDescribingEvent: true, includeEntities: 'all', includeCommonEventProperties: true, includeCommonUserProperties: true, includeAllAtomicProperties: false, logType: 'debug', }; const testEvent = mockEventObjectSelfDesc; // to assert on let argUrl, argOptions, argBody; // Mocks const httpPromiseResolve = promise.create((resolve, reject) => { const resolvedValue = { statusCode: 200, headers: { foo: 'bar' }, body: 'ok', }; resolve(resolvedValue); }); mock('sendHttpRequest', function () { argUrl = arguments[0]; argOptions = arguments[1]; argBody = arguments[2]; return httpPromiseResolve; }); mock('getContainerVersion', function () { let containerVersion = { debugMode: true, previewMode: false, }; return containerVersion; }); mock('getAllEventData', function () { return testEvent; }); mock('getEventData', function (x) { return getFromPath(x, testEvent); }); // Expectations const expectedUrl = 'https://api-0' + mockData.dataCenter + '.moengage.com/v1/transition/' + mockData.apiID; const expectedOptions = { method: 'POST', headers: { 'Content-Type': 'application/json', Authorization: 'Basic ' + toB64(mockData.apiID + ':' + mockData.apiKey), 'MOE-APPKEY': mockData.apiID, }, }; const expectedBody = { type: 'transition', elements: [ { type: 'event', customer_id: 'testUserID', actions: [ { action: 'media_player_event', current_time: '2022-09-05T18:39:56.056Z', attributes: { page_location: 'http://localhost:8000/', page_encoding: 'windows-1252', screen_resolution: '1920x1080', viewport_size: '1044x975', type: 'play', osType: 'myOsType', osVersion: 'myOsVersion', deviceManufacturer: 'myDevMan', deviceModel: 'myDevModel', autoPlay: false, avaliablePlaybackRates: [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2], buffering: false, controls: true, cued: false, loaded: 3, playbackQuality: 'medium', playerId: 'youtube-song', unstarted: false, url: 'https://www.youtube.com/watch?v=foobarbaz', avaliableQualityLevels: [ 'hd1080', 'hd720', 'large', 'medium', 'small', 'tiny', 'auto', ], currentTime: 0.015303093460083008, duration: 190.301, ended: false, loop: false, muted: false, paused: false, playbackRate: 1, volume: 100, id: '68027aa2-34b1-4018-95e3-7176c62dbc84', email_address: 'foo@test.io', userId: 'fd0e5288-e89b-45df-aad5-6d0c6eda6198', sessionId: '1ab28b79-bfdd-4855-9bf1-5199ce15beac', eventIndex: 24, sessionIndex: 1, previousSessionId: null, storageMechanism: 'COOKIE_1', firstEventId: '40fbdb30-1b99-42a3-99f7-850dacf5be43', firstEventTimestamp: '2022-07-23T09:08:04.451Z', }, }, ], }, { type: 'customer', customer_id: 'testUserID', attributes: { language: 'en-US', email: 'foo@test.io', }, }, ], }; // Call runCode to run the template's code runCode(mockData); // Assertions assertApi('sendHttpRequest').wasCalled(); assertThat(argUrl).isStrictlyEqualTo(expectedUrl); assertThat(argOptions).isEqualTo(expectedOptions); const body = jsonApi.parse(argBody); assertThat(body).isEqualTo(expectedBody); assertApi('logToConsole').wasCalled(); - name: Test with promise rejection code: | const mockData = { dataCenter: 2, apiID: 'testApiID', apiKey: 'testApiKey', userID: 'testUserID', includeSelfDescribingEvent: true, includeEntities: 'all', includeCommonEventProperties: true, includeCommonUserProperties: true, includeAllAtomicProperties: true, logType: 'debug', }; const testEvent = mockEventObjectPageView; // to assert on let argUrl, argOptions, argBody; // Mocks const httpPromiseReject = promise.create((resolve, reject) => { const rejectReason = { reason: 'timed_out', }; reject(rejectReason); }); mock('sendHttpRequest', function () { argUrl = arguments[0]; argOptions = arguments[1]; argBody = arguments[2]; return httpPromiseReject; }); mock('getContainerVersion', function () { let containerVersion = { debugMode: true, previewMode: true, }; return containerVersion; }); mock('getAllEventData', function () { return testEvent; }); mock('getEventData', function (x) { return getFromPath(x, testEvent); }); // Call runCode to run the template's code runCode(mockData); // Assertions assertApi('sendHttpRequest').wasCalled(); // Expectations assertApi('logToConsole').wasCalled(); // assertApi('logToConsole').wasCalledWith(); // TODO assertApi('logToConsole').wasCalled(); // assertApi('logToConsole').wasCalledWith(); // TODO assertApi('logToConsole').wasCalled(); // assertApi('logToConsole').wasCalledWith(); // TODO - name: Test include none code: | const mockData = { dataCenter: 2, apiID: 'testApiID', apiKey: 'testApiKey', userID: 'testUserID', userIdentifierType: 'userIdentifierVariable', includeSelfDescribingEvent: true, includeEntities: 'none', includeCommonEventProperties: true, includeCommonUserProperties: true, includeAllAtomicProperties: false, logType: 'debug', }; const testEvent = mockEventObjectPageView; // to assert on let argUrl, argOptions, argBody; // Mocks const httpPromiseResolve = promise.create((resolve, reject) => { const resolvedValue = { statusCode: 200, headers: { foo: 'bar' }, body: 'ok', }; resolve(resolvedValue); }); mock('sendHttpRequest', function () { argUrl = arguments[0]; argOptions = arguments[1]; argBody = arguments[2]; return httpPromiseResolve; }); mock('getContainerVersion', function () { let containerVersion = { debugMode: true, previewMode: false, }; return containerVersion; }); mock('getAllEventData', function () { return testEvent; }); mock('getEventData', function (x) { return getFromPath(x, testEvent); }); // Expectations const expectedUrl = 'https://api-0' + mockData.dataCenter + '.moengage.com/v1/transition/' + mockData.apiID; const expectedOptions = { method: 'POST', headers: { 'Content-Type': 'application/json', Authorization: 'Basic ' + toB64(mockData.apiID + ':' + mockData.apiKey), 'MOE-APPKEY': mockData.apiID, }, }; const expectedBody = { type: 'transition', elements: [ { type: 'event', customer_id: mockData.userID, actions: [ { action: 'page_view', current_time: '2022-09-05T18:39:56.056Z', attributes: { page_location: testEvent.page_location, page_encoding: testEvent.page_encoding, page_referrer: testEvent.page_referrer, page_title: testEvent.page_title, screen_resolution: testEvent.screen_resolution, viewport_size: testEvent.viewport_size, // none of the Snowplow atomic props are included }, }, ], }, { type: 'customer', customer_id: mockData.userID, attributes: { language: testEvent.language, country: testEvent.user_data.address.country, email: testEvent.user_data.email_address, phone: testEvent.user_data.phone_number, postal_code: testEvent.user_data.address.postal_code, address: testEvent.user_data.address.street + ' ' + testEvent.user_data.address.city + ' ' + testEvent.user_data.address.region, }, }, ], }; // Call runCode to run the template's code runCode(mockData); // Assertions assertApi('sendHttpRequest').wasCalled(); assertThat(argUrl).isStrictlyEqualTo(expectedUrl); assertThat(argOptions).isEqualTo(expectedOptions); const body = jsonApi.parse(argBody); assertThat(body).isEqualTo(expectedBody); assertApi('logToConsole').wasCalled(); - name: Test rules and additional pairs code: | const mockData = { dataCenter: 3, apiID: 'testApiId', apiKey: 'testApiKey', userIdentifierType: 'userIdentifierEvent', userID: 'user_id', includeSelfDescribingEvent: true, includeAllAtomicProperties: true, includeEntities: 'all', entityMappingRules: [ { key: 'x-sp-contexts_com_snowplowanalytics_snowplow_mobile_context_1', mappedKey: 'mobile_ctx', // This does not take effect propertiesObjectToPopulate: 'event_object', version: 'control', }, { key: 'x-sp-contexts_com_google_tag-manager_server-side_user_data_1', mappedKey: 'udata_ctx', // This does not take effect propertiesObjectToPopulate: 'user_attributes_object', version: 'free', }, ], entityExclusionRules: [ { key: 'x-sp-contexts_com_snowplowanalytics_snowplow_web_page_1', version: 'control', }, ], includeCommonEventProperties: false, eventMappingRules: [ { key: 'x-sp-contexts_com_snowplowanalytics_snowplow_web_page_1.0.id', mappedKey: 'page_view_id', }, ], includeCommonUserProperties: true, userMappingRules: [ { key: 'x-sp-contexts_com_snowplowanalytics_snowplow_client_session_1.0.userId', mappedKey: 'client_session_id', }, ], additionalEventKeyValue: [ { key: 'additional_event_key', value: 'test_value' }, ], additionalUserKeyValue: [{ key: 'additional_user_key', value: 'test_value' }], logType: 'debug', }; const testEvent = mockEventObjectSelfDesc; // to assert on let argUrl, argOptions, argBody; // Mocks const httpPromiseResolve = promise.create((resolve, reject) => { const resolvedValue = { statusCode: 200, headers: { foo: 'bar' }, body: 'ok', }; resolve(resolvedValue); }); mock('sendHttpRequest', function () { argUrl = arguments[0]; argOptions = arguments[1]; argBody = arguments[2]; return httpPromiseResolve; }); mock('getContainerVersion', function () { let containerVersion = { debugMode: true, previewMode: false, }; return containerVersion; }); mock('getAllEventData', function () { return testEvent; }); mock('getEventData', function (x) { return getFromPath(x, testEvent); }); // Expectations const expectedUrl = 'https://api-0' + mockData.dataCenter + '.moengage.com/v1/transition/' + mockData.apiID; const expectedOptions = { method: 'POST', headers: { 'Content-Type': 'application/json', Authorization: 'Basic ' + toB64(mockData.apiID + ':' + mockData.apiKey), 'MOE-APPKEY': mockData.apiID, }, }; const expectedBody = { type: 'transition', elements: [ { type: 'event', customer_id: testEvent.user_id, actions: [ { action: testEvent.event_name, current_time: '2022-09-05T18:39:56.056Z', attributes: { app_id: testEvent['x-sp-app_id'], platform: testEvent['x-sp-platform'], dvce_created_tstamp: testEvent['x-sp-dvce_created_tstamp'], event_id: testEvent['x-sp-event_id'], name_tracker: testEvent['x-sp-name_tracker'], v_tracker: testEvent['x-sp-v_tracker'], domain_sessionid: testEvent['x-sp-domain_sessionid'], domain_sessionidx: testEvent['x-sp-domain_sessionidx'], br_cookies: testEvent['x-sp-br_cookies'], br_colordepth: testEvent['x-sp-br_colordepth'], br_viewwidth: testEvent['x-sp-br_viewwidth'], br_viewheight: testEvent['x-sp-br_viewheight'], dvce_screenwidth: testEvent['x-sp-dvce_screenwidth'], dvce_screenheight: testEvent['x-sp-dvce_screenheight'], doc_charset: testEvent['x-sp-doc_charset'], doc_width: testEvent['x-sp-doc_width'], doc_height: testEvent['x-sp-doc_height'], dvce_sent_tstamp: testEvent['x-sp-dvce_sent_tstamp'], type: 'play', osType: 'myOsType', osVersion: 'myOsVersion', deviceManufacturer: 'myDevMan', deviceModel: 'myDevModel', autoPlay: false, avaliablePlaybackRates: [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2], buffering: false, controls: true, cued: false, loaded: 3, playbackQuality: 'medium', playerId: 'youtube-song', unstarted: false, url: 'https://www.youtube.com/watch?v=foobarbaz', avaliableQualityLevels: [ 'hd1080', 'hd720', 'large', 'medium', 'small', 'tiny', 'auto', ], currentTime: 0.015303093460083008, duration: 190.301, ended: false, loop: false, muted: false, paused: false, playbackRate: 1, volume: 100, userId: 'fd0e5288-e89b-45df-aad5-6d0c6eda6198', sessionId: '1ab28b79-bfdd-4855-9bf1-5199ce15beac', eventIndex: 24, sessionIndex: 1, previousSessionId: null, storageMechanism: 'COOKIE_1', firstEventId: '40fbdb30-1b99-42a3-99f7-850dacf5be43', firstEventTimestamp: '2022-07-23T09:08:04.451Z', page_view_id: testEvent[ 'x-sp-contexts_com_snowplowanalytics_snowplow_web_page_1' ][0].id, additional_event_key: 'test_value', }, }, ], }, { type: 'customer', customer_id: testEvent.user_id, attributes: { language: 'en-US', email: 'foo@test.io', additional_user_key: 'test_value', client_session_id: testEvent[ 'x-sp-contexts_com_snowplowanalytics_snowplow_client_session_1' ][0].userId, email_address: 'foo@test.io', }, }, ], }; // Call runCode to run the template's code runCode(mockData); // Assertions assertApi('sendHttpRequest').wasCalled(); assertThat(argUrl).isStrictlyEqualTo(expectedUrl); assertThat(argOptions).isEqualTo(expectedOptions); const body = jsonApi.parse(argBody); assertThat(body).isEqualTo(expectedBody); assertApi('logToConsole').wasCalled(); setup: |- const jsonApi = require('JSON'); const getTypeOf = require('getType'); const logToConsole = require('logToConsole'); const promise = require('Promise'); const toB64 = require('toBase64'); const mockEventObjectPageView = { event_name: 'page_view', client_id: 'd54a1904-7798-401a-be0b-1a83bea73634', user_id: 'snow123', language: 'en-GB', page_encoding: 'UTF-8', page_hostname: 'snowplow.io', page_location: 'https://snowplow.io/', page_path: '/', page_referrer: 'referer', page_title: 'Collect, manage and operationalize behavioral data at scale | Snowplow', screen_resolution: '1920x1080', viewport_size: '745x1302', user_agent: 'user-agent', origin: 'origin', host: 'host', 'x-sp-geo_country': 'geo_country', // fake - added by hand to test 'x-sp-app_id': 'website', 'x-sp-platform': 'web', 'x-sp-dvce_created_tstamp': '1628586512246', 'x-sp-event_id': '8676de79-0eba-4435-ad95-8a41a8a0129c', 'x-sp-name_tracker': 'sp', 'x-sp-v_tracker': 'js-2.18.1', 'x-sp-domain_sessionid': 'e7580b71-227b-4868-9ea9-322a263ce885', 'x-sp-domain_sessionidx': 1, 'x-sp-br_cookies': '1', 'x-sp-br_colordepth': '24', 'x-sp-br_viewwidth': 745, 'x-sp-br_viewheight': 1302, 'x-sp-dvce_screenwidth': 1920, 'x-sp-dvce_screenheight': 1080, 'x-sp-doc_charset': 'UTF-8', 'x-sp-doc_width': 730, 'x-sp-doc_height': 12393, 'x-sp-dvce_sent_tstamp': '1628586512248', 'x-sp-contexts_com_snowplowanalytics_snowplow_web_page_1': [ { id: 'a86c42e5-b831-45c8-b706-e214c26b4b3d', }, ], 'x-sp-contexts_org_w3_performance_timing_1': [ { navigationStart: 1628586508610, unloadEventStart: 0, unloadEventEnd: 0, redirectStart: 0, redirectEnd: 0, fetchStart: 1628586508610, domainLookupStart: 1628586508637, domainLookupEnd: 1628586508691, connectStart: 1628586508691, connectEnd: 1628586508763, secureConnectionStart: 1628586508721, requestStart: 1628586508763, responseStart: 1628586508797, responseEnd: 1628586508821, domLoading: 1628586509076, domInteractive: 1628586509381, domContentLoadedEventStart: 1628586509408, domContentLoadedEventEnd: 1628586509417, domComplete: 1628586510332, loadEventStart: 1628586510332, loadEventEnd: 1628586510334, }, ], 'x-sp-contexts_com_google_tag-manager_server-side_user_data_1': [ { email_address: 'foo@example.com', phone_number: '+15551234567', address: { first_name: 'Jane', last_name: 'Doe', street: '123 Fake St', city: 'San Francisco', region: 'CA', postal_code: '94016', country: 'US', }, }, ], user_data: { email_address: 'foo@example.com', phone_number: '+15551234567', address: { first_name: 'Jane', last_name: 'Doe', street: '123 Fake St', city: 'San Francisco', region: 'CA', postal_code: '94016', country: 'US', }, }, ga_session_id: 'e7580b71-227b-4868-9ea9-322a263ce885', ga_session_number: '1', 'x-ga-mp2-seg': '1', 'x-ga-protocol_version': '2', 'x-ga-page_id': 'a86c42e5-b831-45c8-b706-e214c26b4b3d', ip_override: '1.2.3.4', }; const mockEventObjectSelfDesc = { event_name: 'media_player_event', client_id: 'fd0e5288-e89b-45df-aad5-6d0c6eda6198', language: 'en-US', page_encoding: 'windows-1252', page_hostname: 'localhost', page_location: 'http://localhost:8000/', page_path: '/', screen_resolution: '1920x1080', user_id: 'tester', viewport_size: '1044x975', user_agent: 'curl/7.81.0', host: 'host', 'x-sp-app_id': 'media-test', 'x-sp-platform': 'web', 'x-sp-dvce_created_tstamp': '1658567928426', 'x-sp-event_id': 'c2084e30-5e4f-4d9c-86b2-e0bc3781509a', 'x-sp-name_tracker': 'spTest', 'x-sp-v_tracker': 'js-3.5.0', 'x-sp-domain_sessionid': '1ab28b79-bfdd-4855-9bf1-5199ce15beac', 'x-sp-domain_sessionidx': 1, 'x-sp-br_cookies': '1', 'x-sp-br_colordepth': '24', 'x-sp-br_viewwidth': 1044, 'x-sp-br_viewheight': 975, 'x-sp-dvce_screenwidth': 1920, 'x-sp-dvce_screenheight': 1080, 'x-sp-doc_charset': 'windows-1252', 'x-sp-doc_width': 1044, 'x-sp-doc_height': 975, 'x-sp-dvce_sent_tstamp': '1658567928427', 'x-sp-self_describing_event_com_snowplowanalytics_snowplow_media_player_event_1': { type: 'play' }, 'x-sp-contexts_com_snowplowanalytics_snowplow_mobile_context_1': [ { osType: 'myOsType', osVersion: 'myOsVersion', deviceManufacturer: 'myDevMan', deviceModel: 'myDevModel', }, ], 'x-sp-contexts_com_youtube_youtube_1': [ { autoPlay: false, avaliablePlaybackRates: [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2], buffering: false, controls: true, cued: false, loaded: 3, playbackQuality: 'medium', playerId: 'youtube-song', unstarted: false, url: 'https://www.youtube.com/watch?v=foobarbaz', avaliableQualityLevels: [ 'hd1080', 'hd720', 'large', 'medium', 'small', 'tiny', 'auto', ], }, ], 'x-sp-contexts_com_snowplowanalytics_snowplow_media_player_1': [ { currentTime: 0.015303093460083008, duration: 190.301, ended: false, loop: false, muted: false, paused: false, playbackRate: 1, volume: 100, }, ], 'x-sp-contexts_com_snowplowanalytics_snowplow_web_page_1': [ { id: '68027aa2-34b1-4018-95e3-7176c62dbc84' }, ], 'x-sp-contexts_com_google_tag-manager_server-side_user_data_1': [ { email_address: 'foo@test.io' }, ], 'x-sp-contexts_com_snowplowanalytics_snowplow_client_session_1': [ { userId: 'fd0e5288-e89b-45df-aad5-6d0c6eda6198', sessionId: '1ab28b79-bfdd-4855-9bf1-5199ce15beac', eventIndex: 24, sessionIndex: 1, previousSessionId: null, storageMechanism: 'COOKIE_1', firstEventId: '40fbdb30-1b99-42a3-99f7-850dacf5be43', firstEventTimestamp: '2022-07-23T09:08:04.451Z', }, ], 'x-sp-contexts': [ { schema: 'iglu:com.youtube/youtube/jsonschema/1-0-0', data: { autoPlay: false, avaliablePlaybackRates: [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2], buffering: false, controls: true, cued: false, loaded: 3, playbackQuality: 'medium', playerId: 'youtube-song', unstarted: false, url: 'https://www.youtube.com/watch?v=foobarbaz', avaliableQualityLevels: [ 'hd1080', 'hd720', 'large', 'medium', 'small', 'tiny', 'auto', ], }, }, { schema: 'iglu:com.snowplowanalytics.snowplow/media_player/jsonschema/1-0-0', data: { currentTime: 0.015303093460083008, duration: 190.301, ended: false, loop: false, muted: false, paused: false, playbackRate: 1, volume: 100, }, }, { schema: 'iglu:com.snowplowanalytics.snowplow/web_page/jsonschema/1-0-0', data: { id: '68027aa2-34b1-4018-95e3-7176c62dbc84' }, }, { schema: 'iglu:com.google.tag-manager.server-side/user_data/jsonschema/1-0-0', data: { email_address: 'foo@test.io' }, }, { schema: 'iglu:com.snowplowanalytics.snowplow/client_session/jsonschema/1-0-2', data: { userId: 'fd0e5288-e89b-45df-aad5-6d0c6eda6198', sessionId: '1ab28b79-bfdd-4855-9bf1-5199ce15beac', eventIndex: 24, sessionIndex: 1, previousSessionId: null, storageMechanism: 'COOKIE_1', firstEventId: '40fbdb30-1b99-42a3-99f7-850dacf5be43', firstEventTimestamp: '2022-07-23T09:08:04.451Z', }, }, ], user_data: { email_address: 'foo@test.io' }, ga_session_id: '1ab28b79-bfdd-4855-9bf1-5199ce15beac', ga_session_number: '1', 'x-ga-mp2-seg': '1', 'x-ga-protocol_version': '2', 'x-ga-page_id': '68027aa2-34b1-4018-95e3-7176c62dbc84', }; // Helper for mocking const getFromPath = (path, obj) => { if (getTypeOf(path) === 'string' && getTypeOf(obj) === 'object') { const splitPath = path.split('.').filter((prop) => !!prop); return splitPath.reduce((acc, curr) => acc && acc[curr], obj); } return undefined; }; mock('getTimestampMillis', function () { return 1662403196056; // '2022-09-05T18:39:56.056Z' }); const testTime = '2022-09-05T18:39:56.056Z'; ___NOTES___ Created on 11/08/2023, 16:05:57