___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": "FingerprintJS Pro", "categories": [ "SESSION_RECORDING" ], "brand": { "id": "brand_dummy", "displayName": "", "thumbnail": "\u003d" }, "description": "This template adds FingerprintJS Pro agent, the highest accuracy device identification. Visit https://fingerprint.com/ for more information and get you public API key.", "containerContexts": [ "WEB" ] } ___TEMPLATE_PARAMETERS___ [ { "type": "TEXT", "name": "apiKey", "simpleValueType": true, "displayName": "Public API key", "valueValidators": [ { "type": "NON_EMPTY" } ], "help": "Your public API key that authenticates the agent with the API" }, { "type": "SELECT", "name": "region", "displayName": "Region", "macrosInSelect": false, "selectItems": [ { "value": "", "displayValue": "Global" }, { "value": "eu", "displayValue": "EU" }, { "value": "ap", "displayValue": "Asia (Mumbai)" } ], "simpleValueType": true, "help": "Which region to use" }, { "type": "GROUP", "name": "additionalFields", "displayName": "Additional Settings", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "TEXT", "name": "scriptUrlPattern", "displayName": "Script Url Pattern", "simpleValueType": true, "help": "Pattern of the JS agent script URL." }, { "type": "TEXT", "name": "endpoint", "displayName": "Endpoint", "simpleValueType": true, "help": "Server API URL. Should be only used with Subdomain integration." }, { "type": "TEXT", "name": "tag", "displayName": "tag", "simpleValueType": true, "help": "A customer-provided value or an object that will be saved together with the identification event" }, { "type": "TEXT", "name": "linkedId", "displayName": "linkedId", "simpleValueType": true, "help": "A way of linking current identification event with a custom identifier" }, { "type": "CHECKBOX", "name": "extendedResult", "checkboxText": "Extended result", "simpleValueType": true }, { "type": "TEXT", "name": "resultCustomName", "displayName": "Result custom name", "simpleValueType": true, "help": "You can change result variable name that emits in dataLayer", "defaultValue": "FingerprintJSProResult", "valueValidators": [ { "type": "NON_EMPTY" } ] } ] } ] ___SANDBOXED_JS_FOR_WEB_TEMPLATE___ // Enter your template code here. const log = require('logToConsole'); const injectScript = require('injectScript'); const queryPermission = require('queryPermission'); const callInWindow = require('callInWindow'); const createQueue = require('createQueue'); const copyFromWindow = require('copyFromWindow'); const url = 'https://opencdn.fpjs.sh/fingerprintjs-pro-gtm/v0/iife.min.js'; log('data =', data); const loadOptions = { apiKey: data.apiKey, integrationInfo: 'fingerprintjs-pro-gtm-template/0.0.1', }; if (data.region) { loadOptions.region = data.region; } if (data.endpoint) { loadOptions.endpoint = data.endpoint; } if (data.scriptUrlPattern) { loadOptions.scriptUrlPattern = data.scriptUrlPattern; } const getOptions = {}; if (data.tag) { getOptions.tag = data.tag; } if (data.linkedId) { getOptions.linkedId = data.linkedId; } if (data.extendedResult) { getOptions.extendedResult = true; } log('loadOptions =', loadOptions); log('getOptions=', getOptions); const onSuccess = () => { log('FingerprintJS: Script loaded successfully.'); const onFpJsLoad = (result) => { log('result', result); const dataLayerPush = createQueue('dataLayer'); const event = {event: 'FingerprintJSPro.loaded'}; event[data.resultCustomName] = result; dataLayerPush(event); data.gtmOnSuccess(); }; callInWindow('FingerprintjsProGTM.load', loadOptions, getOptions, onFpJsLoad); }; // If the script fails to load, log a message and signal failure const onFailure = () => { log('FingerprintJS: Script load failed.'); data.gtmOnFailure(); }; if (queryPermission('inject_script', url) && queryPermission('access_globals', 'execute', 'FingerprintjsProGTM.load')) { log('try to load'); injectScript(url, onSuccess, onFailure); } else { log('FingerprintJS: Script load failed due to permissions mismatch.'); data.gtmOnFailure(); } ___WEB_PERMISSIONS___ [ { "instance": { "key": { "publicId": "logging", "versionId": "1" }, "param": [ { "key": "environments", "value": { "type": 1, "string": "debug" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "inject_script", "versionId": "1" }, "param": [ { "key": "urls", "value": { "type": 2, "listItem": [ { "type": 1, "string": "https://opencdn.fpjs.sh/fingerprintjs-pro-gtm/v0/iife.min.js" } ] } } ] }, "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": "FingerprintjsProGTM.load" }, { "type": 8, "boolean": false }, { "type": 8, "boolean": false }, { "type": 8, "boolean": true } ] }, { "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": false } ] } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true } ] ___TESTS___ scenarios: - name: check loading code: |- const mockData = { apiKey: 'aspodkasodk', }; mock('injectScript', function(url, onSuccess, onFail) { onSuccess(); }); mock('callInWindow', function(fname, getOptions, loadOptions, callback) { callback({visitorId: 'qwerty'}); }); // Call runCode to run the template's code. runCode(mockData); assertApi('injectScript').wasCalled(); assertApi('createQueue').wasCalled(); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: check loadOptions ang getOptions with minimum params code: |- const mockData = { apiKey: 'aspodkasodk', }; const expectedLoadOptions = { apiKey: 'aspodkasodk', integrationInfo: 'fingerprintjs-pro-gtm-template/0.0.1', }; mock('injectScript', function(url, onSuccess, onFail) { onSuccess(); }); mock('callInWindow', function(fname, loadOptions, getOptions, callback) { assertThat(loadOptions).isEqualTo(expectedLoadOptions); assertThat(getOptions).isEqualTo({}); callback({visitorId: 'qwerty'}); }); // Call runCode to run the template's code. runCode(mockData); assertApi('gtmOnSuccess').wasCalled(); - name: check loadOptions ang getOptions with maximum params code: |- const mockData = { apiKey: 'aspodkasodk', region: 'eu', scriptUrlPattern: 'https://domain.some/v//loader_v.js', endpoint: 'https://end.point/', tag: 'myTag', linkedId: 'some-id', extendedResult: true, }; const expectedLoadOptions = { apiKey: 'aspodkasodk', integrationInfo: 'fingerprintjs-pro-gtm-template/0.0.1', region: 'eu', endpoint: 'https://end.point/', scriptUrlPattern: 'https://domain.some/v//loader_v.js', }; const expectedGetOptions = { tag: 'myTag', linkedId: 'some-id', extendedResult: true, }; mock('injectScript', function(url, onSuccess, onFail) { onSuccess(); }); mock('callInWindow', function(fname, loadOptions, getOptions, callback) { assertThat(loadOptions).isEqualTo(expectedLoadOptions); assertThat(getOptions).isEqualTo(expectedGetOptions); callback({visitorId: 'qwerty'}); }); // Call runCode to run the template's code. runCode(mockData); assertApi('gtmOnSuccess').wasCalled(); - name: check visitorIdCustomName field works code: |- const mockData = { apiKey: 'aspodkasodk', visitorIdCustomName: 'fingerprintJsProVisitorId', resultCustomName: 'result' }; mock('injectScript', function(url, onSuccess, onFail) { onSuccess(); }); mock('callInWindow', function(fname, getOptions, loadOptions, callback) { callback({visitorId: 'qwerty'}); }); mock('createQueue', function(name) { return function(params) { assertThat(params).isEqualTo({ event: 'FingerprintJSPro.loaded', result: {visitorId: 'qwerty'}, }); }; }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); ___NOTES___ Created on 30.03.2022, 20:54:30