2016-07-20 12:00:34 -05:00
|
|
|
import functools
|
|
|
|
import json
|
|
|
|
import logging
|
|
|
|
|
|
|
|
from requests import auth
|
|
|
|
from requests_futures import sessions
|
|
|
|
|
2016-12-21 11:55:43 -08:00
|
|
|
from lbrynet import conf
|
2016-07-20 12:00:34 -05:00
|
|
|
from lbrynet.analytics import utils
|
|
|
|
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
def log_response(fn):
|
|
|
|
def _log(future):
|
|
|
|
if future.cancelled():
|
|
|
|
log.warning('Request was unexpectedly cancelled')
|
2016-11-16 19:10:49 -05:00
|
|
|
elif future.exception():
|
2016-11-22 15:18:01 -05:00
|
|
|
exc, traceback = future.exception_info()
|
|
|
|
log.warning('Failed to send an analytics event', exc_info=(type(exc), exc, traceback))
|
2016-07-20 12:00:34 -05:00
|
|
|
else:
|
|
|
|
response = future.result()
|
|
|
|
log.debug('Response (%s): %s', response.status_code, response.content)
|
|
|
|
|
|
|
|
@functools.wraps(fn)
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
future = fn(*args, **kwargs)
|
|
|
|
future.add_done_callback(_log)
|
|
|
|
return future
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
|
2016-10-01 10:47:37 -05:00
|
|
|
class Api(object):
|
2016-07-20 12:00:34 -05:00
|
|
|
def __init__(self, session, url, write_key):
|
|
|
|
self.session = session
|
|
|
|
self.url = url
|
|
|
|
self.write_key = write_key
|
|
|
|
|
2016-12-01 12:13:12 -06:00
|
|
|
def post(self, endpoint, data):
|
|
|
|
# there is an issue with a timing condition with keep-alive
|
|
|
|
# that is best explained here: https://github.com/mikem23/keepalive-race
|
|
|
|
#
|
|
|
|
# If you make a request, wait just the right amount of time,
|
|
|
|
# then make another request, the requests module may opt to
|
|
|
|
# reuse the connection, but by the time the server gets it the
|
|
|
|
# timeout will have expired.
|
|
|
|
#
|
|
|
|
# by forcing the connection to close, we will disable the keep-alive.
|
|
|
|
assert endpoint[0] == '/'
|
|
|
|
headers = {"Connection": "close"}
|
|
|
|
return self.session.post(
|
|
|
|
self.url + endpoint, json=data, auth=self.auth, headers=headers)
|
|
|
|
|
2016-07-20 12:00:34 -05:00
|
|
|
@property
|
|
|
|
def auth(self):
|
|
|
|
return auth.HTTPBasicAuth(self.write_key, '')
|
|
|
|
|
|
|
|
@log_response
|
|
|
|
def batch(self, events):
|
|
|
|
"""Send multiple events in one request.
|
|
|
|
|
|
|
|
Each event needs to have its type specified.
|
|
|
|
"""
|
|
|
|
data = json.dumps({
|
|
|
|
'batch': events,
|
|
|
|
'sentAt': utils.now(),
|
|
|
|
})
|
|
|
|
log.debug('sending %s events', len(events))
|
|
|
|
log.debug('Data: %s', data)
|
2016-12-01 12:13:12 -06:00
|
|
|
return self.post('/batch', data)
|
2016-07-20 12:00:34 -05:00
|
|
|
|
|
|
|
@log_response
|
|
|
|
def track(self, event):
|
|
|
|
"""Send a single tracking event"""
|
|
|
|
log.debug('Sending track event: %s', event)
|
2016-12-01 12:13:12 -06:00
|
|
|
return self.post('/track', event)
|
2016-07-20 12:00:34 -05:00
|
|
|
|
|
|
|
@classmethod
|
2016-11-10 15:49:51 -05:00
|
|
|
def new_instance(cls, session=None):
|
2016-09-29 14:51:48 -07:00
|
|
|
"""Initialize an instance using values from the configuration"""
|
2016-07-20 12:00:34 -05:00
|
|
|
if not session:
|
|
|
|
session = sessions.FuturesSession()
|
|
|
|
return cls(
|
|
|
|
session,
|
2016-12-21 11:55:43 -08:00
|
|
|
conf.settings.ANALYTICS_ENDPOINT,
|
|
|
|
utils.deobfuscate(conf.settings.ANALYTICS_TOKEN)
|
2016-07-20 12:00:34 -05:00
|
|
|
)
|