___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": "GTM Loader", "__wm": "VGVtcGxhdGUtQXV0aG9yX0dUTUxvYWRlci1TaW1vLUFoYXZh", "categories": ["UTILITY"], "brand": { "id": "brand_dummy", "displayName": "", "thumbnail": "" }, "description": "Load the gtm.js library with the request path of your choosing. Automatically caches the container for a short period of time to reduce network egress costs.", "containerContexts": [ "SERVER" ] } ___TEMPLATE_PARAMETERS___ [ { "type": "TEXT", "name": "requestPath", "displayName": "Request Path", "simpleValueType": true, "defaultValue": "/gtm.js", "valueValidators": [ { "type": "REGEX", "args": [ "^/.*$" ] } ], "alwaysInSummary": true, "help": "Set to the path you\u0027ll use to fetch the GTM library. The default is \u003cstrong\u003e/gtm.js\u003c/strong\u003e, which means that a request to \u003cstrong\u003ehttps://your-gtm-server.com/gtm.js?id\u003dGTM-XYZ\u003c/strong\u003e will activate the Client." }, { "type": "CHECKBOX", "name": "overrideCid", "checkboxText": "Override Container ID", "simpleValueType": true, "help": "If you check this, you can provide a container ID which will always be loaded regardless of what (if anything) was included in the \u003cstrong\u003eid\u003c/strong\u003e parameter of the request." }, { "type": "TEXT", "name": "containerId", "displayName": "Container ID", "simpleValueType": true, "valueValidators": [ { "type": "REGEX", "args": [ "^GTM-.+$" ] } ], "enablingConditions": [ { "paramName": "overrideCid", "paramValue": true, "type": "EQUALS" } ] }, { "type": "CHECKBOX", "name": "encodingHeader", "checkboxText": "Set Encoding Header (Recommended for Cloud Run)", "simpleValueType": true, "help": "If you check this, a \"Content-Encoding\" header with the value \"gzip\" will be set in the response back to the browser." }, { "type": "TEXT", "name": "allowedOrigins", "displayName": "Allowed Origins", "simpleValueType": true, "defaultValue": "*", "valueValidators": [ { "type": "NON_EMPTY" } ], "help": "Enter a comma-separated list of origins (e.g. \u003cstrong\u003ehttps://www.teamsimmer.com,https://www.simoahava.com\u003c/strong\u003e or use the asterisk (\u003cstrong\u003e*\u003c/strong\u003e) as a wildcard. If the request comes from some other origin, the request will not be claimed. To pass an origin manually, add the \u003cstrong\u003e\u0026origin\u003d\u003cyour-origin\u003e\u003c/strong\u003e parameter to the request URL." } ] ___SANDBOXED_JS_FOR_SERVER___ const claimRequest = require('claimRequest'); const getRequestHeader = require('getRequestHeader'); const getRequestPath = require('getRequestPath'); const getRequestQueryParameters = require('getRequestQueryParameters'); const getTimestampMillis = require('getTimestampMillis'); const logToConsole = require('logToConsole'); const parseUrl = require('parseUrl'); const returnResponse = require('returnResponse'); const sendHttpGet = require('sendHttpGet'); const setResponseBody = require('setResponseBody'); const setResponseHeader = require('setResponseHeader'); const setResponseStatus = require('setResponseStatus'); const templateDataStorage = require('templateDataStorage'); const requestPath = getRequestPath(); const requestParams = getRequestQueryParameters(); const origin = getRequestHeader('origin') || (!!getRequestHeader('referer') && parseUrl(getRequestHeader('referer')).origin) || requestParams.origin; // Set max template storage cache to half of GTM container cache const cacheMaxTimeInMs = 450000; const containerId = data.containerId || requestParams.id || ''; // Preview parameters const gtm_auth = requestParams.gtm_auth; const gtm_debug = requestParams.gtm_debug; const gtm_preview = requestParams.gtm_preview; const previewRequest = !!(gtm_auth && gtm_debug && gtm_preview); const dataLayerVariableNameParameter = requestParams.l ? '&l=' + requestParams.l : ''; // Set names for storage const storedJs = 'gtm_js_' + containerId + (requestParams.l ? '_' + requestParams.l : ''); const storedHeaders = storedJs + '_headers'; const storedTimeout = storedJs + '_timeout'; const httpEndpoint = 'https://www.googletagmanager.com/gtm.js?fps=s'; const validateOrigin = () => { return data.allowedOrigins === '*' || data.allowedOrigins.split(',').indexOf(origin) > -1; }; const log = msg => { logToConsole('[GTM Loader] ' + msg); }; const sendResponse = (response, headers, statusCode) => { setResponseStatus(statusCode); setResponseBody(response); for (const key in headers) { // Do not set the "expires" and "date" headers if (['expires', 'date'].indexOf(key) === -1) setResponseHeader(key, headers[key]); } if (data.encodingHeader) { setResponseHeader('Content-Encoding', 'gzip'); } returnResponse(); }; const fetchPreviewContainer = () => { log('Fetching preview container for ' + containerId); sendHttpGet(httpEndpoint + '&id=' + containerId + '>m_auth=' + gtm_auth + '>m_debug=' + gtm_debug + '>m_preview=' + gtm_preview + dataLayerVariableNameParameter, (statusCode, headers, body) => { sendResponse(body, headers, statusCode); }, {timeout: 1500}); }; const fetchLiveContainer = () => { const now = getTimestampMillis(); const storageTimeout = now - cacheMaxTimeInMs; if (!templateDataStorage.getItemCopy(storedJs) || templateDataStorage.getItemCopy(storedTimeout) < storageTimeout) { log('Fetching live container from GTM servers for ' + containerId); sendHttpGet(httpEndpoint + '&id=' + containerId + dataLayerVariableNameParameter, (statusCode, headers, body) => { if (statusCode === 200) { templateDataStorage.setItemCopy(storedJs, body); templateDataStorage.setItemCopy(storedHeaders, headers); templateDataStorage.setItemCopy(storedTimeout, now); } sendResponse(body, headers, statusCode); }, {timeout: 1500}); } else { log('Fetching live container from template cache for ' + containerId); sendResponse( templateDataStorage.getItemCopy(storedJs), templateDataStorage.getItemCopy(storedHeaders), 200 ); } }; if (requestPath === data.requestPath) { if (!containerId.match('^GTM-.+$')) { log('Invalid or missing container ID'); return; } if (!validateOrigin()) { log('Request originated from invalid origin'); return; } log('Processing request for ' + containerId); claimRequest(); if (previewRequest) { fetchPreviewContainer(); } else { fetchLiveContainer(); } log('Processed request for ' + containerId); } ___SERVER_PERMISSIONS___ [ { "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": "return_response", "versionId": "1" }, "param": [] }, "isRequired": true }, { "instance": { "key": { "publicId": "logging", "versionId": "1" }, "param": [ { "key": "environments", "value": { "type": 1, "string": "debug" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "access_template_storage", "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": "send_http", "versionId": "1" }, "param": [ { "key": "allowedUrls", "value": { "type": 1, "string": "specific" } }, { "key": "allowGoogleDomains", "value": { "type": 8, "boolean": true } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true } ] ___TESTS___ scenarios: [] ___NOTES___ Created on 05/05/2021, 15:48:06