___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": "Firestore Web Tag", "categories": ["UTILITY"], "brand": { "id": "brand_dummy", "displayName": "", "thumbnail": "\u003d\u003d" }, "description": "The Firestore Web Tag template allows you to create a list of the rules on how to change the user property.", "containerContexts": [ "WEB" ] } ___TEMPLATE_PARAMETERS___ [ { "type": "TEXT", "name": "serverUrl", "displayName": "Server SIde GTM URL", "simpleValueType": true, "valueHint": "https://", "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "REGEX", "args": [ "^https:\\/\\/.*" ] } ] }, { "type": "TEXT", "name": "userKey", "displayName": "User key", "simpleValueType": true, "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "REGEX", "args": [ "^[^\\/]*$" ] }, { "type": "REGEX", "args": [ "^(?!\\.$).*$" ] }, { "type": "REGEX", "args": [ "^(?!\\.\\.$).*$" ] } ], "help": "\u003ca href\u003d\"https://firebase.google.com/docs/firestore/quotas#limits\"\u003eDocumentation page\u003c/a\u003e about key constrains" }, { "type": "PARAM_TABLE", "name": "rules", "displayName": "User properties you want to change", "paramTableColumns": [ { "param": { "type": "TEXT", "name": "name", "displayName": "Property Name", "simpleValueType": true }, "isUnique": true }, { "param": { "type": "SELECT", "name": "ruleType", "displayName": "Action", "macrosInSelect": false, "selectItems": [ { "value": 1, "displayValue": "Add" }, { "value": 2, "displayValue": "Set" } ], "simpleValueType": true }, "isUnique": false }, { "param": { "type": "TEXT", "name": "value", "displayName": "Value", "simpleValueType": true, "enablingConditions": [], "help": "If Action is «‎Set» the property will be changed. If Action is «‎Add» the user property will be increased on value. The value can be constant or GTM Variable.", "valueValidators": [ { "type": "NON_EMPTY" } ] }, "isUnique": false }, { "param": { "type": "SELECT", "name": "returnValue", "displayName": "Return value", "macrosInSelect": false, "selectItems": [ { "value": "new", "displayValue": "Updated value" }, { "value": "old", "displayValue": "Previous value" } ], "simpleValueType": true, "valueValidators": [], "defaultValue": "new", "help": "Should server return updated value or previous value" }, "isUnique": false } ], "notSetText": "User properties not set. Server tag will change nothing but return current user state." }, { "type": "GROUP", "name": "additionalSettings", "displayName": "Additional settings", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "LABEL", "name": "Cookie settings", "displayName": "Cookie settings" }, { "type": "CHECKBOX", "name": "setCookie", "checkboxText": "Set cookie after user properties will be updated", "simpleValueType": true, "defaultValue": true }, { "type": "TEXT", "name": "cookiePrefix", "displayName": "Cookie Prefix", "simpleValueType": true, "defaultValue": "user", "help": "Server tag will set cookie for each user property. If cookie prefix is not empty it will be added for each property name.", "notSetText": "If not set «user» prefix will be used" }, { "type": "LABEL", "name": "DataLayer settings", "displayName": "DataLayer settings" }, { "type": "TEXT", "name": "dataLayerEventName", "displayName": "DataLayer Event Name", "simpleValueType": true, "help": "The name of event will be pushed after user properties are changed", "defaultValue": "firestore_updated", "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "type": "TEXT", "name": "eventName", "displayName": "Event Type", "simpleValueType": true, "help": "This field can be used if you have a few Firestore tags and what to differentiate dataLayer pushes after user properties are changed. It\u0027s a optional filed." } ] } ] ___SANDBOXED_JS_FOR_WEB_TEMPLATE___ // Enter your template code here. const log = require("logToConsole"); const sendPixel = require("sendPixel"); const createQueue = require("createQueue"); const makeString = require("makeString"); const encodeUriComponent = require("encodeUriComponent"); const isConsentGranted = require("isConsentGranted"); const addConsentListener = require("addConsentListener"); const parseUrl = require("parseUrl"); const dataLayerPush = createQueue("dataLayer"); const version = 1; const onSuccess = () => { // Prepare dataLayer push let dataLayerPushObject = { event: data.dataLayerEventName, }; if (data.eventName) dataLayerPushObject.firestore_event_type = data.eventName; dataLayerPush(dataLayerPushObject); data.gtmOnSuccess(); }; const onFailure = () => { data.gtmOnFailure(); }; const getFinalUrl = (serverUrl) => { let rulesArray = []; if (data.hasOwnProperty("rules")) { data.rules.forEach((rule) => { const returnValue = rule.returnValue === "new" ? "" : ",1"; rulesArray.push( "r=" + encodeUriComponent(rule.name) + "," + rule.ruleType + "," + encodeUriComponent(makeString(rule.value)) + returnValue ); }); } let finalUrl = serverUrl + "/firestore?v=" + version; // Add user key finalUrl += "&u=" + data.userKey; // Add rules if (rulesArray.length > 0) finalUrl += "&" + rulesArray.join("&"); // Add cookies if (!data.setCookie) finalUrl += "&s=0"; if (data.cookiePrefix && data.cookiePrefix !== "user") finalUrl += "&sp=" + encodeUriComponent(data.cookiePrefix); if (data.eventName) finalUrl += "&e=" + encodeUriComponent(data.eventName); return finalUrl; }; const clearServerUrl = () => { const lastIndex = data.serverUrl.lastIndexOf("/"); if (lastIndex == data.serverUrl.length - 1) return data.serverUrl.substring(0, lastIndex); return data.serverUrl; }; const checkServerUrl = (serverUrl) => { const urlObject = parseUrl(serverUrl); if (urlObject && urlObject.hasOwnProperty("origin")) return urlObject.origin === serverUrl; return false; }; const checkUserKey = () => { if (data.userKey.indexOf("/") > -1) return false; if (data.userKey == "." || data.userKey == "..") return false; return true; }; const serverUrl = clearServerUrl(); if (!checkServerUrl(serverUrl) || !checkUserKey()) { data.gtmOnFailure(); } else { const finalUrl = getFinalUrl(serverUrl); if (!isConsentGranted("ad_storage")) { let wasCalled = false; addConsentListener("ad_storage", (consentType, granted) => { if (wasCalled || !granted) return; wasCalled = true; sendPixel(finalUrl, onSuccess, onFailure); }); data.gtmOnSuccess(); } else { sendPixel(finalUrl, onSuccess, onFailure); } } ___WEB_PERMISSIONS___ [ { "instance": { "key": { "publicId": "logging", "versionId": "1" }, "param": [ { "key": "environments", "value": { "type": 1, "string": "debug" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "send_pixel", "versionId": "1" }, "param": [ { "key": "allowedUrls", "value": { "type": 1, "string": "any" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "access_globals", "versionId": "1" }, "param": [ { "key": "keys", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "dataLayer" }, { "type": 8, "boolean": true }, { "type": 8, "boolean": true }, { "type": 8, "boolean": true } ] } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "access_consent", "versionId": "1" }, "param": [ { "key": "consentTypes", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "consentType" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" } ], "mapValue": [ { "type": 1, "string": "ad_storage" }, { "type": 8, "boolean": true }, { "type": 8, "boolean": false } ] } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true } ] ___TESTS___ scenarios: - name: Server URL incorrect code: "mockData.serverUrl = 'https://127.0.0.1eqwscadfwer2@#4'; \nrunCode(mockData);\n\ assertApi('gtmOnSuccess').wasNotCalled();\nassertApi('gtmOnFailure').wasCalled();\n" - name: User Key incorrect code: |- mockData.userKey = '..' ; runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasNotCalled(); assertApi('gtmOnFailure').wasCalled(); - name: One rule add code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,1,val1'; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: One rule add, old value code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1%25%26%D1%8E,1,%25%26%2C%D1%86,1'; mockData.rules = [{"name":"var1%&ю","ruleType":1,"value":"%&,ц","returnValue":"old"}] ; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: One rule set code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,2,val1'; mockData.rules = [{"name":"var1","ruleType":2,"value":"val1","returnValue":"new"}] ; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: One rule set old value code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1%25%26%D1%8E,2,%25%26%2C%D1%86,1'; mockData.rules = [{"name":"var1%&ю","ruleType":2,"value":"%&,ц","returnValue":"old"}] ; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: Few rules code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,1,val1&r=var2,2,val2&r=var3,2,val3'; mockData.rules = [{"name":"var1","ruleType":1,"value":"val1","returnValue":"new"},{"name":"var2","ruleType":2,"value":"val2","returnValue":"new"}, {"name":"var3","ruleType":2,"value":"val3","returnValue":"new"} ] ; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: Few rules old and new values code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,1,val1,1&r=var2,2,val2&r=var3,2,val3,1'; mockData.rules = [{"name":"var1","ruleType":1,"value":"val1","returnValue":"old"},{"name":"var2","ruleType":2,"value":"val2","returnValue":"new"}, {"name":"var3","ruleType":2,"value":"val3","returnValue":"old"} ] ; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: Change dataLayer name code: "const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,1,val1';\n\ mockData.dataLayerEventName = \"firestore_updated_new\";\n\nmock('sendPixel',\ \ function(url, onSuccess, onFailure) {\n assertThat(url).isEqualTo(expectedServerUrl);\n\ \ onSuccess();\n});\n\nconst expectedDataLayerEvent = {\"event\":\"firestore_updated_new\"\ };\nmock(\"createQueue\", function (name) {\n return function (event) {\n \ \ assertThat(event).isEqualTo(expectedDataLayerEvent);\n };\n});\n \n\n\ // Call runCode to run the template's code.\nrunCode(mockData);\n\n// Verify that\ \ the tag finished successfully.\nassertApi('gtmOnSuccess').wasCalled();" - name: Change dataLayer event type code: "const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,1,val1&e=purchase';\n\ mockData.eventName = \"purchase\";\n\nmock('sendPixel', function(url, onSuccess,\ \ onFailure) {\n assertThat(url).isEqualTo(expectedServerUrl);\n onSuccess();\n\ });\n\nconst expectedDataLayerEvent = {\"event\":\"firestore_updated\",\"firestore_event_type\"\ :\"purchase\"};\nmock(\"createQueue\", function (name) {\n return function (event)\ \ {\n assertThat(event).isEqualTo(expectedDataLayerEvent);\n };\n});\n \ \ \n\n// Call runCode to run the template's code.\nrunCode(mockData);\n\n//\ \ Verify that the tag finished successfully.\nassertApi('gtmOnSuccess').wasCalled();" - name: Don't set cookies code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,1,val1&s=0'; mockData.setCookie = false; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: Change cookie prefix code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,1,val1&sp=_gtm'; mockData.cookiePrefix = "_gtm"; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: Consent is not granted code: | mock('isConsentGranted', function(consent){ if(consent==="ad_storage") return false; }); mock('addConsentListener', function(consent, callback){ log("consent",consent); assertThat(consent).isEqualTo("ad_storage"); //callback(); }); runCode(mockData); assertApi('sendPixel').wasNotCalled(); assertApi('gtmOnSuccess').wasCalled(); - name: Consent is not granted, and granted later code: | mock('isConsentGranted', function(consent){ if(consent==="ad_storage") return false; }); mock('addConsentListener', function(consent, callback){ log("addConsentListener consent",consent); assertThat(consent).isEqualTo("ad_storage"); callback("ad_storage", "granted"); }); const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123&r=var1,1,val1'; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); runCode(mockData); assertApi('sendPixel').wasCalled(); assertApi('gtmOnSuccess').wasCalled(); - name: No rules code: |- const expectedServerUrl = 'https://127.0.0.1/firestore?v=1&u=123'; let mockNoRulesData = {"serverUrl":"https://127.0.0.1","dataLayerEventName":"firestore_updated","setCookie":true,"userKey":"123","cookiePrefix":"user","gtmTagId":2147483646,"gtmEventId":3}; mock('sendPixel', function(url, onSuccess, onFailure) { assertThat(url).isEqualTo(expectedServerUrl); onSuccess(); }); // Call runCode to run the template's code. runCode(mockNoRulesData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); setup: |- const log = require("logToConsole"); let mockData = {"serverUrl":"https://127.0.0.1","dataLayerEventName":"firestore_updated","rules":[{"name":"var1","ruleType":1,"value":"val1","returnValue":"new"}],"setCookie":true,"userKey":"123","cookiePrefix":"user","gtmTagId":2147483646,"gtmEventId":3}; mock('isConsentGranted', true); ___NOTES___ Created on 08/05/2022, 21:10:59