Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Initial commit for Hexoskin #211

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions tapiriik/services/Hexoskin/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
from .hexoskin import *
256 changes: 256 additions & 0 deletions tapiriik/services/Hexoskin/hexoskin.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,256 @@
from tapiriik.settings import WEB_ROOT, HEXOSKIN_CLIENT_SECRET, HEXOSKIN_CLIENT_ID
from tapiriik.services.service_base import ServiceAuthenticationType, ServiceBase
from tapiriik.services.service_record import ServiceRecord
from tapiriik.services.interchange import UploadedActivity, ActivityType, ActivityStatistic, ActivityStatisticUnit
from tapiriik.services.api import APIException, UserException, UserExceptionType, APIExcludeActivity
from tapiriik.services.tcx import TCXIO

from django.core.urlresolvers import reverse
from datetime import datetime, timedelta
from urllib.parse import urlencode
import requests
import logging
import pytz
import time

logger = logging.getLogger(__name__)
serverRoot = "https://api.hexoskin.com/"
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Pull this into a constant in the settings file, like the API keys?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The serverRoot shouldn't change, I'd rather hardcode it in the url strings in the code.



class HexoskinService(ServiceBase):
"""Define the base service object"""
ID = "hexoskin"
DisplayName = "Hexoskin"
DisplayAbbreviation = "Hx"
AuthenticationType = ServiceAuthenticationType.OAuth
UserProfileURL = serverRoot + "api/account/"
UserActivityURL = serverRoot + "api/range/"
AuthenticationNoFrame = True # They don't prevent the iframe, it just looks really ugly.
LastUpload = None

SupportsHR = SupportsCadence = SupportsTemp = True

SupportsActivityDeletion = False

# For mapping common->Hexoskin; no ambiguity in Hexoskin activity type
_activityTypeMappings = {
ActivityType.Cycling: "/api/activitytype/1/",
ActivityType.MountainBiking: "/api/activitytype/1/",
ActivityType.Hiking: "/api/activitytype/5/",
ActivityType.Running: "/api/activitytype/6/",
ActivityType.Walking: "/api/activitytype/15/",
ActivityType.Snowboarding: "/api/activitytype/13/",
ActivityType.Skating: "/api/activitytype/11/",
ActivityType.CrossCountrySkiing: "/api/activitytype/3/",
ActivityType.DownhillSkiing: "/api/activitytype/4/",
ActivityType.Swimming: "/api/activitytype/14/",
ActivityType.Gym: "/api/activitytype/28/",
ActivityType.Rowing: "/api/activitytype/9/",
ActivityType.Elliptical: "/api/activitytype/997/",
ActivityType.Other:"/api/activitytype/997/"
}


# For mapping Hexoskin->common
def _reverseActivityTypeMappings(self, key):
_reverseActivityTypeMappingsKeys = {
"/api/activitytype/1/": ActivityType.Cycling,
"/api/activitytype/3/": ActivityType.CrossCountrySkiing,
"/api/activitytype/4/": ActivityType.DownhillSkiing,
"/api/activitytype/5/": ActivityType.Hiking,
"/api/activitytype/6/": ActivityType.Running,
"/api/activitytype/7/": ActivityType.MountainBiking,
"/api/activitytype/9/": ActivityType.Rowing,
"/api/activitytype/10/": ActivityType.Running,
"/api/activitytype/11/": ActivityType.Skating,
"/api/activitytype/13/": ActivityType.Snowboarding,
"/api/activitytype/14/": ActivityType.Swimming,
"/api/activitytype/15/": ActivityType.Walking,
"/api/activitytype/24/": ActivityType.Running,
"/api/activitytype/28/": ActivityType.Gym,
}
if key in _reverseActivityTypeMappingsKeys.keys():
return _reverseActivityTypeMappingsKeys[key]
else:
return ActivityType.Other

SupportedActivities = list(_activityTypeMappings.keys())


def UserUploadedActivityURL(self, uploadId):
return serverRoot + "api/range/%d/" % uploadId


def WebInit(self):
"""
prepare the oauth process request. Done separately because it needs to be
initialized on page display
"""
from uuid import uuid4
params = {'scope':'readwrite',
'client_id':HEXOSKIN_CLIENT_ID,
'response_type':'code',
'state': str(uuid4()),
'redirect_uri':WEB_ROOT + reverse("oauth_return", kwargs={"service": "hexoskin"})}
self.UserAuthorizationURL = serverRoot + "api/connect/oauth2/auth/?" + urlencode(params)


def _apiHeaders(self, serviceRecord):
return {"Authorization": "Bearer " + serviceRecord.Authorization["OAuthToken"]}


def RetrieveAuthorizationToken(self, req, level):
"""In OAuth flow, retrieve the Authorization Token"""
code = req.GET.get("code")
params = {"grant_type": "authorization_code", "code": code,"client_id":HEXOSKIN_CLIENT_ID, "client_secret": HEXOSKIN_CLIENT_SECRET, "redirect_uri": WEB_ROOT + reverse("oauth_return", kwargs={"service": "hexoskin"})}
path = serverRoot + "api/connect/oauth2/token/"
response = requests.post(path, params=params, auth=(HEXOSKIN_CLIENT_ID,HEXOSKIN_CLIENT_SECRET))

if response.status_code != 200:
raise APIException("Invalid code")

data = response.json()
authorizationData = {"OAuthToken": data["access_token"]}
id_resp = requests.get(serverRoot + "api/account/", headers=self._apiHeaders(ServiceRecord({"Authorization": authorizationData})))
return (id_resp.json()['objects'][0]['id'], authorizationData)


def RevokeAuthorization(self, serviceRecord):
"""Delete authorization token"""
path = serverRoot + "api/oauth2token/232/"
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

232 what's this?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry, that slipped my mind when sending that. OAuth tokens are accessed through ids on Hexoskin, so revoke calls must be sent to the token id. Because the token is accessed through it's ID rather than using the public or private key as many other do, i hardcoded mine in the code. The /232/ string would probably be better as an entry in the local_settings though (something like "HEXOSKIN_CLIENT_KEY_ID=XXX")

headers = self._apiHeaders(serviceRecord)
result = requests.delete(path, headers=headers)
if result.status_code is not 204:
APIException("Revoking token was unsuccessful")


def _is_ride_valid(self, ride):
# Sync only top-level activities, and exclude rest, rest test and sleep
valid = (
ride['rank'] is 0
and ride['context']['activitytype'] is not None
and not any(y in ride['context']['activitytype'] for y in ['/8/', '/12/', '/106/'])
)
return valid

def DownloadActivityList(self, serviceRecord, exhaustive=False):
"""
Get list of user's activities in Hexoskin and return it to tapiriik database
"""
logger.debug('Hexoskin - starting to download activity list for user %s' % serviceRecord.ExternalID)
activities = []
exclusions = []
if exhaustive:
listEnd = (datetime.now() + timedelta(days=1.5) - datetime(1970,1,1)).total_seconds()*256
listStart = (datetime(day=21, month=8, year=1985) - datetime(1970,1,1)).total_seconds()*256 # The distant past
resp = requests.get(serverRoot + "api/range/?user=%s&limit=100&rank=0&start__range=%s,%s" % (serviceRecord.ExternalID, int(listStart), int(listEnd)), headers=self._apiHeaders(serviceRecord))
else:
listEnd = (datetime.now() + timedelta(days=1.5) - datetime(1970,1,1)).total_seconds()*256
listStart = (datetime.now() - timedelta(days=30) - datetime(1970,1,1)).total_seconds()*256
resp = requests.get(serverRoot + "api/range/?user=%s&limit=30&rank=0&start__range=%s,%s" % (serviceRecord.ExternalID, int(listStart), int(listEnd)), headers=self._apiHeaders(serviceRecord))
if resp.status_code == 401:
raise APIException("No authorization to retrieve activity list", block=True, user_exception=UserException(UserExceptionType.Authorization, intervention_required=True))
try:
reqdata = resp.json()['objects']
except ValueError:
logger.debug("Failed parsing hexoskin list response %s - %s" % (resp.status_code, resp.text))
raise APIException("Failed parsing hexoskin list response %s - %s" % (resp.status_code, resp.text))
for ride in reqdata:
try:
if not (ride['status'] == 'complete'):
pass # exclude that range for now, without excluding it in the future
elif self._is_ride_valid(ride):
activity = UploadedActivity()
activity.StartTime = pytz.utc.localize(datetime.fromtimestamp(ride['start']/256.0))
activity.EndTime = pytz.utc.localize(datetime.fromtimestamp(ride['end']/256.0))
activity.ServiceData = {"ActivityID": ride["id"]}
activity.Type = self._reverseActivityTypeMappings(ride['context']['activitytype'])
for metric in ride['metrics']:
# TODO check for IDs instead of titles
if metric['resource_uri'] == '/api/metric/17/': # Cadence
activity.Stats.RunCadence.update(ActivityStatistic(ActivityStatisticUnit.StepsPerMinute, value=metric['value']))
if metric['resource_uri'] == '/api/metric/44/': # Heart rate Average
activity.Stats.HR.update(ActivityStatistic(ActivityStatisticUnit.BeatsPerMinute, avg=metric["value"]))
if metric['resource_uri'] == '/api/metric/46/': # Heart rate Max
activity.Stats.HR.update(ActivityStatistic(ActivityStatisticUnit.BeatsPerMinute, max=metric["value"]))
if metric['resource_uri'] == '/api/metric/71/': # Step count
activity.Stats.Strides.update(ActivityStatistic(ActivityStatisticUnit.Strides, value=metric["value"]))
if metric['resource_uri'] == '/api/metric/149/': # Energy kcal
activity.Stats.Energy.update(ActivityStatistic(ActivityStatisticUnit.Kilocalories, value=metric['value']))
if metric['resource_uri'] == '/api/metric/501/': # Speed Max
activity.Stats.Speed.update(ActivityStatistic(ActivityStatisticUnit.MetersPerSecond, max=metric["value"]))
if metric['resource_uri'] == '/api/metric/502/': # Speed Avg
activity.Stats.Speed.update(ActivityStatistic(ActivityStatisticUnit.MetersPerSecond, avg=metric["value"]))
if metric['resource_uri'] == '/api/metric/2038/': # Distance
activity.Stats.Distance.update(ActivityStatistic(ActivityStatisticUnit.Meters, value=metric['value']))

activity.Name = ride["name"]
activity.Stationary = False

ride_track = requests.get(serverRoot + "api/track/?range=%s" % ride['id'], headers=self._apiHeaders(serviceRecord))
time.sleep(0.2)
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

time.sleep?

activity.GPS = (True if ride_track.json()['objects'] else False)

activity.CalculateUID()
activities.append(activity)
else:
exclusions.append(APIExcludeActivity("Unsupported activity type %s" % ride['context']['activitytype'], activity_id=ride["id"], user_exception=UserException(UserExceptionType.Other)))
except TypeError as e:
logger.debug("Failed parsing ranges url, response: %s\n%s" % (resp.url, resp.content))
raise e
logger.debug('Hexoskin - %s activities found, %s excluded. Activities' % ([x.ServiceData['ActivityID'] for x in activities], [x.ExternalActivityID for x in exclusions]))
return activities, exclusions


def DownloadActivity(self, serviceRecord, activity):
"""Extract activity from Hexoskin"""
activityID = activity.ServiceData["ActivityID"]
logger.debug('Hexoskin - Extracting activity %s' % activityID)
headers = self._apiHeaders(serviceRecord)
headers.update({"Accept":"application/vnd.garmin.tcx+xml"})
range_tcx = requests.get(serverRoot + "api/range/%s/" % (str(activityID)), headers=headers)
TCXIO.Parse(range_tcx.content, activity)
activity.Notes = activity.Name
activity.Name = '%s - Hx' % (activity.Type)
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

- Hx?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Contraction for Hexoskin (to indicate where the data is from). If some other name feels more relevant, I'm open to discussions

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There's no need to annotate the activity name - or, at least, nothing else does it.

return activity


def UploadActivity(self, serviceRecord, activity):
"""Import data into Hexoskin using TCX format"""
tcx_data = TCXIO.Dump(activity)
headers = self._apiHeaders(serviceRecord)
headers.update({"Content-Type":"application/vnd.garmin.tcx+xml"})

range_tcx = requests.post(serverRoot + "api/import/", data=tcx_data.encode('utf-8'), headers=headers)
import_id = str(range_tcx.json()['resource_uri'])
# TODO In line below, fix the way the resource_uri is constructed when the /importfile/ fix is deployed
uploaded = False
process_start_time = datetime.now()
while uploaded is False:
time.sleep(1)
range_upload_status = requests.get(serverRoot + import_id, headers=headers).json()
if (datetime.now() - process_start_time).total_seconds() < 20:
if range_upload_status['results'] is not None and 'error' in range_upload_status['results']:
err = range_upload_status['results']['error']
logger.debug('Hexoskin - Import range failed, see error')
raise APIException('Error uploading to Hexoskin: %s' % err)
elif range_upload_status['progress'] == 1 and range_upload_status['results']:
for entry in range_upload_status['results']:
if 'resource_uri' in entry.keys() and "range" in entry['resource_uri'] and entry['rank'] == 0:
upload_id = entry['id']
logger.debug('Hexoskin - Imported range %s' % upload_id)
uploaded = True
break
else:
raise APIException('Timeout uploading activity for import id %s' % import_id) # when the '0' bug is fixed, change this
return upload_id


def DeleteCachedData(self, serviceRecord):
"""No cached data"""
pass


def DeleteActivity(self, serviceRecord, uploadId):
"""We would rather have users delete data from their dashboard instead of an automatic tool"""
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is for the automatic rollback tool (used for recovering from significant issues with duplicates). Here, having an automated system remove only the activities tapiriik uploaded is much more reliable than the user trying to figure out which was the original and which was a duplicate, especially over 1,000s of activities.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The way activities are created on the server makes this a bit non-trivial, but I'll see what I can do.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm trying to implement the rollback, but can't get the hang of how it's triggered or how the celery process is handled. The task entry is correctly created in mongoDB when I visit the /rollback page and accept the operation, but it stops there and nothing more seems to happen.
The python decorator @celery_app.task() in rollback_worker.py seems to indicate that celery is needed, but it is not installed on the default VM provided, so I'm not sure if you are handling the task in some custom manner

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh - yes, when I was first making the vagrant box I intentionally didn't include any celery stuff, as all of it (at the time) wasn't required for local development (e.g. handling deferred webhook calls). You can always set CELERY_ALWAYS_EAGER = True to disable the queuing entirely.

pass
2 changes: 2 additions & 0 deletions tapiriik/services/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@
RunKeeper = RunKeeperService()
from tapiriik.services.Strava import StravaService
Strava = StravaService()
from tapiriik.services.Hexoskin import HexoskinService
Hexoskin = HexoskinService()
from tapiriik.services.Endomondo import EndomondoService
Endomondo = EndomondoService()
from tapiriik.services.Dropbox import DropboxService
Expand Down
5 changes: 3 additions & 2 deletions tapiriik/services/service.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ def FromID(id):
raise ValueError

def List():
return [RunKeeper, Strava, GarminConnect, SportTracks, Dropbox, TrainingPeaks, RideWithGPS, Endomondo, Motivato, NikePlus, VeloHero, TrainerRoad, Smashrun] + PRIVATE_SERVICES
return [RunKeeper, Strava, GarminConnect, SportTracks, Dropbox, TrainingPeaks, RideWithGPS, Endomondo, Motivato, NikePlus, VeloHero, TrainerRoad, Smashrun, Hexoskin] + PRIVATE_SERVICES

def PreferredDownloadPriorityList():
# Ideally, we'd make an informed decision based on whatever features the activity had
Expand All @@ -44,7 +44,8 @@ def PreferredDownloadPriorityList():
Endomondo, # No laps, no cadence
RunKeeper, # No laps, no cadence, no power
Motivato,
NikePlus
NikePlus,
Hexoskin
] + PRIVATE_SERVICES

def WebInit():
Expand Down
Binary file added tapiriik/web/static/img/services/hexoskin.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added tapiriik/web/static/img/services/hexoskin_l.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
1 change: 1 addition & 0 deletions tapiriik/web/views/privacy.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ def privacy(request):
services["dropbox"].update({"email": NO, "password": NO, "tokens": YES, "metadata": YES, "data":CACHED})
services["runkeeper"].update({"email": NO, "password": NO, "tokens": YES, "metadata": YES, "data":NO})
services["rwgps"].update({"email": OPTIN, "password": OPTIN, "tokens": NO, "metadata": YES, "data":NO})
services["hexoskin"].update({"email": NO, "password": NO, "tokens": YES, "metadata": YES, "data":NO})
services["trainingpeaks"].update({"email": OPTIN, "password": OPTIN, "tokens": NO, "metadata": YES, "data":NO})
services["endomondo"].update({"email": NO, "password": NO, "tokens": YES, "metadata": YES, "data":NO})
services["motivato"].update({"email": OPTIN, "password": OPTIN, "tokens": NO, "metadata": YES, "data":NO})
Expand Down