2020-03-26 17:59:41 +01:00
|
|
|
// project includes
|
|
|
|
#include <api/API.h>
|
|
|
|
|
|
|
|
// stl includes
|
|
|
|
#include <iostream>
|
|
|
|
#include <iterator>
|
|
|
|
|
|
|
|
// Qt includes
|
|
|
|
#include <QResource>
|
|
|
|
#include <QDateTime>
|
|
|
|
#include <QCryptographicHash>
|
|
|
|
#include <QImage>
|
2022-02-22 20:58:59 +01:00
|
|
|
#include <QImageReader>
|
2020-03-26 17:59:41 +01:00
|
|
|
#include <QBuffer>
|
|
|
|
#include <QByteArray>
|
|
|
|
#include <QTimer>
|
mDNS Support (#1452)
* Allow build, if no grabbers are enabled
* Align available functions to right Qt version
* Update to next development version
* Align available functions to right Qt version
* fix workflows (apt/nightly)
* Disable QNetworkConfigurationManager deprecation warnings
* Initial go on Smart Pointers
* Add Deallocation
* Correct QT_WARNING_DISABLE_DEPRECATED (available since 5.9)
* Cluster Build Variables
* Hyperion Light
* Address build warnings
* Hyperion Light - UI
* Update Protobuf to latest master
* Removed compiler warnings
* Added restart ability to systray
* Correct Protobuf
* Ignore 'no-return' warning on protobuf build
* hyperion-remote: Fix auto discovery of hyperion server
* Fix Qt version override
* Update changelog
* Remove Grabber Components, if no Grabber exists
* Standalone Grabber - Fix fps default
* Remote Control - Have Source Selction accrosswhole screen
* Enable Blackborder detection only, if relevant input sources available
* Enable Blackborder detection only, if relevant input sources available
* Remote UI - rearrange containers
* Checkout
* Fix compilation on windows
* Re-added qmdnsengine template cmake
* chrono added for linux
* Removed existing AVAHI/Bonjour, allow to enable/disable mDNS
* hyperiond macos typo fix
* Fix macOS Bundle build
* Fix macOS bundle info details
* Correct CMake files
* Removed existing AVAHI/Bonjour (2)
* Share hyperion's services via mDNS
* Add mDNS Browser and mDNS for LED-Devices
* Support mDNS discovery for standalone grabbers
* Remove ZLib Dependency & Cleanup
* mDNS - hanle 2.local2 an ".local." domains equally
* Hue - Link discovery to bridge class, workaround port 443 for mDNS discovery
* Fix save button state when switching between devices
* Removed sessions (of other hyperions)
* mDNS Publisher - Simplify service naming
* mDNS refactoring & Forwarder discovery
* mDNS Updates to use device service name
* Consistency of standalone grabbers with mDNS Service Registry
* Merge branch 'hyperion-project:master' into mDNS
* Start JSON and WebServers only after Instance 0 is available
* Remove bespoke qDebug Output again
* MDNS updates and refactor Forwarder
* Minor updates
* Upgrade to CMake 3.1
* typo
* macOS fix
* Correct merge
* - Remove dynamic linker flag from standalone dispmanX Grabber
- Added ability to use system qmdns libs
* Cec handler library will load at runtime
* typo fix
* protobuf changes
* mDNS changes for Windows/macOS
* test window build qmdnsengine
* absolute path to protobuf cmake dir
* Rework Hue Wizard supporting mDNS
* LED-Devices - Retry support + Refactoring (excl. Hue)
* LED-Devices - Refactoring/Retry support Hue + additional alignments
* Address LGTM findings
* Fix CI-Build, revert test changes
* Build Windows in Release mode to avoid python problem
* Correct that WebServerObject is available earlier
* Ensure that instance name in logs for one instance are presented
* Update content LEDs
* Rework mDNS Address lookup
* Fix LED UI
* Fix for non mDNS Services (ignore default port)
* Disbale device when now input is available
* Revert back some updates, ensure last color is updated when switched on
* Handle reopening case and changed IP, port for API-calls
* Add UPD-DDP Device
* WLED support for DDP
* Fix printout
* LEDDevice - Allow more retries, udapte defaults
* LED-Net Devices - Select Custom device, if configured
Co-authored-by: Paulchen Panther <16664240+Paulchen-Panther@users.noreply.github.com>
Co-authored-by: Paulchen Panther <Paulchen-Panter@protonmail.com>
2022-05-01 19:42:47 +02:00
|
|
|
#include <QThread>
|
2020-03-26 17:59:41 +01:00
|
|
|
|
|
|
|
// hyperion includes
|
|
|
|
#include <utils/jsonschema/QJsonFactory.h>
|
|
|
|
#include <utils/jsonschema/QJsonSchemaChecker.h>
|
|
|
|
#include <HyperionConfig.h>
|
|
|
|
#include <utils/SysInfo.h>
|
|
|
|
#include <utils/ColorSys.h>
|
|
|
|
#include <utils/Process.h>
|
|
|
|
|
|
|
|
// ledmapping int <> string transform methods
|
|
|
|
#include <hyperion/ImageProcessor.h>
|
|
|
|
|
|
|
|
// api includes
|
|
|
|
#include <api/JsonCB.h>
|
|
|
|
|
|
|
|
using namespace hyperion;
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
API::API(Logger *log, bool localConnection, QObject *parent)
|
2020-03-26 17:59:41 +01:00
|
|
|
: QObject(parent)
|
|
|
|
{
|
|
|
|
qRegisterMetaType<int64_t>("int64_t");
|
|
|
|
qRegisterMetaType<VideoMode>("VideoMode");
|
|
|
|
qRegisterMetaType<std::map<int, registerData>>("std::map<int,registerData>");
|
|
|
|
|
|
|
|
// Init
|
|
|
|
_log = log;
|
|
|
|
_authManager = AuthManager::getInstance();
|
mDNS Support (#1452)
* Allow build, if no grabbers are enabled
* Align available functions to right Qt version
* Update to next development version
* Align available functions to right Qt version
* fix workflows (apt/nightly)
* Disable QNetworkConfigurationManager deprecation warnings
* Initial go on Smart Pointers
* Add Deallocation
* Correct QT_WARNING_DISABLE_DEPRECATED (available since 5.9)
* Cluster Build Variables
* Hyperion Light
* Address build warnings
* Hyperion Light - UI
* Update Protobuf to latest master
* Removed compiler warnings
* Added restart ability to systray
* Correct Protobuf
* Ignore 'no-return' warning on protobuf build
* hyperion-remote: Fix auto discovery of hyperion server
* Fix Qt version override
* Update changelog
* Remove Grabber Components, if no Grabber exists
* Standalone Grabber - Fix fps default
* Remote Control - Have Source Selction accrosswhole screen
* Enable Blackborder detection only, if relevant input sources available
* Enable Blackborder detection only, if relevant input sources available
* Remote UI - rearrange containers
* Checkout
* Fix compilation on windows
* Re-added qmdnsengine template cmake
* chrono added for linux
* Removed existing AVAHI/Bonjour, allow to enable/disable mDNS
* hyperiond macos typo fix
* Fix macOS Bundle build
* Fix macOS bundle info details
* Correct CMake files
* Removed existing AVAHI/Bonjour (2)
* Share hyperion's services via mDNS
* Add mDNS Browser and mDNS for LED-Devices
* Support mDNS discovery for standalone grabbers
* Remove ZLib Dependency & Cleanup
* mDNS - hanle 2.local2 an ".local." domains equally
* Hue - Link discovery to bridge class, workaround port 443 for mDNS discovery
* Fix save button state when switching between devices
* Removed sessions (of other hyperions)
* mDNS Publisher - Simplify service naming
* mDNS refactoring & Forwarder discovery
* mDNS Updates to use device service name
* Consistency of standalone grabbers with mDNS Service Registry
* Merge branch 'hyperion-project:master' into mDNS
* Start JSON and WebServers only after Instance 0 is available
* Remove bespoke qDebug Output again
* MDNS updates and refactor Forwarder
* Minor updates
* Upgrade to CMake 3.1
* typo
* macOS fix
* Correct merge
* - Remove dynamic linker flag from standalone dispmanX Grabber
- Added ability to use system qmdns libs
* Cec handler library will load at runtime
* typo fix
* protobuf changes
* mDNS changes for Windows/macOS
* test window build qmdnsengine
* absolute path to protobuf cmake dir
* Rework Hue Wizard supporting mDNS
* LED-Devices - Retry support + Refactoring (excl. Hue)
* LED-Devices - Refactoring/Retry support Hue + additional alignments
* Address LGTM findings
* Fix CI-Build, revert test changes
* Build Windows in Release mode to avoid python problem
* Correct that WebServerObject is available earlier
* Ensure that instance name in logs for one instance are presented
* Update content LEDs
* Rework mDNS Address lookup
* Fix LED UI
* Fix for non mDNS Services (ignore default port)
* Disbale device when now input is available
* Revert back some updates, ensure last color is updated when switched on
* Handle reopening case and changed IP, port for API-calls
* Add UPD-DDP Device
* WLED support for DDP
* Fix printout
* LEDDevice - Allow more retries, udapte defaults
* LED-Net Devices - Select Custom device, if configured
Co-authored-by: Paulchen Panther <16664240+Paulchen-Panther@users.noreply.github.com>
Co-authored-by: Paulchen Panther <Paulchen-Panter@protonmail.com>
2022-05-01 19:42:47 +02:00
|
|
|
_instanceManager = HyperionIManager::getInstance();
|
2020-03-26 17:59:41 +01:00
|
|
|
_localConnection = localConnection;
|
|
|
|
|
|
|
|
_authorized = false;
|
|
|
|
_adminAuthorized = false;
|
|
|
|
|
|
|
|
_currInstanceIndex = 0;
|
|
|
|
|
|
|
|
// connect to possible token responses that has been requested
|
2020-10-18 17:05:07 +02:00
|
|
|
connect(_authManager, &AuthManager::tokenResponse, [=] (bool success, QObject *caller, const QString &token, const QString &comment, const QString &id, const int &tan)
|
|
|
|
{
|
|
|
|
if (this == caller)
|
|
|
|
emit onTokenResponse(success, token, comment, id, tan);
|
|
|
|
});
|
|
|
|
|
|
|
|
// connect to possible startInstance responses that has been requested
|
|
|
|
connect(_instanceManager, &HyperionIManager::startInstanceResponse, [=] (QObject *caller, const int &tan)
|
|
|
|
{
|
|
|
|
if (this == caller)
|
|
|
|
emit onStartInstanceResponse(tan);
|
|
|
|
});
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::init()
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
mDNS Support (#1452)
* Allow build, if no grabbers are enabled
* Align available functions to right Qt version
* Update to next development version
* Align available functions to right Qt version
* fix workflows (apt/nightly)
* Disable QNetworkConfigurationManager deprecation warnings
* Initial go on Smart Pointers
* Add Deallocation
* Correct QT_WARNING_DISABLE_DEPRECATED (available since 5.9)
* Cluster Build Variables
* Hyperion Light
* Address build warnings
* Hyperion Light - UI
* Update Protobuf to latest master
* Removed compiler warnings
* Added restart ability to systray
* Correct Protobuf
* Ignore 'no-return' warning on protobuf build
* hyperion-remote: Fix auto discovery of hyperion server
* Fix Qt version override
* Update changelog
* Remove Grabber Components, if no Grabber exists
* Standalone Grabber - Fix fps default
* Remote Control - Have Source Selction accrosswhole screen
* Enable Blackborder detection only, if relevant input sources available
* Enable Blackborder detection only, if relevant input sources available
* Remote UI - rearrange containers
* Checkout
* Fix compilation on windows
* Re-added qmdnsengine template cmake
* chrono added for linux
* Removed existing AVAHI/Bonjour, allow to enable/disable mDNS
* hyperiond macos typo fix
* Fix macOS Bundle build
* Fix macOS bundle info details
* Correct CMake files
* Removed existing AVAHI/Bonjour (2)
* Share hyperion's services via mDNS
* Add mDNS Browser and mDNS for LED-Devices
* Support mDNS discovery for standalone grabbers
* Remove ZLib Dependency & Cleanup
* mDNS - hanle 2.local2 an ".local." domains equally
* Hue - Link discovery to bridge class, workaround port 443 for mDNS discovery
* Fix save button state when switching between devices
* Removed sessions (of other hyperions)
* mDNS Publisher - Simplify service naming
* mDNS refactoring & Forwarder discovery
* mDNS Updates to use device service name
* Consistency of standalone grabbers with mDNS Service Registry
* Merge branch 'hyperion-project:master' into mDNS
* Start JSON and WebServers only after Instance 0 is available
* Remove bespoke qDebug Output again
* MDNS updates and refactor Forwarder
* Minor updates
* Upgrade to CMake 3.1
* typo
* macOS fix
* Correct merge
* - Remove dynamic linker flag from standalone dispmanX Grabber
- Added ability to use system qmdns libs
* Cec handler library will load at runtime
* typo fix
* protobuf changes
* mDNS changes for Windows/macOS
* test window build qmdnsengine
* absolute path to protobuf cmake dir
* Rework Hue Wizard supporting mDNS
* LED-Devices - Retry support + Refactoring (excl. Hue)
* LED-Devices - Refactoring/Retry support Hue + additional alignments
* Address LGTM findings
* Fix CI-Build, revert test changes
* Build Windows in Release mode to avoid python problem
* Correct that WebServerObject is available earlier
* Ensure that instance name in logs for one instance are presented
* Update content LEDs
* Rework mDNS Address lookup
* Fix LED UI
* Fix for non mDNS Services (ignore default port)
* Disbale device when now input is available
* Revert back some updates, ensure last color is updated when switched on
* Handle reopening case and changed IP, port for API-calls
* Add UPD-DDP Device
* WLED support for DDP
* Fix printout
* LEDDevice - Allow more retries, udapte defaults
* LED-Net Devices - Select Custom device, if configured
Co-authored-by: Paulchen Panther <16664240+Paulchen-Panther@users.noreply.github.com>
Co-authored-by: Paulchen Panther <Paulchen-Panter@protonmail.com>
2022-05-01 19:42:47 +02:00
|
|
|
_hyperion = _instanceManager->getHyperionInstance(0);
|
2022-01-22 17:48:03 +01:00
|
|
|
|
2020-03-26 17:59:41 +01:00
|
|
|
bool apiAuthRequired = _authManager->isAuthRequired();
|
|
|
|
|
|
|
|
// For security we block external connections if default PW is set
|
|
|
|
if (!_localConnection && API::hasHyperionDefaultPw())
|
|
|
|
{
|
|
|
|
emit forceClose();
|
|
|
|
}
|
|
|
|
// if this is localConnection and network allows unauth locals, set authorized flag
|
|
|
|
if (apiAuthRequired && _localConnection)
|
2022-02-22 20:58:59 +01:00
|
|
|
{
|
2020-03-26 17:59:41 +01:00
|
|
|
_authorized = !_authManager->isLocalAuthRequired();
|
2022-02-22 20:58:59 +01:00
|
|
|
}
|
2020-03-26 17:59:41 +01:00
|
|
|
|
|
|
|
// admin access is allowed, when the connection is local and the option for local admin isn't set. Con: All local connections get full access
|
|
|
|
if (_localConnection)
|
|
|
|
{
|
|
|
|
_adminAuthorized = !_authManager->isLocalAdminAuthRequired();
|
|
|
|
// just in positive direction
|
2022-02-22 20:58:59 +01:00
|
|
|
if (_adminAuthorized)
|
|
|
|
{
|
|
|
|
_authorized = true;
|
|
|
|
}
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::setColor(int priority, const std::vector<uint8_t> &ledColors, int timeout_ms, const QString &origin, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
std::vector<ColorRgb> fledColors;
|
|
|
|
if (ledColors.size() % 3 == 0)
|
|
|
|
{
|
|
|
|
for (unsigned i = 0; i < ledColors.size(); i += 3)
|
|
|
|
{
|
|
|
|
fledColors.emplace_back(ColorRgb{ledColors[i], ledColors[i + 1], ledColors[i + 2]});
|
|
|
|
}
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "setColor", Qt::QueuedConnection, Q_ARG(int, priority), Q_ARG(std::vector<ColorRgb>, fledColors), Q_ARG(int, timeout_ms), Q_ARG(QString, origin));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
bool API::setImage(ImageCmdData &data, hyperion::Components comp, QString &replyMsg, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
// truncate name length
|
|
|
|
data.imgName.truncate(16);
|
|
|
|
|
2022-02-22 20:58:59 +01:00
|
|
|
if (!data.format.isEmpty())
|
|
|
|
{
|
|
|
|
if (data.format == "auto")
|
|
|
|
{
|
|
|
|
data.format = "";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!QImageReader::supportedImageFormats().contains(data.format.toLower().toUtf8()))
|
|
|
|
{
|
|
|
|
replyMsg = "The given format [" + data.format + "] is not supported";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QImage img = QImage::fromData(data.data, QSTRING_CSTR(data.format));
|
2020-03-26 17:59:41 +01:00
|
|
|
if (img.isNull())
|
|
|
|
{
|
2022-02-22 20:58:59 +01:00
|
|
|
replyMsg = "Failed to parse picture, the file might be corrupted or content does not match the given format [" + data.format + "]";
|
2020-03-26 17:59:41 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for requested scale
|
|
|
|
if (data.scale > 24)
|
|
|
|
{
|
|
|
|
if (img.height() > data.scale)
|
|
|
|
{
|
|
|
|
img = img.scaledToHeight(data.scale);
|
|
|
|
}
|
|
|
|
if (img.width() > data.scale)
|
|
|
|
{
|
|
|
|
img = img.scaledToWidth(data.scale);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if we need to force a scale
|
|
|
|
if (img.width() > 2000 || img.height() > 2000)
|
|
|
|
{
|
|
|
|
data.scale = 2000;
|
|
|
|
if (img.height() > data.scale)
|
|
|
|
{
|
|
|
|
img = img.scaledToHeight(data.scale);
|
|
|
|
}
|
|
|
|
if (img.width() > data.scale)
|
|
|
|
{
|
|
|
|
img = img.scaledToWidth(data.scale);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
data.width = img.width();
|
|
|
|
data.height = img.height();
|
|
|
|
|
|
|
|
// extract image
|
|
|
|
img = img.convertToFormat(QImage::Format_ARGB32_Premultiplied);
|
|
|
|
data.data.clear();
|
|
|
|
data.data.reserve(img.width() * img.height() * 3);
|
|
|
|
for (int i = 0; i < img.height(); ++i)
|
|
|
|
{
|
|
|
|
const QRgb *scanline = reinterpret_cast<const QRgb *>(img.scanLine(i));
|
|
|
|
for (int j = 0; j < img.width(); ++j)
|
|
|
|
{
|
|
|
|
data.data.append((char)qRed(scanline[j]));
|
|
|
|
data.data.append((char)qGreen(scanline[j]));
|
|
|
|
data.data.append((char)qBlue(scanline[j]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// check consistency of the size of the received data
|
|
|
|
if (data.data.size() != data.width * data.height * 3)
|
|
|
|
{
|
|
|
|
replyMsg = "Size of image data does not match with the width and height";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// copy image
|
|
|
|
Image<ColorRgb> image(data.width, data.height);
|
|
|
|
memcpy(image.memptr(), data.data.data(), data.data.size());
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "registerInput", Qt::QueuedConnection, Q_ARG(int, data.priority), Q_ARG(hyperion::Components, comp), Q_ARG(QString, data.origin), Q_ARG(QString, data.imgName));
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "setInputImage", Qt::QueuedConnection, Q_ARG(int, data.priority), Q_ARG(Image<ColorRgb>, image), Q_ARG(int64_t, data.duration));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
bool API::clearPriority(int priority, QString &replyMsg, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
if (priority < 0 || (priority > 0 && priority < 254))
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "clear", Qt::QueuedConnection, Q_ARG(int, priority));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
replyMsg = QString("Priority %1 is not allowed to be cleared").arg(priority);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
bool API::setComponentState(const QString &comp, bool &compState, QString &replyMsg, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
Components component = stringToComponent(comp);
|
|
|
|
|
|
|
|
if (component != COMP_INVALID)
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "compStateChangeRequest", Qt::QueuedConnection, Q_ARG(hyperion::Components, component), Q_ARG(bool, compState));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
replyMsg = QString("Unknown component name: %1").arg(comp);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::setLedMappingType(int type, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "setLedMappingType", Qt::QueuedConnection, Q_ARG(int, type));
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::setVideoMode(VideoMode mode, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "setVideoMode", Qt::QueuedConnection, Q_ARG(VideoMode, mode));
|
|
|
|
}
|
|
|
|
|
2022-02-11 20:36:15 +01:00
|
|
|
#if defined(ENABLE_EFFECTENGINE)
|
2020-10-18 17:05:07 +02:00
|
|
|
bool API::setEffect(const EffectCmdData &dat, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
2020-10-18 17:05:07 +02:00
|
|
|
int res;
|
2020-03-26 17:59:41 +01:00
|
|
|
if (!dat.args.isEmpty())
|
|
|
|
{
|
2020-10-18 17:05:07 +02:00
|
|
|
QMetaObject::invokeMethod(_hyperion, "setEffect", Qt::BlockingQueuedConnection, Q_RETURN_ARG(int, res), Q_ARG(QString, dat.effectName), Q_ARG(QJsonObject, dat.args), Q_ARG(int, dat.priority), Q_ARG(int, dat.duration), Q_ARG(QString, dat.pythonScript), Q_ARG(QString, dat.origin), Q_ARG(QString, dat.data));
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-10-18 17:05:07 +02:00
|
|
|
QMetaObject::invokeMethod(_hyperion, "setEffect", Qt::BlockingQueuedConnection, Q_RETURN_ARG(int, res), Q_ARG(QString, dat.effectName), Q_ARG(int, dat.priority), Q_ARG(int, dat.duration), Q_ARG(QString, dat.origin));
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
2020-10-18 17:05:07 +02:00
|
|
|
|
|
|
|
return res >= 0;
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
2022-02-11 20:36:15 +01:00
|
|
|
#endif
|
2020-03-26 17:59:41 +01:00
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::setSourceAutoSelect(bool state, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "setSourceAutoSelect", Qt::QueuedConnection, Q_ARG(bool, state));
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::setVisiblePriority(int priority, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "setVisiblePriority", Qt::QueuedConnection, Q_ARG(int, priority));
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::registerInput(int priority, hyperion::Components component, const QString &origin, const QString &owner, hyperion::Components callerComp)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
if (_activeRegisters.count(priority))
|
|
|
|
_activeRegisters.erase(priority);
|
|
|
|
|
|
|
|
_activeRegisters.insert({priority, registerData{component, origin, owner, callerComp}});
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "registerInput", Qt::QueuedConnection, Q_ARG(int, priority), Q_ARG(hyperion::Components, component), Q_ARG(QString, origin), Q_ARG(QString, owner));
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::unregisterInput(int priority)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
if (_activeRegisters.count(priority))
|
|
|
|
_activeRegisters.erase(priority);
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
bool API::setHyperionInstance(quint8 inst)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
if (_currInstanceIndex == inst)
|
|
|
|
return true;
|
|
|
|
bool isRunning;
|
2020-04-17 20:16:33 +02:00
|
|
|
QMetaObject::invokeMethod(_instanceManager, "IsInstanceRunning", Qt::DirectConnection, Q_RETURN_ARG(bool, isRunning), Q_ARG(quint8, inst));
|
2020-03-26 17:59:41 +01:00
|
|
|
if (!isRunning)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
disconnect(_hyperion, 0, this, 0);
|
2020-04-17 20:16:33 +02:00
|
|
|
QMetaObject::invokeMethod(_instanceManager, "getHyperionInstance", Qt::DirectConnection, Q_RETURN_ARG(Hyperion *, _hyperion), Q_ARG(quint8, inst));
|
2020-03-26 17:59:41 +01:00
|
|
|
_currInstanceIndex = inst;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::map<hyperion::Components, bool> API::getAllComponents()
|
|
|
|
{
|
|
|
|
std::map<hyperion::Components, bool> comps;
|
|
|
|
//QMetaObject::invokeMethod(_hyperion, "getAllComponents", Qt::BlockingQueuedConnection, Q_RETURN_ARG(std::map<hyperion::Components, bool>, comps));
|
|
|
|
return comps;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool API::isHyperionEnabled()
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "isComponentEnabled", Qt::BlockingQueuedConnection, Q_RETURN_ARG(int, res), Q_ARG(hyperion::Components, hyperion::COMP_ALL));
|
|
|
|
return res > 0;
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
QVector<QVariantMap> API::getAllInstanceData()
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
QVector<QVariantMap> vec;
|
2020-04-17 20:16:33 +02:00
|
|
|
QMetaObject::invokeMethod(_instanceManager, "getInstanceData", Qt::DirectConnection, Q_RETURN_ARG(QVector<QVariantMap>, vec));
|
2020-03-26 17:59:41 +01:00
|
|
|
return vec;
|
|
|
|
}
|
|
|
|
|
2020-10-18 17:05:07 +02:00
|
|
|
bool API::startInstance(quint8 index, int tan)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
2020-10-18 17:05:07 +02:00
|
|
|
bool res;
|
|
|
|
(_instanceManager->thread() != this->thread())
|
|
|
|
? QMetaObject::invokeMethod(_instanceManager, "startInstance", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, res), Q_ARG(quint8, index), Q_ARG(bool, false), Q_ARG(QObject*, this), Q_ARG(int, tan))
|
|
|
|
: res = _instanceManager->startInstance(index, false, this, tan);
|
|
|
|
|
|
|
|
return res;
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void API::stopInstance(quint8 index)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_instanceManager, "stopInstance", Qt::QueuedConnection, Q_ARG(quint8, index));
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
bool API::deleteInstance(quint8 index, QString &replyMsg)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
if (_adminAuthorized)
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_instanceManager, "deleteInstance", Qt::QueuedConnection, Q_ARG(quint8, index));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
replyMsg = NO_AUTH;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString API::createInstance(const QString &name)
|
|
|
|
{
|
|
|
|
if (_adminAuthorized)
|
|
|
|
{
|
|
|
|
bool success;
|
2020-04-17 20:16:33 +02:00
|
|
|
QMetaObject::invokeMethod(_instanceManager, "createInstance", Qt::DirectConnection, Q_RETURN_ARG(bool, success), Q_ARG(QString, name));
|
2020-03-26 17:59:41 +01:00
|
|
|
if (!success)
|
|
|
|
return QString("Instance name '%1' is already in use").arg(name);
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
return NO_AUTH;
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
QString API::setInstanceName(quint8 index, const QString &name)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
if (_adminAuthorized)
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_instanceManager, "saveName", Qt::QueuedConnection, Q_ARG(quint8, index), Q_ARG(QString, name));
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
return NO_AUTH;
|
|
|
|
}
|
|
|
|
|
2022-02-11 20:36:15 +01:00
|
|
|
#if defined(ENABLE_EFFECTENGINE)
|
2020-03-26 17:59:41 +01:00
|
|
|
QString API::deleteEffect(const QString &name)
|
|
|
|
{
|
|
|
|
if (_adminAuthorized)
|
|
|
|
{
|
|
|
|
QString res;
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "deleteEffect", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QString, res), Q_ARG(QString, name));
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
return NO_AUTH;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString API::saveEffect(const QJsonObject &data)
|
|
|
|
{
|
|
|
|
if (_adminAuthorized)
|
|
|
|
{
|
|
|
|
QString res;
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "saveEffect", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QString, res), Q_ARG(QJsonObject, data));
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
return NO_AUTH;
|
|
|
|
}
|
2022-02-11 20:36:15 +01:00
|
|
|
#endif
|
2020-03-26 17:59:41 +01:00
|
|
|
|
2020-11-01 19:47:30 +01:00
|
|
|
bool API::saveSettings(const QJsonObject &data)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
2020-11-01 19:47:30 +01:00
|
|
|
bool rc = true;
|
2020-03-26 17:59:41 +01:00
|
|
|
if (!_adminAuthorized)
|
2020-11-01 19:47:30 +01:00
|
|
|
{
|
|
|
|
rc = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "saveSettings", Qt::DirectConnection, Q_RETURN_ARG(bool, rc), Q_ARG(QJsonObject, data), Q_ARG(bool, true));
|
|
|
|
}
|
|
|
|
return rc;
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
|
|
|
|
2021-11-17 20:30:43 +00:00
|
|
|
bool API::restoreSettings(const QJsonObject &data)
|
|
|
|
{
|
|
|
|
bool rc = true;
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
{
|
|
|
|
rc = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_hyperion, "restoreSettings", Qt::DirectConnection, Q_RETURN_ARG(bool, rc), Q_ARG(QJsonObject, data), Q_ARG(bool, true));
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2020-03-26 17:59:41 +01:00
|
|
|
bool API::updateHyperionPassword(const QString &password, const QString &newPassword)
|
|
|
|
{
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
return false;
|
|
|
|
bool res;
|
|
|
|
QMetaObject::invokeMethod(_authManager, "updateUserPassword", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, res), Q_ARG(QString, "Hyperion"), Q_ARG(QString, password), Q_ARG(QString, newPassword));
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString API::createToken(const QString &comment, AuthManager::AuthDefinition &def)
|
|
|
|
{
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
return NO_AUTH;
|
|
|
|
if (comment.isEmpty())
|
|
|
|
return "comment is empty";
|
|
|
|
QMetaObject::invokeMethod(_authManager, "createToken", Qt::BlockingQueuedConnection, Q_RETURN_ARG(AuthManager::AuthDefinition, def), Q_ARG(QString, comment));
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
QString API::renameToken(const QString &id, const QString &comment)
|
|
|
|
{
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
return NO_AUTH;
|
|
|
|
if (comment.isEmpty() || id.isEmpty())
|
|
|
|
return "Empty comment or id";
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(_authManager, "renameToken", Qt::QueuedConnection, Q_ARG(QString, id), Q_ARG(QString, comment));
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
QString API::deleteToken(const QString &id)
|
|
|
|
{
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
return NO_AUTH;
|
|
|
|
if (id.isEmpty())
|
|
|
|
return "Empty id";
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(_authManager, "deleteToken", Qt::QueuedConnection, Q_ARG(QString, id));
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2020-10-18 17:05:07 +02:00
|
|
|
void API::setNewTokenRequest(const QString &comment, const QString &id, const int &tan)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
2020-10-18 17:05:07 +02:00
|
|
|
QMetaObject::invokeMethod(_authManager, "setNewTokenRequest", Qt::QueuedConnection, Q_ARG(QObject *, this), Q_ARG(QString, comment), Q_ARG(QString, id), Q_ARG(int, tan));
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void API::cancelNewTokenRequest(const QString &comment, const QString &id)
|
|
|
|
{
|
|
|
|
QMetaObject::invokeMethod(_authManager, "cancelNewTokenRequest", Qt::QueuedConnection, Q_ARG(QObject *, this), Q_ARG(QString, comment), Q_ARG(QString, id));
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
bool API::handlePendingTokenRequest(const QString &id, bool accept)
|
2020-03-26 17:59:41 +01:00
|
|
|
{
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
return false;
|
|
|
|
QMetaObject::invokeMethod(_authManager, "handlePendingTokenRequest", Qt::QueuedConnection, Q_ARG(QString, id), Q_ARG(bool, accept));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool API::getTokenList(QVector<AuthManager::AuthDefinition> &def)
|
|
|
|
{
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
return false;
|
|
|
|
QMetaObject::invokeMethod(_authManager, "getTokenList", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QVector<AuthManager::AuthDefinition>, def));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool API::getPendingTokenRequests(QVector<AuthManager::AuthDefinition> &map)
|
|
|
|
{
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
return false;
|
|
|
|
QMetaObject::invokeMethod(_authManager, "getPendingRequests", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QVector<AuthManager::AuthDefinition>, map));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool API::isUserTokenAuthorized(const QString &userToken)
|
|
|
|
{
|
|
|
|
bool res;
|
|
|
|
QMetaObject::invokeMethod(_authManager, "isUserTokenAuthorized", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, res), Q_ARG(QString, "Hyperion"), Q_ARG(QString, userToken));
|
|
|
|
if (res)
|
|
|
|
{
|
|
|
|
_authorized = true;
|
|
|
|
_adminAuthorized = true;
|
|
|
|
// Listen for ADMIN ACCESS protected signals
|
|
|
|
connect(_authManager, &AuthManager::newPendingTokenRequest, this, &API::onPendingTokenRequest, Qt::UniqueConnection);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool API::getUserToken(QString &userToken)
|
|
|
|
{
|
|
|
|
if (!_adminAuthorized)
|
|
|
|
return false;
|
|
|
|
QMetaObject::invokeMethod(_authManager, "getUserToken", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QString, userToken));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool API::isTokenAuthorized(const QString &token)
|
|
|
|
{
|
2020-10-18 17:05:07 +02:00
|
|
|
(_authManager->thread() != this->thread())
|
|
|
|
? QMetaObject::invokeMethod(_authManager, "isTokenAuthorized", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, _authorized), Q_ARG(QString, token))
|
|
|
|
: _authorized = _authManager->isTokenAuthorized(token);
|
2020-03-26 17:59:41 +01:00
|
|
|
|
2020-10-18 17:05:07 +02:00
|
|
|
return _authorized;
|
2020-03-26 17:59:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool API::isUserAuthorized(const QString &password)
|
|
|
|
{
|
|
|
|
bool res;
|
|
|
|
QMetaObject::invokeMethod(_authManager, "isUserAuthorized", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, res), Q_ARG(QString, "Hyperion"), Q_ARG(QString, password));
|
|
|
|
if (res)
|
|
|
|
{
|
|
|
|
_authorized = true;
|
|
|
|
_adminAuthorized = true;
|
|
|
|
// Listen for ADMIN ACCESS protected signals
|
|
|
|
connect(_authManager, &AuthManager::newPendingTokenRequest, this, &API::onPendingTokenRequest, Qt::UniqueConnection);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool API::hasHyperionDefaultPw()
|
|
|
|
{
|
|
|
|
bool res;
|
|
|
|
QMetaObject::invokeMethod(_authManager, "isUserAuthorized", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, res), Q_ARG(QString, "Hyperion"), Q_ARG(QString, "hyperion"));
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
void API::logout()
|
|
|
|
{
|
|
|
|
_authorized = false;
|
|
|
|
_adminAuthorized = false;
|
|
|
|
// Stop listenig for ADMIN ACCESS protected signals
|
|
|
|
disconnect(_authManager, &AuthManager::newPendingTokenRequest, this, &API::onPendingTokenRequest);
|
|
|
|
stopDataConnectionss();
|
|
|
|
}
|
|
|
|
|
|
|
|
void API::stopDataConnectionss()
|
|
|
|
{
|
|
|
|
}
|