mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2023-10-10 13:36:59 +02:00
160c5d0b3a
* Stop Web-Capture when priority changes * Remote control UI: Treat duration=0 as endless * Stop Web-Capture on non-Image events changes * LED Matrix Layout - Support vertical cabling direction * Additional Yeelight models * Treat http headers case insensitive * Update change log * Treat http headers case insensitive (consider Qt version) * API - Consider provided format when setImage * UI - Support Boblight configuration per LED instance * Support multiple Boblight clients with different priorities * Update changelog * Simplify isGUI rules allowing for QT only builds * Sysinfo: Fix indents * LED-Devices: Show warning, if get properties failed * Qt-Grabber: Fixed position handling of multiple monitors * LED layout: Remove indention limitations * Yeelight: Test YLTD003 * hyperion-remote: Provide image filename to muxer/UI * Refactor PriorityMuxer and related * Temp: Build under Windows 2019 * Yeelight: Remove YLTD003 as it is not working without additional changes * Test Windows-latest with out removing redistributables/new MSVC * correct workflows * correct CI script * Build Windows with Qt 5.15.2 * Priority Muxer: Updates after testing * Fix Typo * Update BGHandler * QTGrabber - Reactivate windows code to avoid cursor issues * Emit prioritiesChanged when autoselect was changed by user Co-authored-by: Paulchen Panther <Paulchen-Panter@protonmail.com>
328 lines
9.7 KiB
C++
328 lines
9.7 KiB
C++
// STL includes
|
|
#include <stdexcept>
|
|
|
|
// project includes
|
|
#include <forwarder/MessageForwarder.h>
|
|
|
|
// hyperion includes
|
|
#include <hyperion/Hyperion.h>
|
|
|
|
// utils includes
|
|
#include <utils/Logger.h>
|
|
#include <utils/NetUtils.h>
|
|
|
|
// qt includes
|
|
#include <QTcpServer>
|
|
#include <QTcpSocket>
|
|
#include <QHostInfo>
|
|
#include <QNetworkInterface>
|
|
|
|
#include <flatbufserver/FlatBufferConnection.h>
|
|
|
|
MessageForwarder::MessageForwarder(Hyperion* hyperion)
|
|
: _hyperion(hyperion)
|
|
, _log(nullptr)
|
|
, _muxer(_hyperion->getMuxerInstance())
|
|
, _forwarder_enabled(true)
|
|
, _priority(140)
|
|
{
|
|
QString subComponent = hyperion->property("instance").toString();
|
|
_log= Logger::getInstance("NETFORWARDER", subComponent);
|
|
|
|
// get settings updates
|
|
connect(_hyperion, &Hyperion::settingsChanged, this, &MessageForwarder::handleSettingsUpdate);
|
|
|
|
// component changes
|
|
connect(_hyperion, &Hyperion::compStateChangeRequest, this, &MessageForwarder::handleCompStateChangeRequest);
|
|
|
|
// connect with Muxer visible priority changes
|
|
connect(_muxer, &PriorityMuxer::visiblePriorityChanged, this, &MessageForwarder::handlePriorityChanges);
|
|
|
|
// init
|
|
handleSettingsUpdate(settings::NETFORWARD, _hyperion->getSetting(settings::NETFORWARD));
|
|
}
|
|
|
|
MessageForwarder::~MessageForwarder()
|
|
{
|
|
while (!_forwardClients.isEmpty())
|
|
{
|
|
delete _forwardClients.takeFirst();
|
|
}
|
|
}
|
|
|
|
void MessageForwarder::handleSettingsUpdate(settings::type type, const QJsonDocument& config)
|
|
{
|
|
if (type == settings::NETFORWARD)
|
|
{
|
|
// clear the current targets
|
|
_jsonTargets.clear();
|
|
_flatbufferTargets.clear();
|
|
while (!_forwardClients.isEmpty())
|
|
{
|
|
delete _forwardClients.takeFirst();
|
|
}
|
|
|
|
// build new one
|
|
const QJsonObject& obj = config.object();
|
|
if (!obj["json"].isNull())
|
|
{
|
|
const QJsonArray& addr = obj["json"].toArray();
|
|
for (const auto& entry : addr)
|
|
{
|
|
addJsonTarget(entry.toObject());
|
|
}
|
|
}
|
|
|
|
if (!obj["flat"].isNull())
|
|
{
|
|
const QJsonArray& addr = obj["flat"].toArray();
|
|
for (const auto& entry : addr)
|
|
{
|
|
addFlatbufferTarget(entry.toObject());
|
|
}
|
|
}
|
|
|
|
bool isForwarderEnabledinSettings = obj["enable"].toBool(false);
|
|
|
|
if (!_jsonTargets.isEmpty() && isForwarderEnabledinSettings && _forwarder_enabled)
|
|
{
|
|
for (const auto& targetHost : qAsConst(_jsonTargets))
|
|
{
|
|
InfoIf(isForwarderEnabledinSettings, _log, "Forwarding now to JSON-target host: %s port: %u", QSTRING_CSTR(targetHost.host.toString()), targetHost.port);
|
|
}
|
|
|
|
connect(_hyperion, &Hyperion::forwardJsonMessage, this, &MessageForwarder::forwardJsonMessage, Qt::UniqueConnection);
|
|
}
|
|
else if (_jsonTargets.isEmpty() || !isForwarderEnabledinSettings || !_forwarder_enabled)
|
|
{
|
|
disconnect(_hyperion, &Hyperion::forwardJsonMessage, nullptr, nullptr);
|
|
}
|
|
|
|
if (!_flatbufferTargets.isEmpty() && isForwarderEnabledinSettings && _forwarder_enabled)
|
|
{
|
|
for (const auto& targetHost : qAsConst(_flatbufferTargets))
|
|
{
|
|
InfoIf(isForwarderEnabledinSettings, _log, "Forwarding now to Flatbuffer-target host: %s port: %u", QSTRING_CSTR(targetHost.host.toString()), targetHost.port);
|
|
}
|
|
}
|
|
else if (_flatbufferTargets.isEmpty() || !isForwarderEnabledinSettings || !_forwarder_enabled)
|
|
{
|
|
disconnect(_hyperion, &Hyperion::forwardSystemProtoMessage, nullptr, nullptr);
|
|
disconnect(_hyperion, &Hyperion::forwardV4lProtoMessage, nullptr, nullptr);
|
|
}
|
|
|
|
// update comp state
|
|
_hyperion->setNewComponentState(hyperion::COMP_FORWARDER, isForwarderEnabledinSettings);
|
|
}
|
|
}
|
|
|
|
void MessageForwarder::handleCompStateChangeRequest(hyperion::Components component, bool enable)
|
|
{
|
|
if (component == hyperion::COMP_FORWARDER && _forwarder_enabled != enable)
|
|
{
|
|
_forwarder_enabled = enable;
|
|
handleSettingsUpdate(settings::NETFORWARD, _hyperion->getSetting(settings::NETFORWARD));
|
|
Info(_log, "Forwarder change state to %s", (_forwarder_enabled ? "enabled" : "disabled"));
|
|
_hyperion->setNewComponentState(component, _forwarder_enabled);
|
|
}
|
|
}
|
|
|
|
void MessageForwarder::handlePriorityChanges(int priority)
|
|
{
|
|
const QJsonObject obj = _hyperion->getSetting(settings::NETFORWARD).object();
|
|
if (priority != 0 && _forwarder_enabled && obj["enable"].toBool())
|
|
{
|
|
hyperion::Components activeCompId = _hyperion->getPriorityInfo(priority).componentId;
|
|
if (activeCompId == hyperion::COMP_GRABBER || activeCompId == hyperion::COMP_V4L)
|
|
{
|
|
if (!obj["flat"].isNull())
|
|
{
|
|
const QJsonArray& addr = obj["flat"].toArray();
|
|
for (const auto& entry : addr)
|
|
{
|
|
addFlatbufferTarget(entry.toObject());
|
|
}
|
|
}
|
|
|
|
switch (activeCompId)
|
|
{
|
|
case hyperion::COMP_GRABBER:
|
|
{
|
|
disconnect(_hyperion, &Hyperion::forwardV4lProtoMessage, nullptr, nullptr);
|
|
connect(_hyperion, &Hyperion::forwardSystemProtoMessage, this, &MessageForwarder::forwardFlatbufferMessage, Qt::UniqueConnection);
|
|
}
|
|
break;
|
|
case hyperion::COMP_V4L:
|
|
{
|
|
disconnect(_hyperion, &Hyperion::forwardSystemProtoMessage, nullptr, nullptr);
|
|
connect(_hyperion, &Hyperion::forwardV4lProtoMessage, this, &MessageForwarder::forwardFlatbufferMessage, Qt::UniqueConnection);
|
|
}
|
|
break;
|
|
default:
|
|
{
|
|
disconnect(_hyperion, &Hyperion::forwardSystemProtoMessage, nullptr, nullptr);
|
|
disconnect(_hyperion, &Hyperion::forwardV4lProtoMessage, nullptr, nullptr);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
disconnect(_hyperion, &Hyperion::forwardSystemProtoMessage, nullptr, nullptr);
|
|
disconnect(_hyperion, &Hyperion::forwardV4lProtoMessage, nullptr, nullptr);
|
|
}
|
|
}
|
|
}
|
|
|
|
void MessageForwarder::addJsonTarget(const QJsonObject& targetConfig)
|
|
{
|
|
TargetHost targetHost;
|
|
|
|
QString config_host = targetConfig["host"].toString();
|
|
if (NetUtils::resolveHostAddress(_log, config_host, targetHost.host))
|
|
{
|
|
int config_port = targetConfig["port"].toInt();
|
|
if (NetUtils::isValidPort(_log, config_port, config_host))
|
|
{
|
|
targetHost.port = static_cast<quint16>(config_port);
|
|
|
|
// verify loop with JSON-server
|
|
const QJsonObject& obj = _hyperion->getSetting(settings::JSONSERVER).object();
|
|
if ((QNetworkInterface::allAddresses().indexOf(targetHost.host) != -1) && targetHost.port == static_cast<quint16>(obj["port"].toInt()))
|
|
{
|
|
Error(_log, "Loop between JSON-Server and Forwarder! Configuration for host: %s, port: %d is ignored.", QSTRING_CSTR(config_host), config_port);
|
|
}
|
|
else
|
|
{
|
|
if (_forwarder_enabled)
|
|
{
|
|
if (_jsonTargets.indexOf(targetHost) == -1)
|
|
{
|
|
Info(_log, "JSON-Forwarder settings: Adding target host: %s port: %u", QSTRING_CSTR(targetHost.host.toString()), targetHost.port);
|
|
_jsonTargets << targetHost;
|
|
}
|
|
else
|
|
{
|
|
Warning(_log, "JSON Forwarder settings: Duplicate target host configuration! Configuration for host: %s, port: %d is ignored.", QSTRING_CSTR(targetHost.host.toString()), targetHost.port);
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void MessageForwarder::addFlatbufferTarget(const QJsonObject& targetConfig)
|
|
{
|
|
TargetHost targetHost;
|
|
|
|
QString config_host = targetConfig["host"].toString();
|
|
if (NetUtils::resolveHostAddress(_log, config_host, targetHost.host))
|
|
{
|
|
int config_port = targetConfig["port"].toInt();
|
|
if (NetUtils::isValidPort(_log, config_port, config_host))
|
|
{
|
|
targetHost.port = static_cast<quint16>(config_port);
|
|
|
|
// verify loop with Flatbuffer-server
|
|
const QJsonObject& obj = _hyperion->getSetting(settings::FLATBUFSERVER).object();
|
|
if ((QNetworkInterface::allAddresses().indexOf(targetHost.host) != -1) && targetHost.port == static_cast<quint16>(obj["port"].toInt()))
|
|
{
|
|
Error(_log, "Loop between Flatbuffer-Server and Forwarder! Configuration for host: %s, port: %d is ignored.", QSTRING_CSTR(config_host), config_port);
|
|
}
|
|
else
|
|
{
|
|
if (_forwarder_enabled)
|
|
{
|
|
if (_flatbufferTargets.indexOf(targetHost) == -1)
|
|
{
|
|
Info(_log, "Flatbuffer-Forwarder settings: Adding target host: %s port: %u", QSTRING_CSTR(targetHost.host.toString()), targetHost.port);
|
|
_flatbufferTargets << targetHost;
|
|
|
|
FlatBufferConnection* flatbuf = new FlatBufferConnection("Forwarder", targetHost.host.toString(), _priority, false, targetHost.port);
|
|
_forwardClients << flatbuf;
|
|
}
|
|
else
|
|
{
|
|
Warning(_log, "Flatbuffer Forwarder settings: Duplicate target host configuration! Configuration for host: %s, port: %d is ignored.", QSTRING_CSTR(targetHost.host.toString()), targetHost.port);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void MessageForwarder::forwardJsonMessage(const QJsonObject& message)
|
|
{
|
|
if (_forwarder_enabled)
|
|
{
|
|
QTcpSocket client;
|
|
for (const auto& targetHost : qAsConst(_jsonTargets))
|
|
{
|
|
client.connectToHost(targetHost.host, targetHost.port);
|
|
if (client.waitForConnected(500))
|
|
{
|
|
sendJsonMessage(message, &client);
|
|
client.close();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void MessageForwarder::forwardFlatbufferMessage(const QString& /*name*/, const Image<ColorRgb>& image)
|
|
{
|
|
if (_forwarder_enabled)
|
|
{
|
|
for (int i = 0; i < _forwardClients.size(); i++)
|
|
{
|
|
_forwardClients.at(i)->setImage(image);
|
|
}
|
|
}
|
|
}
|
|
|
|
void MessageForwarder::sendJsonMessage(const QJsonObject& message, QTcpSocket* socket)
|
|
{
|
|
// for hyperion classic compatibility
|
|
QJsonObject jsonMessage = message;
|
|
if (jsonMessage.contains("tan") && jsonMessage["tan"].isNull())
|
|
{
|
|
jsonMessage["tan"] = 100;
|
|
}
|
|
|
|
// serialize message
|
|
QJsonDocument writer(jsonMessage);
|
|
QByteArray serializedMessage = writer.toJson(QJsonDocument::Compact) + "\n";
|
|
|
|
// write message
|
|
socket->write(serializedMessage);
|
|
if (!socket->waitForBytesWritten())
|
|
{
|
|
Debug(_log, "Error while writing data to host");
|
|
return;
|
|
}
|
|
|
|
// read reply data
|
|
QByteArray serializedReply;
|
|
while (!serializedReply.contains('\n'))
|
|
{
|
|
// receive reply
|
|
if (!socket->waitForReadyRead())
|
|
{
|
|
Debug(_log, "Error while writing data from host");
|
|
return;
|
|
}
|
|
|
|
serializedReply += socket->readAll();
|
|
}
|
|
|
|
// parse reply data
|
|
QJsonParseError error;
|
|
QJsonDocument::fromJson(serializedReply, &error);
|
|
|
|
if (error.error != QJsonParseError::NoError)
|
|
{
|
|
Error(_log, "Error while parsing reply: invalid json");
|
|
return;
|
|
}
|
|
}
|
|
|