Emulates old 1st party cookies set for utm_
(source
, medium
, campaign
, term
, content
, id
) + the 3 new source_platform
, marketing_tactic
, creative_format
parameters in Classic Google Analytics (ga.js). Easily fill hidden form fields with the stored values of your first and last touch data (great for closed-loop/ROAS reporting needs). This script is meant to be a standardized means of implementing UTM cookies for campaign attribution.
1st party cookies attr_first
and attr_last
are stored at root domain .domain.com
.
attr_last
- Has an expiration date of 30-minutes. The 30-minutes is reset on each subsequent page load. This is meant to emulate the same session windows as Google Analytics.
attr_first
- Has an expiration date of 2-years. This cookie will never be set more than once (unless the cookie is of course deleted).
Each of these cookies contains an encoded JSON object of key:value pairs for the following standard UTM parameters:
- source
- medium
- campaign
- term
- content
- id
- source_platform
- marketing_tactic
- creative_format
and the following synthetic UTMs that are commonly used in Google Ads (and other ad platforms):
- adextension
- adgroup
- adgroupid
- adplacement
- adposition
- campaignid
- geo
- keymatch
- device
- matchtype
- network
The data points above are pulled from standard UTM parameters or are assigned implicit values via referral parsing.
The easiest way to use Attributor is by loading the script via jsdelivr. This can be done via a Custom HTML tag inside of Google Tag Manager.
<script src="https://cdn.jsdelivr.net/gh/derekcavaliero/attributor@latest/dist/attributor.min.js"></script>
<script>
(function(){
window.__utmz = new Attributor();
})();
</script>
The Attributor
constructor accepts a single configuration object - the library uses (some) sensible defaults, so you only need to override what configuration options you need to change.
var __utmz = new Attributor({
cookieDomain: 'domain.com'
});
Attributor uses sensible defaults for mapping stored campaign data to html form inputs. Fields are mapped per cookie using HTML <input>
name
attributes. The script can easily be modified to use classes for field mapping. The default field mapping is defined as a simple object literal. The structure of the fieldMap obecjt (for both first/last) is {fromStorage}: {toField}
{
first: {
// Standard UTMs supported by Google Analytics
source: 'utm_source_1st',
medium: 'utm_medium_1st',
campaign: 'utm_campaign_1st',
term: 'utm_term_1st',
content: 'utm_content_1st',
source_platform: 'utm_source_platform_1st',
marketing_tactic: 'utm_marketing_tactic_1st',
creative_format: 'utm_creative_format_1st',
id: 'utm_id_1st',
// Synthetic UTMs for "Value Track" parameters
adextension: 'utm_adextension_1st',
adgroup: 'utm_adgroup_1st',
adgroupid: 'utm_adgroupid_1st',
adplacement: 'utm_adplacement_1st',
adposition: 'utm_adposition_1st',
campaignid: 'utm_campaignid_1st',
geo: 'utm_geo_1st',
keymatch: 'utm_keymatch_1st',
device: 'utm_device_1st',
matchtype: 'utm_matchtype_1st',
network: 'utm_network_1st',
},
last: {
// Standard UTMs supported by Google Analytics
source: 'utm_source',
medium: 'utm_medium',
campaign: 'utm_campaign',
id: 'utm_id',
term: 'utm_term',
content: 'utm_content',
source_platform: 'utm_source_platform',
marketing_tactic: 'utm_marketing_tactic',
creative_format: 'utm_creative_format',
// Synthetic UTMs for "Value Track" parameters
adextension: 'utm_adextension',
adgroup: 'utm_adgroup',
adgroupid: 'utm_adgroupid',
adplacement: 'utm_adplacement',
adposition: 'utm_adposition',
campaignid: 'utm_campaignid',
geo: 'utm_geo',
keymatch: 'utm_keymatch',
device: 'utm_device',
matchtype: 'utm_matchtype',
network: 'utm_network',
}
cookies: null,
globals: null
}
This means that if you were to use the following hidden field markup (for each datapoint you want to collect) - you wouldn't need to configure a custom field mapping object:
<input type="hidden" name="{toField}">
If you cannot (or wish not to) use the default field mapping. You can easily create your own by defining you own fieldMap
in the configuration object passed to the Attributor
constructor. Custom field mappings are merged into the default fieldMap
object.
var __attribution = new Attributor({
cookieDomain: 'domain.com',
fieldMap: {
first: {
source: 'source_1st',
medium: 'medium_1st',
campaign: 'campaign_1st',
term: 'term_1st',
content: 'content_1st',
},
last: {
source: 'source_last',
medium: 'medium_last',
campaign: 'campaign_last',
term: 'term_last',
content: 'content_last',
}
}
});
If you only need/wish to change a few of the default field mappings - you can set those explicitly without affecting the other defaults:
var __attribution = new Attributor({
cookieDomain: 'mydomain.com',
fieldMap: {
cookies: {
_ga: 'ga_client_id'
}
}
});
Grabbing values from cookies set by other ad/martech is a pretty common need (especially for paid media). Attributor's cookies
property in the config fieldMap
can be customized to grab any 1st party cookie value and have it mapped to a form input similar to the first/last touch campaign data mentioned above.
For example, you could grab the _gcl_aw
(Google Ads Click ID), and _ga
(Google Analytics Client ID) cookie values + the current user agent via navigator.userAgent
as so:
fieldMap: {
// ...
cookies: {
_ga: 'ga_client_id',
_gcl_aw: 'gclid'
},
globals: {
'navigator.userAgent': 'user_agent'
}
}
The above example would map the data to the 3 form fields with the approprate names:
<input type="hidden" name="ga_client_id">
<input type="hidden" name="gclid">
<input type="hidden" name="user_agent">
The library has some predefined filters to handle filtering the cookie and global values. Note - these filters only work on the cookies
and globals
. You can define filters using the filters
property in the config object. For example, the following filter will remove the first two delimited values from the _ga
cookie. The filter property name must match the name of the cookie or global window path reference.
filters: {
_ga: function(val) {
// e.g: GA1.2.1234567890.0987654321
// Should return 1234567890.0987654321
return val.split('.').slice(2).join('.');
}
}
For convenience, each Attributor instance will have access to a grab()
method that will return an object literal of all the data in the fieldMap object. This can be useful if you would like to store the data as a JSON string in a single record rather than explicit form fields. The grab()
method accepts a single argument - sessionMode
- which can be either first
, last
, or all
(default). If no argument is passed, the method will return all stored session data.
The script will automatically run it's Attributor.fillFormFields()
method when initialized.
You can also manually call the method as such:
var __attribution = new Attributor(config);
__attribution.fillFormFields();
This can be helpful for a few scenarios:
- Forms loaded via JS or 3rd party systems
- When a form is loaded into the DOM after initial pageload (e.g: a modal/popup)
window.addEventListener('message', function(event) {
if (event.data.type === 'hsFormCallback' && event.data.eventName === 'onFormReady') {
__attribution.fillFormFields();
}
});
if (typeof MktoForms2 !== 'undefined') {
MktoForms2.whenReady( function( form ) {
__attribution.fillFormFields();
});
}
By default, Attributor will use <input>
name
attributes for updating/prefilling. This can be customized when initializing attributor using the fieldTargetMethod
property.
The fieldTargetMethod
property can be set to one or more of the following values as an array (or string) the configuration object.
name
(default) - usesdocument.getElementsByName('{field_map_key}')
class
- usesdocument.querySelectorAll('input.{field_map_key}')
parentClass
- usesdocument.querySelectorAll('.{field_map_key} input')
dataAttribute
- usesdocument.querySelectorAll(
input[data-${config.fieldDataAttribute}="{$fieldMapValue}"])
Note - fieldDataAttribute
is a property in the configuration object that defaults to data-attributor-field
. This can be changed to any string value.
For example:
var __attribution = new Attributor({
cookieDomain: 'mydomain.com',
fieldTargetMethod: ['name', 'class', 'dataAttribute'],
fieldMap: {
last: {
source: 'custom_source_last'
}
}
});
<input type="hidden" name="input_9a8dfs" class="custom_source_last">
<input type="hidden" name="custom_source_last" class="input_9a8dfs">
<input type="hidden" data-attributor-field="custom_source_last">