___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": "CLIENT", "id": "cvt_temp_public_id", "version": 1, "securityGroups": [], "displayName": "Data Client", "brand": { "id": "brand_dummy", "displayName": "Stape", "thumbnail": "" }, "description": "Use this Client as a mapper from Request Data to Event Data.", "containerContexts": [ "SERVER" ] } ___TEMPLATE_PARAMETERS___ [ { "type": "CHECKBOX", "name": "exposeFPIDCookie", "checkboxText": "Expose FPID Cookie", "simpleValueType": true, "help": "If enabled, the FPID cookie only accessible by the server and generated by the GA4 Client, will be duplicated to FPIDP cookie, which will be accessible from in the browser by JavaScript. \n\u003cbr/\u003e\nIt\u0027s highly recommend using this option only in case it is necessary." }, { "type": "CHECKBOX", "name": "httpOnlyCookie", "checkboxText": "Write the _dcid cookie as HttpOnly", "simpleValueType": true, "help": "If enabled, the _dcid cookie will be written with the HttpOnly flag, making it non-accessible in the browser by JavaScript.", "defaultValue": false }, { "type": "CHECKBOX", "name": "generateClientId", "checkboxText": "Always generate client_id parameter", "simpleValueType": true, "help": "If enabled, even if the `client_id` parameter is not determined from the request, it will still be generated. The `client_id` parameter is required by GA4 tags.", "defaultValue": true }, { "type": "CHECKBOX", "name": "prolongCookies", "checkboxText": "Automatically prolong Data Tag cookies", "simpleValueType": true, "help": "If enabled, cookies generated by Data Tag will be reset from the server with an expiration time of 2 years. Its\u0027 useful if you use Data Tag store functionality.", "defaultValue": true }, { "type": "CHECKBOX", "name": "acceptMultipleEvents", "checkboxText": "Accept Multiple Events", "simpleValueType": true, "help": "When the Accept Multiple Events is set to true, the Data Client will parse an array of objects, in the request body, as separate events.\n\u003cbr/\u003e\nExample:\n\u003cbr/\u003e\n[\n {\"event\":\"page_view\"},\n {\"event\":\"view_item\"}\n]", "defaultValue": false }, { "type": "GROUP", "name": "responseSettings", "displayName": "Response Settings", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "SELECT", "name": "responseStatusCode", "displayName": "Response Status Code", "selectItems": [ { "value": 200, "displayValue": "200" }, { "value": 201, "displayValue": "201" }, { "value": 301, "displayValue": "301" }, { "value": 302, "displayValue": "302" }, { "value": 403, "displayValue": "403" }, { "value": 404, "displayValue": "404" } ], "simpleValueType": true, "defaultValue": 200 }, { "type": "GROUP", "name": "regularResponseSettings", "groupStyle": "NO_ZIPPY", "subParams": [ { "type": "SELECT", "name": "responseBody", "displayName": "Response Body", "macrosInSelect": false, "selectItems": [ { "value": "timestamp", "displayValue": "JSON Object with timestamp (recommended)" }, { "value": "eventData", "displayValue": "JSON Object with Event Data" }, { "value": "empty", "displayValue": "Empty" } ], "simpleValueType": true, "defaultValue": "timestamp" }, { "type": "CHECKBOX", "name": "responseBodyGet", "checkboxText": "Send Response Body for GET request", "simpleValueType": true, "help": "By default, for the GET request type, the answer is image pixel. \u003ca target\u003d\"_blank\" href\u003d\"https://developers.google.com/tag-manager/serverside/api#setpixelresponse\"\u003eMore Info\u003c/a\u003e." } ], "enablingConditions": [ { "paramName": "responseStatusCode", "paramValue": 200, "type": "EQUALS" }, { "paramName": "responseStatusCode", "paramValue": 201, "type": "EQUALS" } ] }, { "type": "GROUP", "name": "redirectResponseSettings", "groupStyle": "NO_ZIPPY", "subParams": [ { "type": "TEXT", "name": "redirectTo", "displayName": "Redirect To", "simpleValueType": true, "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "REGEX", "args": [ "^https?:\\/\\/.*" ] } ] }, { "type": "CHECKBOX", "name": "lookupForRedirectToParam", "checkboxText": "Try to find redirect destination in query params", "simpleValueType": true, "help": "Override destination URL with query param from request url" }, { "type": "TEXT", "name": "redirectToQueryParamName", "displayName": "Query Param Name", "simpleValueType": true, "enablingConditions": [ { "paramName": "lookupForRedirectToParam", "paramValue": true, "type": "EQUALS" } ], "valueValidators": [ { "type": "NON_EMPTY" } ] } ], "enablingConditions": [ { "paramName": "responseStatusCode", "paramValue": 301, "type": "EQUALS" }, { "paramName": "responseStatusCode", "paramValue": 302, "type": "EQUALS" } ] }, { "type": "GROUP", "name": "clientErrorResponseSettings", "groupStyle": "NO_ZIPPY", "subParams": [ { "type": "TEXT", "name": "clientErrorResponseMessage", "displayName": "Client Error Response Message", "simpleValueType": true, "valueValidators": [ { "type": "NON_EMPTY" } ] } ], "enablingConditions": [ { "paramName": "responseStatusCode", "paramValue": 403, "type": "EQUALS" }, { "paramName": "responseStatusCode", "paramValue": 404, "type": "EQUALS" } ] } ] }, { "type": "GROUP", "name": "pathSettings", "displayName": "Accepted Path Settings", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "SIMPLE_TABLE", "name": "path", "displayName": "Type additional paths that will be claimed by this client", "simpleTableColumns": [ { "defaultValue": "", "displayName": "For example: /callback", "name": "path", "type": "TEXT", "isUnique": true, "valueValidators": [ { "type": "NON_EMPTY" } ] } ], "newRowButtonText": "Add path", "help": "By default path \u003cb\u003e/data\u003c/b\u003e will be claimed. But you can add more paths that will be claimed by this client." } ] } ] ___SANDBOXED_JS_FOR_SERVER___ const returnResponse = require('returnResponse'); const runContainer = require('runContainer'); const setResponseHeader = require('setResponseHeader'); const setResponseStatus = require('setResponseStatus'); const setResponseBody = require('setResponseBody'); const JSON = require('JSON'); const fromBase64 = require('fromBase64'); const getTimestampMillis = require('getTimestampMillis'); const getCookieValues = require('getCookieValues'); const getRequestBody = require('getRequestBody'); const getRequestMethod = require('getRequestMethod'); const getRequestHeader = require('getRequestHeader'); const getRequestPath = require('getRequestPath'); const getRequestQueryParameters = require('getRequestQueryParameters'); const makeInteger = require('makeInteger'); const getRemoteAddress = require('getRemoteAddress'); const setCookie = require('setCookie'); const setPixelResponse = require('setPixelResponse'); const generateRandom = require('generateRandom'); const computeEffectiveTldPlusOne = require('computeEffectiveTldPlusOne'); const getRequestQueryParameter = require('getRequestQueryParameter'); const getType = require('getType'); const decodeUriComponent = require('decodeUriComponent'); const createRegex = require('createRegex'); const makeString = require('makeString'); const Object = require('Object'); /*============================================================================== ==============================================================================*/ const requestMethod = getRequestMethod(); const path = getRequestPath(); let isClientUsed = false; let isEventModelsWrappedInArray = false; if (path === '/data') { runClient(); } if (data.path && !isClientUsed) { for (let key in data.path) { if (!isClientUsed && data.path[key].path === path) { runClient(); } } } /*============================================================================== Vendor related functions ==============================================================================*/ function runClient() { isClientUsed = true; require('claimRequest')(); if (requestMethod === 'OPTIONS') { setCommonResponseHeaders(200); returnResponse(); return; } const baseEventModel = getBaseEventModelWithQueryParameters(); let eventModels = getEventModels(baseEventModel); const clientId = getClientId(eventModels); eventModels = eventModels.map((eventModel) => { eventModel = addRequiredParametersToEventModel(eventModel); eventModel = addCommonParametersToEventModel(eventModel); eventModel = addClientIdToEventModel(eventModel, clientId); if (eventModel._dcid_temp) Object.delete(eventModel, '_dcid_temp'); return eventModel; }); storeClientId(eventModels[0]); exposeFPIDCookie(eventModels[0]); prolongDataTagCookies(eventModels[0]); const responseStatusCode = makeInteger(data.responseStatusCode || 200); setCommonResponseHeaders(responseStatusCode); let counter = 0; eventModels.forEach((event) => { runContainer(event, () => { if (++counter === eventModels.length) { switch (responseStatusCode) { case 200: case 201: if (requestMethod === 'POST' || data.responseBodyGet) { prepareResponseBody(eventModels); } else { setPixelResponse(); } break; case 301: case 302: setRedirectLocation(); break; case 403: case 404: setClientErrorResponseMessage(); break; } returnResponse(); } }); }); } function addCommonParametersToEventModel(eventModel) { if (!eventModel.ip_override) { if (eventModel.ip) eventModel.ip_override = eventModel.ip; else if (eventModel.ipOverride) eventModel.ip_override = eventModel.ipOverride; else eventModel.ip_override = getRemoteAddress(); } if (!eventModel.user_agent) { if (eventModel.userAgent) eventModel.user_agent = eventModel.userAgent; else if (getRequestHeader('User-Agent')) eventModel.user_agent = getRequestHeader('User-Agent'); } if (!eventModel.language) { const acceptLanguageHeader = getRequestHeader('Accept-Language'); if (acceptLanguageHeader) { eventModel.language = acceptLanguageHeader .split(';')[0] .substring(0, 2) .toLowerCase(); } } if (!eventModel.page_hostname) { if (eventModel.pageHostname) eventModel.page_hostname = eventModel.pageHostname; else if (eventModel.hostname) eventModel.page_hostname = eventModel.hostname; } if (!eventModel.page_location) { if (eventModel.pageLocation) eventModel.page_location = eventModel.pageLocation; else if (eventModel.url) eventModel.page_location = eventModel.url; else if (eventModel.href) eventModel.page_location = eventModel.href; } if (!eventModel.page_referrer) { if (eventModel.pageReferrer) eventModel.page_referrer = eventModel.pageReferrer; else if (eventModel.referrer) eventModel.page_referrer = eventModel.referrer; else if (eventModel.urlref) eventModel.page_referrer = eventModel.urlref; } if (!eventModel.value && eventModel.e_v) eventModel.value = eventModel.e_v; if (getType(eventModel.items) === 'array' && eventModel.items.length) { const firstItem = eventModel.items[0]; if (!eventModel.currency && firstItem.currency) eventModel.currency = firstItem.currency; if (eventModel.items.length === 1) { if (!eventModel.item_id && firstItem.item_id) eventModel.item_id = firstItem.item_id; if (!eventModel.item_name && firstItem.item_name) eventModel.item_name = firstItem.item_name; if (!eventModel.item_brand && firstItem.item_brand) eventModel.item_brand = firstItem.item_brand; if (!eventModel.item_quantity && firstItem.quantity) eventModel.item_quantity = firstItem.quantity; if (!eventModel.item_category && firstItem.item_category) eventModel.item_category = firstItem.item_category; if (!eventModel.item_price && firstItem.price) eventModel.item_price = firstItem.price; } if (!eventModel.value) { const valueFromItems = eventModel.items.reduce((acc, item) => { if (!item.price) return acc; const quantity = item.quantity ? item.quantity : 1; return acc + quantity * item.price; }, 0); if (valueFromItems) eventModel.value = valueFromItems; } } const ecommerceAction = getEcommerceAction(eventModel); if (ecommerceAction) { if (!eventModel['x-ga-mp1-pa']) eventModel['x-ga-mp1-pa'] = ecommerceAction; if ( ecommerceAction === 'purchase' && eventModel.ecommerce.purchase.actionField ) { if (!eventModel['x-ga-mp1-tr']) eventModel['x-ga-mp1-tr'] = eventModel.ecommerce.purchase.actionField.revenue; if (!eventModel.revenue) eventModel.revenue = eventModel.ecommerce.purchase.actionField.revenue; if (!eventModel.affiliation) eventModel.affiliation = eventModel.ecommerce.purchase.actionField.affiliation; if (!eventModel.tax) eventModel.tax = eventModel.ecommerce.purchase.actionField.tax; if (!eventModel.shipping) eventModel.shipping = eventModel.ecommerce.purchase.actionField.shipping; if (!eventModel.coupon) eventModel.coupon = eventModel.ecommerce.purchase.actionField.coupon; if (!eventModel.transaction_id) eventModel.transaction_id = eventModel.ecommerce.purchase.actionField.id; } } if (!eventModel.page_encoding && eventModel.pageEncoding) eventModel.page_encoding = eventModel.pageEncoding; if (!eventModel.page_path && eventModel.pagePath) eventModel.page_path = eventModel.pagePath; if (!eventModel.page_title && eventModel.pageTitle) eventModel.page_title = eventModel.pageTitle; if (!eventModel.screen_resolution && eventModel.screenResolution) eventModel.screen_resolution = eventModel.screenResolution; if (!eventModel.viewport_size && eventModel.viewportSize) eventModel.viewport_size = eventModel.viewportSize; if (!eventModel.user_id && eventModel.userId) eventModel.user_id = eventModel.userId; if (!eventModel.user_data) { let userData = {}; let userAddressData = {}; if (!userData.email_address) { if (eventModel.userEmail) userData.email_address = eventModel.userEmail; else if (eventModel.email_address) userData.email_address = eventModel.email_address; else if (eventModel.email) userData.email_address = eventModel.email; else if (eventModel.mail) userData.email_address = eventModel.mail; } if (!userData.phone_number) { if (eventModel.userPhoneNumber) userData.phone_number = eventModel.userPhoneNumber; else if (eventModel.phone_number) userData.phone_number = eventModel.phone_number; else if (eventModel.phoneNumber) userData.phone_number = eventModel.phoneNumber; else if (eventModel.phone) userData.phone_number = eventModel.phone; } if (!userAddressData.street && eventModel.street) userAddressData.street = eventModel.street; if (!userAddressData.city && eventModel.city) userAddressData.city = eventModel.city; if (!userAddressData.region && eventModel.region) userAddressData.region = eventModel.region; if (!userAddressData.country && eventModel.country) userAddressData.country = eventModel.country; if (!userAddressData.first_name) { if (eventModel.userFirstName) userAddressData.first_name = eventModel.userFirstName; else if (eventModel.first_name) userAddressData.first_name = eventModel.first_name; else if (eventModel.firstName) userAddressData.first_name = eventModel.firstName; else if (eventModel.name) userAddressData.first_name = eventModel.name; } if (!userAddressData.last_name) { if (eventModel.userLastName) userAddressData.last_name = eventModel.userLastName; else if (eventModel.last_name) userAddressData.last_name = eventModel.last_name; else if (eventModel.lastName) userAddressData.last_name = eventModel.lastName; else if (eventModel.surname) userAddressData.last_name = eventModel.surname; else if (eventModel.family_name) userAddressData.last_name = eventModel.family_name; else if (eventModel.familyName) userAddressData.last_name = eventModel.familyName; } if (!userAddressData.region) { if (eventModel.region) userAddressData.region = eventModel.region; else if (eventModel.state) userAddressData.region = eventModel.state; } if (!userAddressData.postal_code) { if (eventModel.postal_code) userAddressData.postal_code = eventModel.postal_code; else if (eventModel.postalCode) userAddressData.postal_code = eventModel.postalCode; else if (eventModel.zip) userAddressData.postal_code = eventModel.zip; } if (getObjectLength(userAddressData) !== 0) { userData.address = userAddressData; } if (!eventModel.user_data && getObjectLength(userData) !== 0) { eventModel.user_data = userData; } } return eventModel; } function getBaseEventModelWithQueryParameters() { const requestQueryParameters = getRequestQueryParameters(); const eventModel = {}; if (requestQueryParameters) { for (let queryParameterKey in requestQueryParameters) { if ( (queryParameterKey === 'dtcd' || queryParameterKey === 'dtdc') && requestMethod === 'GET' ) { let dt = queryParameterKey === 'dtcd' ? JSON.parse(requestQueryParameters[queryParameterKey]) : JSON.parse(fromBase64(requestQueryParameters[queryParameterKey])); for (let dtKey in dt) { eventModel[dtKey] = dt[dtKey]; } } else { eventModel[queryParameterKey] = requestQueryParameters[queryParameterKey]; } } } return eventModel; } function addClientIdToEventModel(eventModel, clientId) { eventModel.client_id = clientId; return eventModel; } function prolongDataTagCookies(eventModel) { if (data.prolongCookies) { let stapeData = getCookieValues('stape'); if (stapeData.length) { setCookie('stape', stapeData[0], { domain: 'auto', path: '/', samesite: getCookieType(eventModel), secure: true, 'max-age': 63072000, // 2 years httpOnly: false }); } } } function addRequiredParametersToEventModel(eventModel) { if (!eventModel.event_name) { let eventName = 'Data'; if (eventModel.eventName) eventName = eventModel.eventName; else if (eventModel.event) eventName = eventModel.event; else if (eventModel.e_n) eventName = eventModel.e_n; eventModel.event_name = eventName; } return eventModel; } function exposeFPIDCookie(eventModel) { if (data.exposeFPIDCookie) { let fpid = getCookieValues('FPID'); if (fpid.length) { setCookie('FPIDP', fpid[0], { domain: 'auto', path: '/', samesite: getCookieType(eventModel), secure: true, 'max-age': 63072000, // 2 years httpOnly: false }); } } } function storeClientId(eventModel) { if (data.generateClientId) { setCookie('_dcid', eventModel.client_id, { domain: 'auto', path: '/', samesite: getCookieType(eventModel), secure: true, 'max-age': 63072000, // 2 years httpOnly: data.httpOnlyCookie || false }); } } function setCommonResponseHeaders(statusCode) { setResponseHeader('Access-Control-Max-Age', '600'); setResponseHeader('Access-Control-Allow-Origin', getRequestHeader('origin')); setResponseHeader( 'Access-Control-Allow-Methods', 'GET,POST,PUT,DELETE,OPTIONS' ); setResponseHeader( 'Access-Control-Allow-Headers', 'content-type,set-cookie,x-robots-tag,x-gtm-server-preview,x-stape-preview,x-stape-app-version' ); setResponseHeader('Access-Control-Allow-Credentials', 'true'); setResponseStatus(statusCode); } function getCookieType(eventModel) { if (!eventModel.page_location) { return 'Lax'; } const host = getRequestHeader('host'); const effectiveTldPlusOne = computeEffectiveTldPlusOne( eventModel.page_location ); if (!host || !effectiveTldPlusOne) { return 'Lax'; } if (host && host.indexOf(effectiveTldPlusOne) !== -1) { return 'Lax'; } return 'None'; } function prepareResponseBody(eventModels) { if (data.responseBody === 'empty') { return; } const responseModel = isEventModelsWrappedInArray ? eventModels[0] : eventModels; setResponseHeader('Content-Type', 'application/json'); if (data.responseBody === 'eventData') { setResponseBody(JSON.stringify(responseModel)); return; } if (isEventModelsWrappedInArray) { setResponseBody( JSON.stringify({ timestamp: responseModel.timestamp, unique_event_id: responseModel.unique_event_id }) ); return; } setResponseBody( JSON.stringify( eventModels.map((eventModel) => { return { timestamp: eventModel.timestamp, unique_event_id: eventModel.unique_event_id }; }) ) ); } function getEcommerceAction(eventModel) { if (eventModel.ecommerce) { const actions = [ 'detail', 'click', 'add', 'remove', 'checkout', 'checkout_option', 'purchase', 'refund' ]; for (let index = 0; index < actions.length; ++index) { const action = actions[index]; if (eventModel.ecommerce[action]) { return action; } } } return null; } function setRedirectLocation() { let location = data.redirectTo; if (data.lookupForRedirectToParam && data.redirectToQueryParamName) { const param = getRequestQueryParameter(data.redirectToQueryParamName); if (param && param.startsWith('http')) { location = param; } } setResponseHeader('location', location); } function setClientErrorResponseMessage() { if (data.clientErrorResponseMessage) { setResponseBody(data.clientErrorResponseMessage); } } function getEventModels(baseEventModel) { const body = getRequestBody(); if (body) { const contentType = getRequestHeader('content-type'); const isFormUrlEncoded = !!contentType && contentType.indexOf('application/x-www-form-urlencoded') !== -1; let bodyJson = isFormUrlEncoded ? parseUrlEncoded(body) : JSON.parse(body); if (bodyJson) { const bodyType = getType(bodyJson); const shouldUseOriginalBody = data.acceptMultipleEvents && bodyType === 'array'; if (!shouldUseOriginalBody) { bodyJson = [bodyJson]; isEventModelsWrappedInArray = true; } return bodyJson.map((bodyItem) => { const eventModel = assign( { timestamp: makeInteger(getTimestampMillis() / 1000), unique_event_id: getTimestampMillis() + '_' + generateRandom(100000000, 999999999) }, baseEventModel ); for (let bodyItemKey in bodyItem) { eventModel[bodyItemKey] = bodyItem[bodyItemKey]; } return eventModel; }); } } return [ assign( { timestamp: makeInteger(getTimestampMillis() / 1000), unique_event_id: getTimestampMillis() + '_' + generateRandom(100000000, 999999999) }, baseEventModel ) ]; } function getClientId(eventModels) { for (let i = 0; i < eventModels.length; i++) { const eventModel = eventModels[i]; const clientId = eventModel.client_id || eventModel.data_client_id || eventModel._dcid; if (clientId) return clientId; } const dcid = getCookieValues('_dcid'); if (dcid && dcid[0]) return dcid[0]; if (data.generateClientId) { for (let i = 0; i < eventModels.length; i++) { const eventModel = eventModels[i]; const tempClientId = eventModel._dcid_temp; if (tempClientId) return tempClientId; } return ( 'dcid.1.' + getTimestampMillis() + '.' + generateRandom(100000000, 999999999) ); } return ''; } /*============================================================================== Helpers ==============================================================================*/ function getObjectLength(object) { let length = 0; for (let key in object) { if (object.hasOwnProperty(key)) { ++length; } } return length; } function assign() { const target = arguments[0]; for (let i = 1; i < arguments.length; i++) { for (let key in arguments[i]) { target[key] = arguments[i][key]; } } return target; } function parseUrlEncoded(data) { const pairs = data.split('&'); const parsedData = {}; const regex = createRegex('\\+', 'g'); for (const pair of pairs) { const pairValue = pair.split('='); const key = pairValue[0]; const value = pairValue[1]; const keys = key .split('.') .map((k) => decodeUriComponent(k.replace(regex, ' '))); let currentObject = parsedData; for (let i = 0; i < keys.length - 1; i++) { const currentKey = keys[i]; if (!currentObject[currentKey]) { const nextKey = keys[i + 1]; const nextKeyIsNumber = makeString(makeInteger(nextKey)) === nextKey; currentObject[currentKey] = nextKeyIsNumber ? [] : {}; } currentObject = currentObject[currentKey]; } const lastKey = keys[keys.length - 1]; const decodedValue = decodeUriComponent(value.replace(regex, ' ')); const parsedValue = JSON.parse(decodedValue) || decodedValue; if (getType(currentObject) === 'array') { currentObject.push(parsedValue); } else { currentObject[lastKey] = parsedValue; } } return parsedData; } ___SERVER_PERMISSIONS___ [ { "instance": { "key": { "publicId": "return_response", "versionId": "1" }, "param": [] }, "isRequired": true }, { "instance": { "key": { "publicId": "access_response", "versionId": "1" }, "param": [ { "key": "writeResponseAccess", "value": { "type": 1, "string": "any" } }, { "key": "writeHeaderAccess", "value": { "type": 1, "string": "specific" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "run_container", "versionId": "1" }, "param": [] }, "isRequired": true }, { "instance": { "key": { "publicId": "get_cookies", "versionId": "1" }, "param": [ { "key": "cookieAccess", "value": { "type": 1, "string": "specific" } }, { "key": "cookieNames", "value": { "type": 2, "listItem": [ { "type": 1, "string": "stape" }, { "type": 1, "string": "_dcid" }, { "type": 1, "string": "FPIDP" }, { "type": 1, "string": "FPID" } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "read_request", "versionId": "1" }, "param": [ { "key": "requestAccess", "value": { "type": 1, "string": "any" } }, { "key": "headerAccess", "value": { "type": 1, "string": "any" } }, { "key": "queryParameterAccess", "value": { "type": 1, "string": "any" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "set_cookies", "versionId": "1" }, "param": [ { "key": "allowedCookies", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "name" }, { "type": 1, "string": "domain" }, { "type": 1, "string": "path" }, { "type": 1, "string": "secure" }, { "type": 1, "string": "session" } ], "mapValue": [ { "type": 1, "string": "stape" }, { "type": 1, "string": "*" }, { "type": 1, "string": "*" }, { "type": 1, "string": "any" }, { "type": 1, "string": "any" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "name" }, { "type": 1, "string": "domain" }, { "type": 1, "string": "path" }, { "type": 1, "string": "secure" }, { "type": 1, "string": "session" } ], "mapValue": [ { "type": 1, "string": "_dcid" }, { "type": 1, "string": "*" }, { "type": 1, "string": "*" }, { "type": 1, "string": "any" }, { "type": 1, "string": "any" } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "name" }, { "type": 1, "string": "domain" }, { "type": 1, "string": "path" }, { "type": 1, "string": "secure" }, { "type": 1, "string": "session" } ], "mapValue": [ { "type": 1, "string": "FPIDP" }, { "type": 1, "string": "*" }, { "type": 1, "string": "*" }, { "type": 1, "string": "any" }, { "type": 1, "string": "any" } ] } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true } ] ___TESTS___ scenarios: - name: runContainer is called succesfully for a single event and response is sent back once code: "mock('getRequestPath', '/data');\nmock('requestMethod', 'POST');\nmock('getRequestBody',\ \ '{\"event\":\"page_view\",\"client_id\":\"client_id\"}');\n\n/* \n For some\ \ reason when we use \"assertApi('claimRequest').wasCalled()\" AND we run all\ \ the tests,\n it produces the following error \"Tried to claim a request after\ \ a Client had returned. Calling claimRequest from a callback is not supported.\"\ \n If we run only this single test, the error does not occur.\n A workaround\ \ is to mock 'claimRequest' API and make a dummy assertion in the mocked function.\ \ This way we make sure it's been called.\n*/\nlet claimRequestWasCalled;\nmock('claimRequest',\ \ () => {\n claimRequestWasCalled = true;\n});\n\nlet runContainerExecutions\ \ = 0;\nconst runContainerExpectedExecutions = 1;\nmock('runContainer', (eventData,\ \ onCompleteCallback, onStartCallback) => {\n assertThat(eventData).isObject();\n\ \ runContainerExecutions++;\n assertThat(onCompleteCallback).isFunction();\n\ \ onCompleteCallback();\n});\n\nmock('returnResponse', () => {\n if (runContainerExecutions\ \ !== runContainerExpectedExecutions) {\n fail('returnResponse should be called\ \ by the last runContainer onComplete callback execution only.');\n return;\n\ \ }\n assertThat(true).isTrue(); // Dummy to cover the case when runContainerExecutions\ \ === runContainerExpectedExecutions.\n});\n\nrunCode(mockData);\n\ncallLater(()\ \ => {\n assertThat(claimRequestWasCalled).isTrue();\n assertApi('setResponseStatus').wasCalledWith(200);\n\ \ assertThat(runContainerExecutions).isEqualTo(runContainerExpectedExecutions);\n\ });" - name: runContainer is called succesfully for multiple events and response is sent back once code: "mockData.acceptMultipleEvents = true;\n\nmock('getRequestPath', '/data');\n\ mock('requestMethod', 'POST');\nmock('getRequestBody', '[{\"event\":\"page_view\"\ ,\"client_id\":\"client_id\"},{\"event\":\"view_item\",\"client_id\":\"client_id\"\ }]');\n\n/* \n For some reason when we use \"assertApi('claimRequest').wasCalled()\"\ \ AND we run all the tests,\n it produces the following error \"Tried to claim\ \ a request after a Client had returned. Calling claimRequest from a callback\ \ is not supported.\"\n If we run only this single test, the error does not occur.\n\ \ A workaround is to mock 'claimRequest' API and make a dummy assertion in the\ \ mocked function. This way we make sure it's been called.\n*/\nlet claimRequestWasCalled;\n\ mock('claimRequest', () => {\n claimRequestWasCalled = true;\n});\n\nlet runContainerExecutions\ \ = 0;\nconst runContainerExpectedExecutions = 2;\nmock('runContainer', (eventData,\ \ onCompleteCallback, onStartCallback) => {\n assertThat(eventData).isObject();\n\ \ runContainerExecutions++;\n assertThat(onCompleteCallback).isFunction();\n\ \ onCompleteCallback();\n});\n\nmock('returnResponse', () => {\n if (runContainerExecutions\ \ !== runContainerExpectedExecutions) {\n fail('returnResponse should be called\ \ by the last runContainer onComplete callback execution only.');\n return;\n\ \ }\n assertThat(true).isTrue(); // Dummy to cover the case when runContainerExecutions\ \ === runContainerExpectedExecutions.\n});\n\nrunCode(mockData);\n\ncallLater(()\ \ => {\n assertThat(claimRequestWasCalled).isTrue();\n assertApi('setResponseStatus').wasCalledWith(200);\n\ \ assertThat(runContainerExecutions).isEqualTo(runContainerExpectedExecutions);\n\ });" - name: Should fallback to 200 if Response Status Code is absent from the data object code: "mockData.acceptMultipleEvents = true;\n\nObject.delete(mockData, 'responseStatusCode');\n\ \nmock('getRequestPath', '/data');\nmock('requestMethod', 'POST');\nmock('getRequestBody',\ \ '[{\"event\":\"page_view\",\"client_id\":\"client_id\"},{\"event\":\"view_item\"\ ,\"client_id\":\"client_id\"}]');\n\n/* \n For some reason when we use \"assertApi('claimRequest').wasCalled()\"\ \ AND we run all the tests,\n it produces the following error \"Tried to claim\ \ a request after a Client had returned. Calling claimRequest from a callback\ \ is not supported.\"\n If we run only this single test, the error does not occur.\n\ \ A workaround is to mock 'claimRequest' API and make a dummy assertion in the\ \ mocked function. This way we make sure it's been called.\n*/\nlet claimRequestWasCalled;\n\ mock('claimRequest', () => {\n claimRequestWasCalled = true;\n});\n\nmock('setResponseStatus',\ \ (responseStatus) => {\n assertThat(responseStatus).isEqualTo(200);\n});\n\n\ /*\n For some reason when we mock 'claimRequest', the following error is shown\ \ when the test is run: 'Request must be claimed before calling runContainer.'\n\ \ A workaround is to mock 'runContainer' and simply call its callbacks.\n*/\n\ mock('runContainer', (eventData, onCompleteCallback, onStartCallback) => {\n \ \ onCompleteCallback();\n});\n\nrunCode(mockData);\n\ncallLater(() => {\n assertThat(claimRequestWasCalled).isTrue();\n\ \ assertApi('returnResponse').wasCalled();\n});" - name: Client ID retrieval and generation (from _dcid cookie, from Event Data pre-defined keys and from Temporary Client ID) code: "const originalMockData = mockData;\n\nmock('getRequestPath', '/data');\n\ mock('requestMethod', 'POST');\n\n[\n {\n description: 'Should use Client\ \ ID from Event Model if one is present',\n mockData: {\n generateClientId:\ \ true,\n acceptMultipleEvents: true\n },\n mock: () => {\n mock('getRequestBody',\ \ '[{\"event\":\"page_view\",\"client_id\":\"client_id\",\"_dcid_temp\":\"_dcid_temp\"\ },{\"event\":\"view_item\",\"_dcid\":\"_dcid\",\"_dcid_temp\":\"_dcid_temp\"}]');\n\ \ mock('getCookieValues', []);\n },\n assert: (eventData) => {\n \ \ assertThat(eventData.client_id).isEqualTo('client_id'); // The first found\ \ Client ID is used for all events in the payload.\n }\n },\n {\n description:\ \ 'Should use Client ID from _dcid cookie if present and no Client ID is present\ \ in Event Model',\n mockData: {\n generateClientId: true,\n acceptMultipleEvents:\ \ true\n }, \n mock: () => {\n mock('getRequestBody', '[{\"event\"\ :\"page_view\",\"_dcid_temp\":\"_dcid_temp\"},{\"event\":\"view_item\",\"_dcid_temp\"\ :\"_dcid_temp\"}]');\n mock('getCookieValues', ['_dcid_from_cookie']);\n\ \ },\n assert: (eventData) => {\n assertThat(eventData.client_id).isEqualTo('_dcid_from_cookie');\n\ \ }\n },\n { \n description: 'Should use Client ID from temporary Client\ \ ID if option is enabled, and it is present in Event Model and, _dcid cookie\ \ or Client ID in Event Data do not exist',\n mockData: {\n generateClientId:\ \ true,\n acceptMultipleEvents: true\n },\n mock: () => {\n mock('getRequestBody',\ \ '[{\"event\":\"page_view\",\"_dcid_temp\":\"_dcid_temp\"},{\"event\":\"view_item\"\ ,\"_dcid_temp\":\"_dcid_temp\"}]');\n mock('getCookieValues', []);\n },\n\ \ assert: (eventData) => {\n assertThat(eventData.client_id).isEqualTo('_dcid_temp');\n\ \ }\n },\n { \n description: 'Should generate a random Client ID if option\ \ is enabled, and _dcid cookie or Client ID in Event Data or temporary Client\ \ ID do not exist',\n mockData: {\n generateClientId: true,\n acceptMultipleEvents:\ \ true\n },\n mock: () => {\n mock('getRequestBody', '[{\"event\":\"\ page_view\"},{\"event\":\"view_item\"}]');\n mock('getCookieValues', []);\n\ \ },\n assert: (eventData) => {\n assertThat(eventData.client_id).isEqualTo('dcid.1.1747945830456.123456789');\n\ \ }\n },\n { \n description: 'Should NOT generate a random Client ID or\ \ use temporary Client ID if option is disabled',\n mockData: {\n generateClientId:\ \ false,\n acceptMultipleEvents: true\n },\n mock: () => {\n mock('getRequestBody',\ \ '[{\"event\":\"page_view\"},{\"event\":\"view_item\"}]');\n mock('getCookieValues',\ \ []);\n },\n assert: (eventData) => {\n assertThat(eventData.client_id).isFalsy();\n\ \ }\n }\n].forEach(scenario => {\n const copyMockData = JSON.parse(JSON.stringify(originalMockData));\n\ \ mergeObj(copyMockData, scenario.mockData);\n \n scenario.mock();\n \n /*\ \ \n For some reason when we use \"assertApi('claimRequest').wasCalled()\"\ \ AND we run all the tests,\n it produces the following error \"Tried to claim\ \ a request after a Client had returned. Calling claimRequest from a callback\ \ is not supported.\"\n If we run only this single test, the error does not\ \ occur.\n A workaround is to mock 'claimRequest' API and make a dummy assertion\ \ in the mocked function. This way we make sure it's been called.\n */\n let\ \ claimRequestWasCalled;\n mock('claimRequest', () => {\n claimRequestWasCalled\ \ = true;\n });\n \n mock('setResponseStatus', (responseStatus) => {\n assertThat(responseStatus).isEqualTo(200);\n\ \ });\n \n /*\n For some reason when we mock 'claimRequest', the following\ \ error is shown when the test is run: 'Request must be claimed before calling\ \ runContainer.'\n A workaround is to mock 'runContainer' and simply call its\ \ callbacks.\n */\n mock('runContainer', (eventData, onCompleteCallback, onStartCallback)\ \ => {\n assertThat(eventData.hasOwnProperty('_temp_dcid')).isFalse();\n \ \ scenario.assert(eventData);\n onCompleteCallback(); \n });\n \n runCode(copyMockData);\n\ \ \n callLater(() => {\n assertThat(claimRequestWasCalled).isTrue();\n \ \ assertApi('returnResponse').wasCalled();\n });\n});" setup: |- const JSON = require('JSON'); const Object = require('Object'); const callLater = require('callLater'); function mergeObj(target, source) { for (const key in source) { if (source.hasOwnProperty(key)) target[key] = source[key]; } return target; } const mockData = { responseStatusCode: 200, generateClientId: true, responseBody: 'timestamp', prolongCookies: true, httpOnlyCookie: false, responseBodyGet: false, acceptMultipleEvents: false, exposeFPIDCookie: false }; mock('getRequestHeader', (header) => { if (header === 'trace-id') return 'expectedTraceId'; }); mock('getTimestampMillis', 1747945830456); mock('generateRandom', 123456789); ___NOTES___ Created on 21/03/2021, 11:24:30