mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2023-10-10 13:36:59 +02:00
f0bd38d473
* Discover additional Nanoleaf devices * Fix Nanoleaf not turning on * Added LGTM configuration file * Allow to pass QJsonObject as payload for put * Nanoleaf - Support Restore State & Overwrite Brightness * Removed because this is already included Co-authored-by: Markus <16664240+Paulchen-Panther@users.noreply.github.com>
279 lines
6.4 KiB
C++
279 lines
6.4 KiB
C++
// Local-Hyperion includes
|
|
#include "ProviderRestApi.h"
|
|
|
|
// Qt includes
|
|
#include <QEventLoop>
|
|
#include <QNetworkReply>
|
|
#include <QByteArray>
|
|
#include <QJsonObject>
|
|
|
|
//std includes
|
|
#include <iostream>
|
|
#include <chrono>
|
|
|
|
// Constants
|
|
namespace {
|
|
|
|
const QChar ONE_SLASH = '/';
|
|
|
|
const int HTTP_STATUS_NO_CONTENT = 204;
|
|
const int HTTP_STATUS_BAD_REQUEST = 400;
|
|
const int HTTP_STATUS_UNAUTHORIZED = 401;
|
|
const int HTTP_STATUS_NOT_FOUND = 404;
|
|
|
|
constexpr std::chrono::milliseconds DEFAULT_REST_TIMEOUT{ 400 };
|
|
|
|
} //End of constants
|
|
|
|
ProviderRestApi::ProviderRestApi(const QString &host, int port, const QString &basePath)
|
|
:_log(Logger::getInstance("LEDDEVICE"))
|
|
,_networkManager(nullptr)
|
|
,_scheme("http")
|
|
,_hostname(host)
|
|
,_port(port)
|
|
{
|
|
_networkManager = new QNetworkAccessManager();
|
|
|
|
_apiUrl.setScheme(_scheme);
|
|
_apiUrl.setHost(host);
|
|
_apiUrl.setPort(port);
|
|
_basePath = basePath;
|
|
}
|
|
|
|
ProviderRestApi::ProviderRestApi(const QString &host, int port)
|
|
: ProviderRestApi(host, port, "") {}
|
|
|
|
ProviderRestApi::ProviderRestApi()
|
|
: ProviderRestApi("", -1) {}
|
|
|
|
ProviderRestApi::~ProviderRestApi()
|
|
{
|
|
delete _networkManager;
|
|
}
|
|
|
|
void ProviderRestApi::setBasePath(const QString &basePath)
|
|
{
|
|
_basePath.clear();
|
|
appendPath (_basePath, basePath );
|
|
}
|
|
|
|
void ProviderRestApi::setPath ( const QString &path )
|
|
{
|
|
_path.clear();
|
|
appendPath (_path, path );
|
|
}
|
|
|
|
void ProviderRestApi::appendPath ( const QString &path )
|
|
{
|
|
appendPath (_path, path );
|
|
}
|
|
|
|
void ProviderRestApi::appendPath ( QString& path, const QString &appendPath)
|
|
{
|
|
if ( !appendPath.isEmpty() && appendPath != ONE_SLASH )
|
|
{
|
|
if (path.isEmpty() || path == ONE_SLASH )
|
|
{
|
|
path.clear();
|
|
if (appendPath[0] != ONE_SLASH )
|
|
{
|
|
path.push_back(ONE_SLASH);
|
|
}
|
|
}
|
|
else if (path[path.size()-1] == ONE_SLASH && appendPath[0] == ONE_SLASH)
|
|
{
|
|
path.chop(1);
|
|
}
|
|
else if (path[path.size()-1] != ONE_SLASH && appendPath[0] != ONE_SLASH)
|
|
{
|
|
path.push_back(ONE_SLASH);
|
|
}
|
|
else
|
|
{
|
|
// Only one slash.
|
|
}
|
|
|
|
path.append(appendPath);
|
|
}
|
|
}
|
|
|
|
void ProviderRestApi::setFragment(const QString &fragment)
|
|
{
|
|
_fragment = fragment;
|
|
}
|
|
|
|
void ProviderRestApi::setQuery(const QUrlQuery &query)
|
|
{
|
|
_query = query;
|
|
}
|
|
|
|
QUrl ProviderRestApi::getUrl() const
|
|
{
|
|
QUrl url = _apiUrl;
|
|
|
|
QString fullPath = _basePath;
|
|
appendPath (fullPath, _path );
|
|
|
|
url.setPath(fullPath);
|
|
url.setFragment( _fragment );
|
|
url.setQuery( _query );
|
|
return url;
|
|
}
|
|
|
|
httpResponse ProviderRestApi::get()
|
|
{
|
|
return get( getUrl() );
|
|
}
|
|
|
|
httpResponse ProviderRestApi::get(const QUrl &url)
|
|
{
|
|
// Perform request
|
|
QNetworkRequest request(url);
|
|
QNetworkReply* reply = _networkManager->get(request);
|
|
|
|
// Connect requestFinished signal to quit slot of the loop.
|
|
QEventLoop loop;
|
|
QEventLoop::connect(reply, &QNetworkReply::finished, &loop, &QEventLoop::quit);
|
|
|
|
ReplyTimeout::set(reply, DEFAULT_REST_TIMEOUT.count());
|
|
|
|
// Go into the loop until the request is finished.
|
|
loop.exec();
|
|
|
|
httpResponse response;
|
|
if(reply->operation() == QNetworkAccessManager::GetOperation)
|
|
{
|
|
if(reply->error() != QNetworkReply::NoError)
|
|
{
|
|
Debug(_log, "GET: [%s]", QSTRING_CSTR( url.toString() ));
|
|
}
|
|
response = getResponse(reply );
|
|
}
|
|
// Free space.
|
|
reply->deleteLater();
|
|
// Return response
|
|
return response;
|
|
}
|
|
|
|
httpResponse ProviderRestApi::put(const QJsonObject &body)
|
|
{
|
|
return put( getUrl(), QJsonDocument(body).toJson(QJsonDocument::Compact));
|
|
}
|
|
|
|
httpResponse ProviderRestApi::put(const QString &body)
|
|
{
|
|
return put( getUrl(), body.toUtf8() );
|
|
}
|
|
|
|
httpResponse ProviderRestApi::put(const QUrl &url, const QByteArray &body)
|
|
{
|
|
// Perform request
|
|
QNetworkRequest request(url);
|
|
QNetworkReply* reply = _networkManager->put(request, body);
|
|
// Connect requestFinished signal to quit slot of the loop.
|
|
QEventLoop loop;
|
|
QEventLoop::connect(reply, &QNetworkReply::finished, &loop, &QEventLoop::quit);
|
|
|
|
ReplyTimeout::set(reply, DEFAULT_REST_TIMEOUT.count());
|
|
|
|
// Go into the loop until the request is finished.
|
|
loop.exec();
|
|
|
|
httpResponse response;
|
|
if(reply->operation() == QNetworkAccessManager::PutOperation)
|
|
{
|
|
if(reply->error() != QNetworkReply::NoError)
|
|
{
|
|
Debug(_log, "PUT: [%s] [%s]", QSTRING_CSTR( url.toString() ),body.constData() );
|
|
}
|
|
response = getResponse(reply);
|
|
}
|
|
// Free space.
|
|
reply->deleteLater();
|
|
|
|
// Return response
|
|
return response;
|
|
}
|
|
|
|
httpResponse ProviderRestApi::getResponse(QNetworkReply* const &reply)
|
|
{
|
|
httpResponse response;
|
|
|
|
int httpStatusCode = reply->attribute( QNetworkRequest::HttpStatusCodeAttribute ).toInt();
|
|
response.setHttpStatusCode(httpStatusCode);
|
|
response.setNetworkReplyError(reply->error());
|
|
|
|
if(reply->error() == QNetworkReply::NoError)
|
|
{
|
|
if ( httpStatusCode != HTTP_STATUS_NO_CONTENT ){
|
|
QByteArray replyData = reply->readAll();
|
|
|
|
if ( !replyData.isEmpty())
|
|
{
|
|
QJsonParseError error;
|
|
QJsonDocument jsonDoc = QJsonDocument::fromJson(replyData, &error);
|
|
|
|
if (error.error != QJsonParseError::NoError)
|
|
{
|
|
//Received not valid JSON response
|
|
//std::cout << "Response: [" << replyData.toStdString() << "]" << std::endl;
|
|
response.setError(true);
|
|
response.setErrorReason(error.errorString());
|
|
}
|
|
else
|
|
{
|
|
//std::cout << "Response: [" << QString (jsonDoc.toJson(QJsonDocument::Compact)).toStdString() << "]" << std::endl;
|
|
response.setBody( jsonDoc );
|
|
}
|
|
}
|
|
else
|
|
{ // Create valid body which is empty
|
|
response.setBody( QJsonDocument() );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Debug(_log, "Reply.httpStatusCode [%d]", httpStatusCode );
|
|
QString errorReason;
|
|
if ( httpStatusCode > 0 ) {
|
|
QString httpReason = reply->attribute( QNetworkRequest::HttpReasonPhraseAttribute ).toString();
|
|
QString advise;
|
|
switch ( httpStatusCode ) {
|
|
case HTTP_STATUS_BAD_REQUEST:
|
|
advise = "Check Request Body";
|
|
break;
|
|
case HTTP_STATUS_UNAUTHORIZED:
|
|
advise = "Check Authentication Token (API Key)";
|
|
break;
|
|
case HTTP_STATUS_NOT_FOUND:
|
|
advise = "Check Resource given";
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
errorReason = QString ("[%3 %4] - %5").arg(QString(httpStatusCode) , httpReason, advise);
|
|
}
|
|
else {
|
|
|
|
errorReason = reply->errorString();
|
|
|
|
if ( reply->error() == QNetworkReply::OperationCanceledError )
|
|
{
|
|
//Do not report errors caused by request cancellation because of timeouts
|
|
Debug(_log, "Reply: [%s]", QSTRING_CSTR(errorReason) );
|
|
}
|
|
else
|
|
{
|
|
response.setError(true);
|
|
response.setErrorReason(errorReason);
|
|
}
|
|
}
|
|
|
|
// Create valid body which is empty
|
|
response.setBody( QJsonDocument() );
|
|
}
|
|
return response;
|
|
}
|
|
|