Skip to content

Instantly share code, notes, and snippets.

@cbarley10
Created November 30, 2021 16:49
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 cbarley10/f1335a260ed30b09c63e7223f18202df to your computer and use it in GitHub Desktop.
Save cbarley10/f1335a260ed30b09c63e7223f18202df to your computer and use it in GitHub Desktop.
Serverside Container for Klaviyo Medium Blog Article
___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": "Klaviyo API Wrapper Template",
"brand": {
"id": "brand_dummy",
"displayName": "",
"thumbnail": ""
},
"description": "Klaviyo Wrapper for sending events mapped through GA4 into Klaviyo though GTM\u0027s Serverside Container technology.",
"containerContexts": [
"SERVER"
]
}
___TEMPLATE_PARAMETERS___
[
{
"type": "TEXT",
"name": "public_key",
"displayName": "Klaviyo Public API Key",
"simpleValueType": true,
"help": "Use your Public API Key from your Account Settings. To locate your Public Key, go here: https://help.klaviyo.com/hc/en-us/articles/115005062267-Manage-Your-Account-s-API-Keys#your-public-api-key-site-id2",
"valueValidators": [
{
"type": "NON_EMPTY"
}
],
"notSetText": "Public API Key cannot be blank!"
}
]
___SANDBOXED_JS_FOR_SERVER___
//////////////
// Packages //
//////////////
const JSON = require("JSON");
const log = require("logToConsole");
const parseUrl = require('parseUrl');
const toBase64 = require('toBase64');
const sendHttpRequest = require('sendHttpRequest');
const setResponseBody = require('setResponseBody');
const getAllEventData = require('getAllEventData');
const setResponseStatus = require('setResponseStatus');
const encodeUriComponent = require('encodeUriComponent');
///////////////
// Constants //
///////////////
const KL_PUBLIC_KEY = data.public_key;
const EVENT_DATA = getAllEventData();
const HEADERS = {
'user-agent': 'Klaviyo/GTM-Serverside',
'Accept': 'text/html',
'Content-Type': 'application/x-www-form-urlencoded'
};
/////////////
// Helpers //
/////////////
// Encode JSON payload to Base64 in order to send event to Klaviyo
const encodePayload = payload => encodeUriComponent(toBase64(JSON.stringify(payload)));
// Return only unique values in an array
const returnUniques = (value, index, self) => {
return self.indexOf(value) === index;
};
// pull out nested fields in "items" array of the ecommerce object, and add them as top-level fields
const flatten = event => {
const flattenedObj = event.ecommerce;
const objectKeys = [];
const ecommerceObjectItems = event.ecommerce.items;
for(let property in ecommerceObjectItems[0]){
objectKeys.push(property);
}
objectKeys.forEach(item => {
flattenedObj[item] = ecommerceObjectItems.map(i => i[item]).filter(returnUniques);
});
return flattenedObj;
};
// Normalize event names so that they look better in Klaviyo and conform to Klaviyo naming convention
const normalizeEventNames = name => {
if (name === "add_to_cart") {
return "Added To Cart";
} else if (name === "view_item") {
return "Viewed Product";
} else if (name === "begin_checkout") {
return "Started Checkout";
} else if (name === "page_view"){
return "Page Viewed";
} else if (name.indexOf("_") > -1) {
return name.split("_").map(item => item.charAt(0).toUpperCase() + item.slice(1)).join(" ");
}
return name.charAt(0).toUpperCase() + name.slice(1);
};
// Build track payload
const buildPayload = (event, user) => {
const customerProperties = {};
if (user.email) {
customerProperties["$email"] = user.email;
}
if (user.exchange_id) {
customerProperties["$exchange_id"] = user.exchange_id;
}
return {
"token": KL_PUBLIC_KEY,
"event": normalizeEventNames(eventName),
"customer_properties": customerProperties,
"properties": event.hasOwnProperty("ecommerce") ? flatten(EVENT_DATA) : EVENT_DATA
};
};
// Handle _kx and utm_email parameters
const getUser = (url, eventData) => {
const parsedUrl = parseUrl(url) || "";
const userEmail = eventData.email;
if (parsedUrl && parsedUrl.searchParams._kx){
const kx = parsedUrl.searchParams._kx;
return {
"email": "",
"exchange_id": kx
};
} else if (parsedUrl && parsedUrl.searchParams.utm_email){
const email = parsedUrl.searchParams.utm_email;
return {
"email": email,
"exchange_id": ""
};
} else if (userEmail){
return {
"email": userEmail,
"exchange_id": ""
};
} else {
return {
"email": "",
"exchange_id": ""
};
}
};
///////////////////
// API Functions //
///////////////////
const sendTrackOrIdentifyRequest = (method, payload) => {
const encodedPayload = encodePayload(payload);
const url = "https://a.klaviyo.com/api/" + method + "?data=" + encodedPayload;
return sendHttpRequest(url, (statusCode, headers, body) => {}, {headers: HEADERS, method: 'GET'});
};
///////////////
// Execution //
///////////////
log("==== STARTING SCRIPT ====");
const user = getUser(EVENT_DATA.page_location, EVENT_DATA);
const eventName = EVENT_DATA.event_name;
// if page_location contains _kx identify user and send Active On Site Metric
if (user.email || user.exchange_id){
log("_kx or email were found. Tracking an event");
const payload = buildPayload(EVENT_DATA, user);
sendTrackOrIdentifyRequest("track", payload);
} else {
log("No customer $exchange_id, $email, or $id found. Not sending Klaviyo any data...");
}
log("==== ENDING SCRIPT ====");
// Call data.gtmOnSuccess when the tag is finished.
data.gtmOnSuccess();
___SERVER_PERMISSIONS___
[
{
"instance": {
"key": {
"publicId": "logging",
"versionId": "1"
},
"param": [
{
"key": "environments",
"value": {
"type": 1,
"string": "debug"
}
}
]
},
"clientAnnotations": {
"isEditedByUser": true
},
"isRequired": true
},
{
"instance": {
"key": {
"publicId": "send_http",
"versionId": "1"
},
"param": [
{
"key": "allowedUrls",
"value": {
"type": 1,
"string": "any"
}
}
]
},
"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": "read_event_data",
"versionId": "1"
},
"param": [
{
"key": "eventDataAccess",
"value": {
"type": 1,
"string": "any"
}
}
]
},
"clientAnnotations": {
"isEditedByUser": true
},
"isRequired": true
}
]
___TESTS___
scenarios: []
___NOTES___
Created on 11/30/2021, 11:48:09 AM
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment