Files
GoldenCheetah/deprecated/TodaysPlanBodyMeasures.cpp
Alejandro Martinez 642eae96cb Deprecate Today's Plan integration
Fixes #4450
2024-03-02 19:02:30 -03:00

179 lines
6.6 KiB
C++

/*
* Copyright (c) 2017 Joern Rischmueller (joern.rm@gmail.com)
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the Free
* Software Foundation; either version 2 of the License, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc., 51
* Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "TodaysPlanBodyMeasures.h"
#include "Settings.h"
#include "Athlete.h"
#include <QJsonDocument>
#include <QJsonArray>
#include <QJsonObject>
#include <QJsonValue>
#include <QNetworkReply>
TodaysPlanBodyMeasures::TodaysPlanBodyMeasures(Context *context) : context(context) {
nam = new QNetworkAccessManager(this);
connect(nam, SIGNAL(sslErrors(QNetworkReply*, const QList<QSslError> & )), this, SLOT(onSslErrors(QNetworkReply*, const QList<QSslError> & )));
}
TodaysPlanBodyMeasures::~TodaysPlanBodyMeasures() {
delete nam;
}
void
TodaysPlanBodyMeasures::onSslErrors(QNetworkReply *reply, const QList<QSslError>&)
{
reply->ignoreSslErrors();
}
bool
TodaysPlanBodyMeasures::getBodyMeasures(QString &error, QDateTime from, QDateTime to, QList<Measure> &data) {
// do we have a token
QString token = appsettings->cvalue(context->athlete->cyclist, GC_TODAYSPLAN_TOKEN, "").toString();
if (token == "") {
error = tr("You must authorise with Today's Plan first");
return false;
}
// Do Paginated Access to the Activities List
const int pageSize = 100;
int offset = 0;
int resultCount = INT_MAX;
while (offset < resultCount) {
QString url;
QString searchCommand;
if (offset == 0) {
// fist call
searchCommand = "search";
} else {
// subsequent pages
searchCommand = "page";
}
url = QString("%1/rest/users/day/%2/%3/%4")
.arg(appsettings->cvalue(context->athlete->cyclist, GC_TODAYSPLAN_URL, "https://whats.todaysplan.com.au").toString())
.arg(searchCommand)
.arg(QString::number(offset))
.arg(QString::number(pageSize));;
// request using the bearer token
QNetworkRequest request(url);
QNetworkReply *reply;
request.setRawHeader("Authorization", (QString("Bearer %1").arg(token)).toLatin1());
if (offset == 0) {
// Prepare the Search Payload for First Call to Search
QString userId = appsettings->cvalue(context->athlete->cyclist, GC_TODAYSPLAN_ATHLETE_ID, "").toString();
// application/json
QString jsonString;
jsonString += "{\"criteria\": { ";
if (userId.length()>0)
jsonString += " \"userIds\": [ "+ QString("%1").arg(userId) +" ], ";
jsonString += "\"ranges\": [ {";
jsonString += "\"floorTs\": \""+ QString("%1").arg(from.toMSecsSinceEpoch()) +"\", ";
jsonString += "\"ceilTs\": \"" + QString("%1").arg(to.addDays(1).addSecs(-1).toMSecsSinceEpoch()) +"\"";
jsonString += "} ] }, ";
jsonString += "\"fields\": [\"att.ts\",\"att.weight\", \"att.fat\",\"att.muscleMass\",\"att.boneMass\",\"att.fatMass\", \"att.height\" , \"att.source\"] ";
jsonString += "}";
QByteArray jsonStringAsUTF8 = jsonString.toUtf8();
QByteArray jsonStringDataSize = QByteArray::number(jsonStringAsUTF8.size());
request.setHeader(QNetworkRequest::ContentTypeHeader,"application/json");
request.setRawHeader("Content-Length", jsonStringDataSize);
reply = nam->post(request, jsonStringAsUTF8);
} else {
// get further pages of the Search
reply = nam->get(request);
}
// blocking request
QEventLoop loop;
connect(reply, SIGNAL(finished()), &loop, SLOT(quit()));
loop.exec();
// did we get a good response ?
QByteArray r = reply->readAll();
QJsonParseError parseError;
QJsonDocument document = QJsonDocument::fromJson(r, &parseError);
// if path was returned all is good, lets set root
if (parseError.error == QJsonParseError::NoError) {
// number of Result Items
if (offset == 0) {
resultCount = document.object()["cnt"].toInt();
emit downloadStarted(resultCount);
}
// results ?
QJsonObject result = document.object()["result"].toObject();
QJsonArray results = result["results"].toArray();
// lets look at that then
for(int i=0; i<results.size(); i++) {
QJsonObject each = results.at(i).toObject();
// check if we have attributes
if (!each.contains("atts")) continue;
// we have an array of attributes for a day
QJsonArray atts = each["atts"].toArray();
for (int j=0; j<atts.size(); j++) {
QJsonObject record = atts.at(j).toObject();
if (record.contains("ts") && record.contains("weight")) {
Measure add;
add.when = QDateTime::fromMSecsSinceEpoch(record["ts"].toDouble());
add.values[Measure::WeightKg] = record["weight"].toDouble();
add.values[Measure::BonesKg] = record["boneMass"].toDouble();
add.values[Measure::FatKg] = record["fatMass"].toDouble();
add.values[Measure::MuscleKg] = record["muscleMass"].toDouble();
add.values[Measure::FatPercent] = record["fat"].toDouble();
add.originalSource = record["source"].toString();
add.source = Measure::TodaysPlan;
data.append(add);
}
}
}
// next page
offset += pageSize;
emit downloadProgress(offset);
} else {
// we had a parsing error - so something is wrong - stop requesting more data
error = tr("Response parsing error: %1").arg(parseError.errorString());
return false;
}
}
// all good
emit downloadEnded(resultCount);
return true;
}