mirror of
https://github.com/Tautulli/Tautulli.git
synced 2025-07-08 06:00:51 -07:00
Clean up tracker
This commit is contained in:
parent
90647628c9
commit
655a359ef4
1 changed files with 69 additions and 95 deletions
|
@ -20,14 +20,14 @@ import hashlib
|
||||||
import socket
|
import socket
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def generate_uuid(basedata=None):
|
def generate_uuid(basedata=None):
|
||||||
""" Provides a _random_ UUID with no input, or a UUID4-format MD5 checksum of any input data provided """
|
""" Provides a _random_ UUID with no input, or a UUID4-format MD5 checksum of any input data provided """
|
||||||
if basedata is None:
|
if basedata is None:
|
||||||
return str(uuid.uuid4())
|
return str(uuid.uuid4())
|
||||||
elif isinstance(basedata, basestring):
|
elif isinstance(basedata, basestring):
|
||||||
checksum = hashlib.md5(basedata).hexdigest()
|
checksum = hashlib.md5(basedata).hexdigest()
|
||||||
return '%8s-%4s-%4s-%4s-%12s' % (checksum[0:8], checksum[8:12], checksum[12:16], checksum[16:20], checksum[20:32])
|
return '%8s-%4s-%4s-%4s-%12s' % (
|
||||||
|
checksum[0:8], checksum[8:12], checksum[12:16], checksum[16:20], checksum[20:32])
|
||||||
|
|
||||||
|
|
||||||
class Time(datetime.datetime):
|
class Time(datetime.datetime):
|
||||||
|
@ -61,7 +61,6 @@ class Time(datetime.datetime):
|
||||||
return (now - base) * 1000
|
return (now - base) * 1000
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class HTTPRequest(object):
|
class HTTPRequest(object):
|
||||||
""" URL Construction and request handling abstraction.
|
""" URL Construction and request handling abstraction.
|
||||||
This is not intended to be used outside this module.
|
This is not intended to be used outside this module.
|
||||||
|
@ -72,7 +71,6 @@ class HTTPRequest(object):
|
||||||
|
|
||||||
endpoint = 'https://www.google-analytics.com/collect'
|
endpoint = 'https://www.google-analytics.com/collect'
|
||||||
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def debug():
|
def debug():
|
||||||
""" Activate debugging on urllib2 """
|
""" Activate debugging on urllib2 """
|
||||||
|
@ -84,7 +82,6 @@ class HTTPRequest(object):
|
||||||
def __init__(self, user_agent=None, *args, **opts):
|
def __init__(self, user_agent=None, *args, **opts):
|
||||||
self.user_agent = user_agent or 'Analytics Pros - Universal Analytics (Python)'
|
self.user_agent = user_agent or 'Analytics Pros - Universal Analytics (Python)'
|
||||||
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def fixUTF8(cls, data): # Ensure proper encoding for UA's servers...
|
def fixUTF8(cls, data): # Ensure proper encoding for UA's servers...
|
||||||
""" Convert all strings to UTF-8 """
|
""" Convert all strings to UTF-8 """
|
||||||
|
@ -93,8 +90,6 @@ class HTTPRequest(object):
|
||||||
data[key] = data[key].encode('utf-8')
|
data[key] = data[key].encode('utf-8')
|
||||||
return data
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# Apply stored properties to the given dataset & POST to the configured endpoint
|
# Apply stored properties to the given dataset & POST to the configured endpoint
|
||||||
def send(self, data):
|
def send(self, data):
|
||||||
request = Request(
|
request = Request(
|
||||||
|
@ -120,8 +115,6 @@ class HTTPRequest(object):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class HTTPPost(HTTPRequest):
|
class HTTPPost(HTTPRequest):
|
||||||
|
|
||||||
# Apply stored properties to the given dataset & POST to the configured endpoint
|
# Apply stored properties to the given dataset & POST to the configured endpoint
|
||||||
|
@ -136,17 +129,12 @@ class HTTPPost(HTTPRequest):
|
||||||
self.open(request)
|
self.open(request)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class Tracker(object):
|
class Tracker(object):
|
||||||
""" Primary tracking interface for Universal Analytics """
|
""" Primary tracking interface for Universal Analytics """
|
||||||
params = None
|
params = None
|
||||||
parameter_alias = {}
|
parameter_alias = {}
|
||||||
valid_hittypes = ('pageview', 'event', 'social', 'screenview', 'transaction', 'item', 'exception', 'timing')
|
valid_hittypes = ('pageview', 'event', 'social', 'screenview', 'transaction', 'item', 'exception', 'timing')
|
||||||
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def alias(cls, typemap, base, *names):
|
def alias(cls, typemap, base, *names):
|
||||||
""" Declare an alternate (humane) name for a measurement protocol parameter """
|
""" Declare an alternate (humane) name for a measurement protocol parameter """
|
||||||
|
@ -164,7 +152,6 @@ class Tracker(object):
|
||||||
else:
|
else:
|
||||||
raise KeyError, 'Parameter "{0}" is not recognized'.format(name)
|
raise KeyError, 'Parameter "{0}" is not recognized'.format(name)
|
||||||
|
|
||||||
|
|
||||||
def payload(self, data):
|
def payload(self, data):
|
||||||
for key, value in data.iteritems():
|
for key, value in data.iteritems():
|
||||||
try:
|
try:
|
||||||
|
@ -172,8 +159,6 @@ class Tracker(object):
|
||||||
except KeyError:
|
except KeyError:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
option_sequence = {
|
option_sequence = {
|
||||||
'pageview': [(basestring, 'dp')],
|
'pageview': [(basestring, 'dp')],
|
||||||
'event': [(basestring, 'ec'), (basestring, 'ea'), (basestring, 'el'), (int, 'ev')],
|
'event': [(basestring, 'ec'), (basestring, 'ea'), (basestring, 'el'), (int, 'ev')],
|
||||||
|
@ -192,9 +177,6 @@ class Tracker(object):
|
||||||
data[optname] = args[opt_position]
|
data[optname] = args[opt_position]
|
||||||
opt_position += 1
|
opt_position += 1
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def hittime(cls, timestamp=None, age=None, milliseconds=None):
|
def hittime(cls, timestamp=None, age=None, milliseconds=None):
|
||||||
""" Returns an integer represeting the milliseconds offset for a given hit (relative to now) """
|
""" Returns an integer represeting the milliseconds offset for a given hit (relative to now) """
|
||||||
|
@ -205,14 +187,12 @@ class Tracker(object):
|
||||||
if isinstance(age, (int, float)):
|
if isinstance(age, (int, float)):
|
||||||
return int(age * 1000) + (milliseconds or 0)
|
return int(age * 1000) + (milliseconds or 0)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def account(self):
|
def account(self):
|
||||||
return self.params.get('tid', None)
|
return self.params.get('tid', None)
|
||||||
|
|
||||||
|
def __init__(self, account, name=None, client_id=None, hash_client_id=False, user_id=None, user_agent=None,
|
||||||
def __init__(self, account, name = None, client_id = None, hash_client_id = False, user_id = None, user_agent = None, use_post = True):
|
use_post=True):
|
||||||
|
|
||||||
if use_post is False:
|
if use_post is False:
|
||||||
self.http = HTTPRequest(user_agent=user_agent)
|
self.http = HTTPRequest(user_agent=user_agent)
|
||||||
|
@ -231,7 +211,6 @@ class Tracker(object):
|
||||||
if user_id is not None:
|
if user_id is not None:
|
||||||
self.params['uid'] = user_id
|
self.params['uid'] = user_id
|
||||||
|
|
||||||
|
|
||||||
def set_timestamp(self, data):
|
def set_timestamp(self, data):
|
||||||
""" Interpret time-related options, apply queue-time parameter as needed """
|
""" Interpret time-related options, apply queue-time parameter as needed """
|
||||||
if 'hittime' in data: # an absolute timestamp
|
if 'hittime' in data: # an absolute timestamp
|
||||||
|
@ -239,7 +218,6 @@ class Tracker(object):
|
||||||
if 'hitage' in data: # a relative age (in seconds)
|
if 'hitage' in data: # a relative age (in seconds)
|
||||||
data['qt'] = self.hittime(age=data.pop('hitage', None))
|
data['qt'] = self.hittime(age=data.pop('hitage', None))
|
||||||
|
|
||||||
|
|
||||||
def send(self, hittype, *args, **data):
|
def send(self, hittype, *args, **data):
|
||||||
""" Transmit HTTP requests to Google Analytics using the measurement protocol """
|
""" Transmit HTTP requests to Google Analytics using the measurement protocol """
|
||||||
|
|
||||||
|
@ -258,7 +236,6 @@ class Tracker(object):
|
||||||
if k not in data:
|
if k not in data:
|
||||||
data[k] = v
|
data[k] = v
|
||||||
|
|
||||||
|
|
||||||
data = dict(self.payload(data))
|
data = dict(self.payload(data))
|
||||||
|
|
||||||
if self.hash_client_id:
|
if self.hash_client_id:
|
||||||
|
@ -267,9 +244,6 @@ class Tracker(object):
|
||||||
# Transmit the hit to Google...
|
# Transmit the hit to Google...
|
||||||
self.http.send(data)
|
self.http.send(data)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# Setting persistent attibutes of the session/hit/etc (inc. custom dimensions/metrics)
|
# Setting persistent attibutes of the session/hit/etc (inc. custom dimensions/metrics)
|
||||||
def set(self, name, value=None):
|
def set(self, name, value=None):
|
||||||
if isinstance(name, dict):
|
if isinstance(name, dict):
|
||||||
|
@ -286,8 +260,6 @@ class Tracker(object):
|
||||||
except KeyError:
|
except KeyError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def __getitem__(self, name):
|
def __getitem__(self, name):
|
||||||
param, value = self.coerceParameter(name, None)
|
param, value = self.coerceParameter(name, None)
|
||||||
return self.params.get(param, None)
|
return self.params.get(param, None)
|
||||||
|
@ -301,6 +273,7 @@ class Tracker(object):
|
||||||
if param in self.params:
|
if param in self.params:
|
||||||
del self.params[param]
|
del self.params[param]
|
||||||
|
|
||||||
|
|
||||||
def safe_unicode(obj):
|
def safe_unicode(obj):
|
||||||
""" Safe convertion to the Unicode string version of the object """
|
""" Safe convertion to the Unicode string version of the object """
|
||||||
try:
|
try:
|
||||||
|
@ -331,7 +304,6 @@ Tracker.alias(int, 'qt', 'queueTime', 'queue-time')
|
||||||
Tracker.alias(safe_unicode, 't', 'hitType', 'hittype')
|
Tracker.alias(safe_unicode, 't', 'hitType', 'hittype')
|
||||||
Tracker.alias(int, 'aip', 'anonymizeIp', 'anonIp', 'anonymize-ip')
|
Tracker.alias(int, 'aip', 'anonymizeIp', 'anonIp', 'anonymize-ip')
|
||||||
|
|
||||||
|
|
||||||
# Campaign attribution
|
# Campaign attribution
|
||||||
Tracker.alias(safe_unicode, 'cn', 'campaign', 'campaignName', 'campaign-name')
|
Tracker.alias(safe_unicode, 'cn', 'campaign', 'campaignName', 'campaign-name')
|
||||||
Tracker.alias(safe_unicode, 'cs', 'source', 'campaignSource', 'campaign-source')
|
Tracker.alias(safe_unicode, 'cs', 'source', 'campaignSource', 'campaign-source')
|
||||||
|
@ -360,7 +332,8 @@ Tracker.alias(safe_unicode, 'ti', 'transaction', 'transactionId', 'transaction-i
|
||||||
Tracker.alias(float, 'tr', 'revenue', 'transactionRevenue', 'transaction-revenue')
|
Tracker.alias(float, 'tr', 'revenue', 'transactionRevenue', 'transaction-revenue')
|
||||||
Tracker.alias(float, 'ts', 'shipping', 'transactionShipping', 'transaction-shipping')
|
Tracker.alias(float, 'ts', 'shipping', 'transactionShipping', 'transaction-shipping')
|
||||||
Tracker.alias(float, 'tt', 'tax', 'transactionTax', 'transaction-tax')
|
Tracker.alias(float, 'tt', 'tax', 'transactionTax', 'transaction-tax')
|
||||||
Tracker.alias(safe_unicode, 'cu', 'currency', 'transactionCurrency', 'transaction-currency') # Currency code, e.g. USD, EUR
|
Tracker.alias(safe_unicode, 'cu', 'currency', 'transactionCurrency',
|
||||||
|
'transaction-currency') # Currency code, e.g. USD, EUR
|
||||||
Tracker.alias(safe_unicode, 'in', 'item-name', 'itemName')
|
Tracker.alias(safe_unicode, 'in', 'item-name', 'itemName')
|
||||||
Tracker.alias(float, 'ip', 'item-price', 'itemPrice')
|
Tracker.alias(float, 'ip', 'item-price', 'itemPrice')
|
||||||
Tracker.alias(float, 'iq', 'item-quantity', 'itemQuantity')
|
Tracker.alias(float, 'iq', 'item-quantity', 'itemQuantity')
|
||||||
|
@ -374,7 +347,6 @@ Tracker.alias(safe_unicode, 'el', 'event-label', 'eventLabel', 'label')
|
||||||
Tracker.alias(int, 'ev', 'event-value', 'eventValue', 'value')
|
Tracker.alias(int, 'ev', 'event-value', 'eventValue', 'value')
|
||||||
Tracker.alias(int, 'ni', 'noninteractive', 'nonInteractive', 'noninteraction', 'nonInteraction')
|
Tracker.alias(int, 'ni', 'noninteractive', 'nonInteractive', 'noninteraction', 'nonInteraction')
|
||||||
|
|
||||||
|
|
||||||
# Social
|
# Social
|
||||||
Tracker.alias(safe_unicode, 'sa', 'social-action', 'socialAction')
|
Tracker.alias(safe_unicode, 'sa', 'social-action', 'socialAction')
|
||||||
Tracker.alias(safe_unicode, 'sn', 'social-network', 'socialNetwork')
|
Tracker.alias(safe_unicode, 'sn', 'social-network', 'socialNetwork')
|
||||||
|
@ -434,8 +406,10 @@ for product_index in range(1, MAX_EC_PRODUCTS):
|
||||||
Tracker.alias(int, 'il{0}pi{1}pr'.format(list_index, product_index)) # Product impression price
|
Tracker.alias(int, 'il{0}pi{1}pr'.format(list_index, product_index)) # Product impression price
|
||||||
|
|
||||||
for custom_index in range(MAX_CUSTOM_DEFINITIONS):
|
for custom_index in range(MAX_CUSTOM_DEFINITIONS):
|
||||||
Tracker.alias(str, 'il{0}pi{1}cd{2}'.format(list_index, product_index, custom_index)) # Product impression custom dimension
|
Tracker.alias(str, 'il{0}pi{1}cd{2}'.format(list_index, product_index,
|
||||||
Tracker.alias(int, 'il{0}pi{1}cm{2}'.format(list_index, product_index, custom_index)) # Product impression custom metric
|
custom_index)) # Product impression custom dimension
|
||||||
|
Tracker.alias(int, 'il{0}pi{1}cm{2}'.format(list_index, product_index,
|
||||||
|
custom_index)) # Product impression custom metric
|
||||||
|
|
||||||
for list_index in range(1, MAX_EC_LISTS):
|
for list_index in range(1, MAX_EC_LISTS):
|
||||||
Tracker.alias(unicode, 'il{0}nm'.format(list_index)) # Product impression list name
|
Tracker.alias(unicode, 'il{0}nm'.format(list_index)) # Product impression list name
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue