Skip to content

Instantly share code, notes, and snippets.

@sahava
Last active February 16, 2021 07:08
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save sahava/8b7bfd714aac4786e4d005774ec103ff to your computer and use it in GitHub Desktop.
Save sahava/8b7bfd714aac4786e4d005774ec103ff to your computer and use it in GitHub Desktop.
Advanced Universal Analytics (updated) - Custom Template demo
___INFO___
{
"type": "CLIENT",
"id": "cvt_temp_public_id",
"__wm": "VGVtcGxhdGUtQXV0aG9yX0FkdmFuY2VkVW5pdmVyc2FsQW5hbHl0aWNzLVNpbW8tQWhhdmE\u003d",
"categories": [
"ANALYTICS"
],
"version": 1,
"securityGroups": [],
"displayName": "Advanced Universal Analytics",
"brand": {
"id": "brand_dummy",
"displayName": "",
"thumbnail": "\u003d"
},
"description": "Unofficial template for Google\u0027s Universal Analytics. In addition to sending the hit to Google, there\u0027s also the option of proxying analytics.js, and for rewriting cookies in the HTTP response.",
"containerContexts": [
"SERVER"
]
}
___TEMPLATE_PARAMETERS___
[
{
"type": "CHECKBOX",
"name": "analyticsJs",
"checkboxText": "Proxy requests to /analytics.js",
"simpleValueType": true,
"defaultValue": false
},
{
"type": "CHECKBOX",
"name": "reWriteGa",
"checkboxText": "Set the Google Analytics Client ID cookie in the response",
"simpleValueType": true,
"subParams": [
{
"type": "TEXT",
"name": "gaCookieName",
"displayName": "Cookie Name",
"simpleValueType": true,
"defaultValue": "_ga",
"valueValidators": [
{
"type": "NON_EMPTY"
}
],
"enablingConditions": [
{
"paramName": "reWriteGa",
"paramValue": true,
"type": "EQUALS"
}
]
},
{
"type": "TEXT",
"name": "gaCookieDomain",
"displayName": "Cookie Domain",
"simpleValueType": true,
"defaultValue": "auto",
"valueValidators": [
{
"type": "NON_EMPTY"
}
],
"enablingConditions": [
{
"paramName": "reWriteGa",
"paramValue": true,
"type": "EQUALS"
}
]
},
{
"type": "TEXT",
"name": "gaCookieExpiration",
"displayName": "Cookie Expiration",
"simpleValueType": true,
"valueValidators": [
{
"type": "POSITIVE_NUMBER"
},
{
"type": "NON_EMPTY"
}
],
"valueUnit": "seconds",
"defaultValue": 63072000,
"enablingConditions": [
{
"paramName": "reWriteGa",
"paramValue": true,
"type": "EQUALS"
}
]
}
]
},
{
"type": "CHECKBOX",
"name": "reWriteGaExp",
"checkboxText": "Set the Google Optimize cookie (_gaexp) in the response",
"simpleValueType": true
},
{
"type": "CHECKBOX",
"name": "realAnonymization",
"checkboxText": "Prevent User IP And User-Agent From Being Passed Through",
"simpleValueType": true
}
]
___SANDBOXED_JS_FOR_SERVER___
const claimRequest = require('claimRequest');
const extractEventsFromMpv1 = require('extractEventsFromMpv1');
const getCookie = require('getCookieValues');
const getGoogleScript = require('getGoogleScript');
const getRemoteAddress = require('getRemoteAddress');
const getRequestHeader = require('getRequestHeader');
const getRequestPath = require('getRequestPath');
const getTimestamp = require('getTimestampMillis');
const isRequestMpv1 = require('isRequestMpv1');
const log = require('logToConsole');
const returnResponse = require('returnResponse');
const runContainer = require('runContainer');
const setCookie = require('setCookie');
const setPixelResponse = require('setPixelResponse');
const setResponseBody = require('setResponseBody');
const setResponseHeader = require('setResponseHeader');
const templateDataStorage = require('templateDataStorage');
// Get User-Agent and IP from incoming request
const ua = getRequestHeader('user-agent');
const ip = getRemoteAddress();
// Helper
const sendResponse = () => {
// Prevent CORS errors
const origin = getRequestHeader('Origin');
if (origin) {
setResponseHeader('Access-Control-Allow-Origin', origin);
setResponseHeader('Access-Control-Allow-Credentials', 'true');
}
returnResponse();
};
// Check if request is for the analytics.js library or a MP request
if (getRequestPath() === '/analytics.js' || isRequestMpv1()) {
// Claim the requst
claimRequest();
log('Valid Universal Analytics request, claimed...');
if (getRequestPath() === '/analytics.js' && data.analyticsJs) {
const now = getTimestamp();
const twoHoursAgo = now - 1000 * 60 * 60 * 2;
// If the library is NOT found in cache, or if the cache timestamp is over two hours old,
// fetch the resource from Google servers
if (!templateDataStorage.getItemCopy('analytics_js') || templateDataStorage.getItemCopy('analytics_js_timestamp') < twoHoursAgo) {
getGoogleScript('ANALYTICS', (script, metadata) => {
setResponseHeader('content-type', 'text/javascript');
setResponseBody(script);
log('Fetched analytics.js from Google servers, sending response...');
// Once the library is fetched, write it and the timestamp to template storage
templateDataStorage.setItemCopy('analytics_js', script);
templateDataStorage.setItemCopy('analytics_js_timestamp', now);
sendResponse();
});
} else {
// If the library is found in template storage and the timestamp hasn't expired,
// send the cached library in the response.
setResponseHeader('content-type', 'text/javascript');
setResponseBody(templateDataStorage.getItemCopy('analytics_js'));
log('Analytics.js found in template data cache, returning cached resource...');
sendResponse();
}
} else {
const events = extractEventsFromMpv1();
const max = events.length - 1;
events.forEach((event, i) => {
// Unless the event had IP and user-agent overrides, manually
// override them with the IP and user-agent from the request
// That way the GA collect call will appear to have originated
// from the user's browser / device.
//
// If the user has unchecked the pass-through option, do not send
// the IP address and User-Agent to Google.
if (!data.realAnonymization) {
log('Passing user IP and User-Agent through to Google');
if(!event.ip_override && ip) event.ip_override = ip;
if(!event.user_agent && ua) event.user_agent = ua;
}
// Pass the event to a virtual container
runContainer(event, () => {
if (i === max) {
// Rewrite _ga to avoid Safari 7 day expiration
if (data.reWriteGa) {
const ga = getCookie(data.gaCookieName);
if (ga && ga.length) {
log('GA cookie found, rewriting...');
setCookie(data.gaCookieName, ga[0], {
domain: data.gaCookieDomain,
'max-age': data.gaCookieExpiration,
path: '/',
secure: true,
sameSite: 'lax'
});
}
}
// Rewrite _gaexp to avoid Safari 7 day expiration
if (data.reWriteGaExp) {
const gaexp = getCookie('_gaexp');
if (gaexp && gaexp.length) {
setCookie('_gaexp', gaexp[0], {
domain: 'auto',
'max-age': 7257600,
path: '/',
secure: true,
sameSite: 'lax'
});
}
}
setPixelResponse();
log('Sending pixel response back to the browser...');
sendResponse();
}
});
});
}
}
___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": "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": "_ga"
},
{
"type": 1,
"string": "*"
},
{
"type": 1,
"string": "*"
},
{
"type": 1,
"string": "secure"
},
{
"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": "_gaexp"
},
{
"type": 1,
"string": "*"
},
{
"type": 1,
"string": "*"
},
{
"type": 1,
"string": "secure"
},
{
"type": 1,
"string": "any"
}
]
}
]
}
}
]
},
"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": "_ga"
},
{
"type": 1,
"string": "_gaexp"
}
]
}
}
]
},
"clientAnnotations": {
"isEditedByUser": true
},
"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
},
{
"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
}
]
___TESTS___
scenarios:
- name: AnalyticsJs proxied
code: |-
mock('getRequestPath', () => {
return '/analytics.js';
});
runCode(mockData);
assertApi('claimRequest').wasCalled();
assertApi('getGoogleScript').wasCalledWith('ANALYTICS', googleScriptCb);
assertApi('setResponseHeader').wasCalledWith('header', 'headerValue');
assertApi('setResponseHeader').wasCalledWith('content-type', 'text/javascript');
assertApi('setResponseBody').wasCalledWith('body');
assertApi('getRequestHeader').wasCalledWith('Origin');
assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Origin', 'Origin');
assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Credentials', 'true');
assertApi('returnResponse').wasCalled();
- name: Container run with GA data
code: |-
mock('getRequestPath', () => {
return '';
});
mock('isRequestMpv1', () => {
return true;
});
runCode(mockData);
assertApi('claimRequest').wasCalled();
assertApi('extractEventsFromMpv1').wasCalled();
assertApi('runContainer').wasCalledWith({event_name: 'page_view', ip_override: '1.2.3.4', user_agent: 'user-agent'}, runContainerCb);
assertApi('getCookieValues').wasCalledWith('_ga');
assertApi('getCookieValues').wasCalledWith('_gaexp');
assertApi('setCookie').wasCalledWith('_ga', '_ga', {domain: 'domain.com', 'max-age': 12345, path: '/', secure: true, sameSite: 'lax'});
assertApi('setCookie').wasCalledWith('_gaexp', '_gaexp', {domain: 'auto', 'max-age': 7257600, path: '/', secure: true, sameSite: 'lax'});
assertApi('setPixelResponse').wasCalled();
assertApi('getRequestHeader').wasCalledWith('Origin');
assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Origin', 'Origin');
assertApi('setResponseHeader').wasCalledWith('Access-Control-Allow-Credentials', 'true');
assertApi('returnResponse').wasCalled();
setup: "const mockData = {\n analyticsJs: true,\n reWriteGa: true,\n gaCookieName:\
\ '_ga',\n gaCookieDomain: 'domain.com',\n gaCookieExpiration: 12345,\n reWriteGaExp:\
\ true\n};\n\nmock('getRequestHeader', header => {\n return header;\n});\n\nmock('getRemoteAddress',\
\ () => {\n return '1.2.3.4';\n});\n\nlet googleScriptCb;\nmock('getGoogleScript',\
\ (s, cb) => {\n googleScriptCb = cb;\n cb('body', {header: 'headerValue'});\n\
});\n\nmock('extractEventsFromMpv1', () => {\n return [{\n event_name: 'page_view'\n\
\ }];\n});\n\nlet runContainerCb;\nmock('runContainer', (e, cb) => {\n runContainerCb\
\ = cb;\n cb();\n});\n\nmock('getCookieValues', c => {\n return [c];\n});\n \
\ "
___NOTES___
Created on 24/07/2020, 15:17:13
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment