___INFO___ { "type": "TAG", "id": "spectacle_server_template", "version": 1, "securityGroups": [], "displayName": "Spectacle - Server Side", "categories": [ "ATTRIBUTION", "ANALYTICS", "ADVERTISING" ], "brand": { "id": "spectacle", "displayName": "Spectacle", "thumbnail": "" }, "containerContexts": [ "SERVER" ] } ___TEMPLATE_PARAMETERS___ [ { "type": "TEXT", "name": "workspaceId", "displayName": "Spectacle Workspace ID", "simpleValueType": true, "alwaysInSummary": true, "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "REGEX", "args": [ "^ws_[a-zA-Z0-9-_]+$" ], "errorMessage": "Workspace ID must start with \u0027ws_\u0027" } ] }, { "type": "SELECT", "name": "methodType", "displayName": "Method Type", "selectItems": [ { "value": "page", "displayValue": "Page - Track page views" }, { "value": "identify", "displayValue": "Identify - Identify users" }, { "value": "track", "displayValue": "Track - Track events" }, { "value": "group", "displayValue": "Group - Link user to Company" } ], "simpleValueType": true, "defaultValue": "page" }, { "type": "CHECKBOX", "name": "lensEnabled", "displayName": "Spectacle Lens", "checkboxText": "Enable Lens", "help": "Spectacle Lens can reveal company data of anonymous visitors. Learn more", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "page", "type": "EQUALS" } ] }, { "type": "GROUP", "name": "advanced", "displayName": "Advanced Configuration", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "TEXT", "name": "baseUrl", "displayName": "API Base URL", "simpleValueType": true, "defaultValue": "https://t.spectaclehq.com", "valueValidators": [ { "type": "NON_EMPTY" }, { "type": "REGEX", "args": [ "^https://.+" ], "errorMessage": "Must be HTTPS URL" } ] }, { "type": "TEXT", "name": "cookieDomain", "displayName": "Cookie Domain", "simpleValueType": true, "help": "Leave empty for auto-detection (e.g., \u0027.example.com\u0027)" } ] }, { "type": "TEXT", "name": "userId", "displayName": "User ID", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "identify", "type": "EQUALS" } ] }, { "type": "TEXT", "name": "email", "displayName": "Email", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "identify", "type": "EQUALS" } ], "valueValidators": [ { "type": "REGEX", "args": [ "^[^@]+@[^@]+\\.[^@]+$" ], "errorMessage": "Must be a valid email address" } ] }, { "type": "TEXT", "name": "firstName", "displayName": "First Name", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "identify", "type": "EQUALS" } ] }, { "type": "TEXT", "name": "lastName", "displayName": "Last Name", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "identify", "type": "EQUALS" } ] }, { "type": "SIMPLE_TABLE", "name": "userTraits", "displayName": "Additional User Traits", "simpleTableColumns": [ { "defaultValue": "", "displayName": "Key", "name": "key", "type": "TEXT" }, { "defaultValue": "", "displayName": "Value", "name": "value", "type": "TEXT" } ], "newRowButtonText": "Add Trait", "enablingConditions": [ { "paramName": "methodType", "paramValue": "identify", "type": "EQUALS" } ] }, { "type": "TEXT", "name": "eventName", "displayName": "Event Name", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "track", "type": "EQUALS" } ], "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "type": "GROUP", "name": "revenueSettings", "displayName": "Revenue", "groupStyle": "ZIPPY_OPEN", "enablingConditions": [ { "paramName": "methodType", "paramValue": "track", "type": "EQUALS" } ], "subParams": [ { "type": "CHECKBOX", "name": "useGA4EcomData", "checkboxText": "Use GA4 ecommerce data", "simpleValueType": true }, { "type": "GROUP", "name": "manualRevenueData", "displayName": "Data", "groupStyle": "ZIPPY_OPEN", "enablingConditions": [ { "paramName": "useGA4EcomData", "paramValue": false, "type": "EQUALS" } ], "subParams": [ { "type": "RADIO", "name": "revenueFormat", "displayName": "Revenue format", "radioItems": [ { "value": "majorUnits", "displayValue": "Full amount - (Major units, e.g., 99.95)" }, { "value": "cents", "displayValue": "Cents - (Minor units e.g., 9995)" } ], "simpleValueType": true, "defaultValue": "majorUnits", "enablingConditions": [ { "paramName": "methodType", "paramValue": "track", "type": "EQUALS" } ] }, { "type": "TEXT", "name": "revenueMajorUnits", "displayName": "Revenue Amount", "simpleValueType": true, "valueHint": "99.95", "enablingConditions": [ { "paramName": "revenueFormat", "paramValue": "majorUnits", "type": "EQUALS" } ] }, { "type": "TEXT", "name": "revenue", "displayName": "Revenue Amount", "simpleValueType": true, "valueHint": "9995", "enablingConditions": [ { "paramName": "revenueFormat", "paramValue": "cents", "type": "EQUALS" } ] }, { "type": "TEXT", "name": "currency", "displayName": "Currency", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "track", "type": "EQUALS" } ] }, { "type": "TEXT", "name": "transactionId", "displayName": "Transaction id", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "track", "type": "EQUALS" } ], "help": "A unique transaction id. Used for de-duplication of events with revenue. If no transaction id is provided, Spectacle automatically uses a combination of the visitor id and timestamp as the transaction id. For refunds, make sure to use a different transaction id than the original transaction id being refunded." } ] } ] }, { "type": "SIMPLE_TABLE", "name": "eventProperties", "displayName": "Event Properties", "simpleTableColumns": [ { "defaultValue": "", "displayName": "Key", "name": "key", "type": "TEXT" }, { "defaultValue": "", "displayName": "Value", "name": "value", "type": "TEXT" } ], "newRowButtonText": "Add Property", "enablingConditions": [ { "paramName": "methodType", "paramValue": "track", "type": "EQUALS" } ] }, { "type": "TEXT", "name": "groupId", "displayName": "Group ID", "simpleValueType": true, "enablingConditions": [ { "paramName": "methodType", "paramValue": "group", "type": "EQUALS" } ], "valueValidators": [ { "type": "NON_EMPTY" } ] }, { "type": "SIMPLE_TABLE", "name": "groupTraits", "displayName": "Group Traits", "simpleTableColumns": [ { "defaultValue": "", "displayName": "Key", "name": "key", "type": "TEXT" }, { "defaultValue": "", "displayName": "Value", "name": "value", "type": "TEXT" } ], "newRowButtonText": "Add Trait", "enablingConditions": [ { "paramName": "methodType", "paramValue": "group", "type": "EQUALS" } ] }, { "type": "CHECKBOX", "name": "debugMode", "checkboxText": "Enable debug logging", "simpleValueType": true } ] ___SANDBOXED_JS_FOR_SERVER___ const sendHttpRequest = require('sendHttpRequest'); const JSON = require('JSON'); const getEventData = require('getEventData'); const getAllEventData = require('getAllEventData'); const getCookieValues = require('getCookieValues'); const setCookie = require('setCookie'); const getRequestHeader = require('getRequestHeader'); const sha256Sync = require('sha256Sync'); const getTimestampMillis = require('getTimestampMillis'); const logToConsole = require('logToConsole'); const generateRandom = require('generateRandom'); const parseUrl = require('parseUrl'); const getType = require('getType'); const makeString = require('makeString'); const makeInteger = require('makeInteger'); const makeNumber = require('makeNumber'); const Math = require('Math'); const getRemoteAddress = require('getRemoteAddress'); // Constants matching your pixel implementation const ANON_COOKIE_KEY = 'sp__anon_id'; const USER_COOKIE_KEY = 'sp__user_id'; const TRANSACTION_DEDUP_COOKIE_KEY = 'sp__transaction_ids'; const COOKIE_EXPIRY_DAYS = 365; /** * Generate UUID v4 compatible anonymous ID */ function generateAnonymousId() { // Generate segments for UUID format const seg1 = generateRandom(10000000, 99999999); const seg2 = generateRandom(1000, 9999); const seg3 = generateRandom(1000, 9999); const seg4 = generateRandom(1000, 9999); const seg5 = generateRandom(100000000000, 999999999999); return seg1 + '-' + seg2 + '-' + seg3 + '-' + seg4 + '-' + seg5; } /** * Get or create anonymous ID */ function getOrCreateAnonymousId() { // Try to get existing anonymous ID from cookies let anonymousId; const anonCookieValues = getCookieValues(ANON_COOKIE_KEY); if (anonCookieValues && anonCookieValues.length > 0) { anonymousId = anonCookieValues[0]; if (data.debugMode) { logToConsole('Spectacle: Found existing anonymous ID:', anonymousId); } } // Generate new ID if none exists if (!anonymousId) { anonymousId = generateAnonymousId(); if (data.debugMode) { logToConsole('Spectacle: Generated new anonymous ID:', anonymousId); } } // Set/refresh the cookie setCookie(ANON_COOKIE_KEY, anonymousId, { domain: getCookieDomain(data.cookieDomain), path: '/', 'max-age': COOKIE_EXPIRY_DAYS * 24 * 60 * 60, secure: true, sameSite: 'lax' }); return anonymousId; } /** * Get stored user ID if exists */ function getStoredUserId() { const userCookieValues = getCookieValues(USER_COOKIE_KEY); if (userCookieValues && userCookieValues.length > 0) { return userCookieValues[0]; } return null; } /** * Store user ID when identified */ function storeUserId(userId) { if (userId) { setCookie(USER_COOKIE_KEY, makeString(userId), { domain: getCookieDomain(data.cookieDomain), path: '/', 'max-age': COOKIE_EXPIRY_DAYS * 24 * 60 * 60, secure: true, sameSite: 'lax' }); } } /** * Get cookie domain to use. Prefix with '.' if missing. */ function getCookieDomain(cookieDomain) { if (cookieDomain) { if (cookieDomain[0] !== '.') { cookieDomain = '.' + cookieDomain; } return cookieDomain; } return 'auto'; } function getTransactionIds() { const transactionIds = getCookieValues(TRANSACTION_DEDUP_COOKIE_KEY); if (transactionIds && transactionIds.length > 0) { return transactionIds[0].split(','); } return transactionIds ? transactionIds : []; } function hasTransactionId(transactionId) { return getTransactionIds().indexOf(transactionId) > -1; } function storeTransactionId(transactionId) { const transactionIds = getTransactionIds(); if (transactionIds.indexOf(transactionId) === -1) { transactionIds.push(transactionId); } setCookie(TRANSACTION_DEDUP_COOKIE_KEY, transactionIds.join(','), { domain: getCookieDomain(data.cookieDomain), path: '/', 'max-age': COOKIE_EXPIRY_DAYS * 24 * 60 * 60, secure: true, sameSite: 'lax' }); } /** * Extract campaign/UTM parameters from URL */ function extractCampaign(url) { const campaign = {}; if (!url) return campaign; const parsedUrl = parseUrl(url); if (!parsedUrl || !parsedUrl.searchParams) return campaign; // Check each UTM parameter directly if (parsedUrl.searchParams.utm_source) { campaign.source = parsedUrl.searchParams.utm_source; } if (parsedUrl.searchParams.utm_medium) { campaign.medium = parsedUrl.searchParams.utm_medium; } if (parsedUrl.searchParams.utm_campaign) { campaign.name = parsedUrl.searchParams.utm_campaign; // Note: maps to 'name' } if (parsedUrl.searchParams.utm_term) { campaign.term = parsedUrl.searchParams.utm_term; } if (parsedUrl.searchParams.utm_content) { campaign.content = parsedUrl.searchParams.utm_content; } return campaign; } /** * Build page context matching your pixel implementation */ function buildPageContext() { const pageLocation = getEventData('page_location') || ''; const pageReferrer = getEventData('page_referrer') || ''; const pageTitle = getEventData('page_title') || ''; const parsedUrl = parseUrl(pageLocation); return { path: parsedUrl ? parsedUrl.pathname : '', referrer: pageReferrer, search: parsedUrl ? parsedUrl.search : '', title: pageTitle, url: pageLocation }; } /** * Build base payload matching your Segment-like API format */ function buildBasePayload(method) { const now = getTimestampMillis(); const anonymousId = getOrCreateAnonymousId(); const userId = getStoredUserId() || getEventData('user_id') || null; // Get page and campaign context const pageContext = buildPageContext(); const campaign = extractCampaign(pageContext.url); // Get user agent from headers const userAgent = getEventData('user_agent') || getRequestHeader('user-agent') || ''; // Get timezone from event data or default const timezone = getEventData('ga_session_data.timezone') || getEventData('timezone') || 'UTC'; // Get locale const locale = getEventData('language') || getEventData('user_properties.language') || null; return { type: method, context: { timezone: timezone, campaign: campaign, userAgent: userAgent, page: pageContext, locale: locale }, ip: getRemoteAddress(), lens: data.lensEnabled, userId: userId, anonymousId: anonymousId, writeKey: data.workspaceId, }; } /** * Helper to iterate over object properties (replaces Object.keys) */ function iterateObject(obj, callback) { if (!obj || getType(obj) !== 'object') return; for (let key in obj) { if (obj.hasOwnProperty(key)) { callback(key, obj[key]); } } } /** * Handle PAGE method */ function handlePage() { const payload = buildBasePayload('page'); // Get page data from event const pageLocation = getEventData('page_location') || ''; const pageTitle = getEventData('page_title') || ''; const parsedUrl = parseUrl(pageLocation); // Get screen dimensions if available const screenResolution = getEventData('screen_resolution') || ''; let width = null; let height = null; if (screenResolution && screenResolution.indexOf('x') > -1) { const dimensions = screenResolution.split('x'); width = makeInteger(dimensions[0]); height = makeInteger(dimensions[1]); } payload.properties = { "title": pageTitle, "url": pageLocation, "path": parsedUrl ? parsedUrl.pathname : '', "hash": parsedUrl ? parsedUrl.hash || '' : '', "search": parsedUrl ? parsedUrl.search : '', "width": width, "height": height }; return sendToSpectacle('/p', payload); } /** * Handle IDENTIFY method */ function handleIdentify() { const payload = buildBasePayload('identify'); const traits = {}; // Get user ID and store it const userId = data.userId || getEventData('user_id') || getEventData('user_data.email_address'); if (userId) { payload.userId = makeString(userId); storeUserId(payload.userId); } // Extract email from various sources const email = data.email || getEventData('user_data.email_address') || getEventData('user_properties.email'); if (email) traits.email = email; // Extract names const firstName = data.firstName || getEventData('user_data.first_name') || getEventData('user_properties.first_name'); if (firstName) traits.firstName = firstName; const lastName = data.lastName || getEventData('user_data.last_name') || getEventData('user_properties.last_name'); if (lastName) traits.lastName = lastName; // Extract phone const phone = getEventData('user_data.phone_number') || getEventData('user_properties.phone'); if (phone) traits.phone = phone; // Add custom traits from template configuration if (data.userTraits && getType(data.userTraits) === 'array') { for (let i = 0; i < data.userTraits.length; i++) { const trait = data.userTraits[i]; if (trait.key && trait.value) { traits[trait.key] = trait.value; } } } payload.traits = traits; return sendToSpectacle('/i', payload); } /** * Handle TRACK method */ function handleTrack() { const payload = buildBasePayload('track'); // Get event name const eventName = data.eventName || getEventData('event_name'); if (!eventName) { logToConsole('Spectacle: No event name provided for track call'); data.gtmOnFailure(); return; } payload.event = eventName; // Build properties const properties = {}; // Add revenue if present if (!data.useGA4EcomData) { if (data.revenue) { properties.revenue = Math.round(makeNumber(data.revenue)).toString(); } if (data.revenueFormat === 'majorUnits' && data.revenueMajorUnits) { properties.revenue = Math.round(makeNumber(data.revenueMajorUnits) * 100).toString(); } if (data.currency) { properties.currency = data.currency; } if (data.transactionId) { properties.transactionId = data.transactionId; } } if (data.useGA4EcomData) { const transactionId = getEventData('transaction_id'); if (transactionId) { // Don't send duplicate transactions if (hasTransactionId(transactionId)) { return; } storeTransactionId(transactionId); properties.transactionId = transactionId; } const amount = makeNumber(getEventData('value')); if (amount >= 0) { if (eventName === "purchase") { properties.revenue = Math.round(amount * 100).toString(); } if (eventName === "refund") { properties.revenue = "-" + Math.round(amount * 100).toString(); } } const currency = getEventData('value'); if (currency && currency.length === 3) { properties.currency = currency; } } // Add custom properties from template const eventProperties = data.eventProperties && getType(data.eventProperties) === 'array' ? data.eventProperties : []; for (let i = 0; i < eventProperties.length; i++) { const prop = eventProperties[i]; if (prop.key && prop.value) { properties[prop.key] = prop.value; } } payload.properties = properties; return sendToSpectacle('/t', payload); } /** * Handle GROUP method */ function handleGroup() { const payload = buildBasePayload('group'); const groupId = data.groupId || getEventData('group_id'); if (!groupId) { logToConsole('Spectacle: No group ID provided for group call'); data.gtmOnFailure(); return; } payload.groupId = makeString(groupId); // Build traits const traits = {}; if (data.groupTraits && getType(data.groupTraits) === 'array') { for (let i = 0; i < data.groupTraits.length; i++) { const trait = data.groupTraits[i]; if (trait.key && trait.value) { traits[trait.key] = trait.value; } } } payload.traits = traits; return sendToSpectacle('/g', payload); } /** * Send request to Spectacle */ function sendToSpectacle(endpoint, payload) { const url = data.baseUrl + endpoint; if (data.debugMode) { logToConsole('Spectacle: Sending to', url); logToConsole('Spectacle: Payload', payload); } // Note: Server-side doesn't need no-cors mode sendHttpRequest( url, { headers: { 'Content-Type': 'text/plain', 'User-Agent': payload.context.userAgent, }, method: 'POST', timeout: 5000 }, JSON.stringify(payload) ).then((result) => { if (data.debugMode) { logToConsole('Spectacle: Response', result); } if (result.statusCode >= 200 && result.statusCode < 300) { data.gtmOnSuccess(); } else { logToConsole('Spectacle: Error response', result.statusCode, result.body); data.gtmOnFailure(); } }).catch((error) => { logToConsole('Spectacle: Request failed', error); data.gtmOnFailure(); }); } // Main execution const methodType = data.methodType; // Execute the appropriate method switch (methodType) { case 'page': handlePage(); break; case 'identify': handleIdentify(); break; case 'track': handleTrack(); break; case 'group': handleGroup(); break; default: logToConsole('Spectacle: Unknown method type', methodType); data.gtmOnFailure(); break; } ___SERVER_PERMISSIONS___ [ { "instance": { "key": { "publicId": "send_http", "versionId": "1" }, "param": [ { "key": "allowedUrls", "value": { "type": 1, "string": "specific" } }, { "key": "urls", "value": { "type": 2, "listItem": [ { "type": 1, "string": "https://t.spectaclehq.com/*" } ] } } ] }, "isRequired": true }, { "instance": { "key": { "publicId": "read_event_data", "versionId": "1" }, "param": [ { "key": "eventDataAccess", "value": { "type": 1, "string": "any" } } ] }, "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": "sp__anon_id" }, { "type": 1, "string": "sp__user_id" }, { "type": 1, "string": "sp__transaction_ids" } ] } } ] }, "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": "sp__anon_id" }, { "type": 1, "string": "*" }, { "type": 1, "string": "/" }, { "type": 1, "string": "any" }, { "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": "sp__user_id" }, { "type": 1, "string": "*" }, { "type": 1, "string": "/" }, { "type": 1, "string": "any" }, { "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": "sp__transaction_ids" }, { "type": 1, "string": "*" }, { "type": 1, "string": "/" }, { "type": 1, "string": "any" }, { "type": 1, "string": "any" } ] }, ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "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" } } ] }, "isRequired": true }, { "instance": { "key": { "publicId": "logging", "versionId": "1" }, "param": [ { "key": "environments", "value": { "type": 1, "string": "all" } } ] }, "isRequired": true } ] ___TESTS___ scenarios: [] ___NOTES___ Created on 2025-08-11 Server-side template for Spectacle tracking that maintains anonymous ID persistence and supports page, identify, track, and group methods matching the client-side pixel implementation.