2016-02-16 14:59:31 +01:00
|
|
|
// STL includes
|
2016-02-16 15:56:20 +01:00
|
|
|
#include <stdexcept>
|
2016-02-16 14:59:31 +01:00
|
|
|
|
2019-02-03 14:36:57 +01:00
|
|
|
// project includes
|
2016-02-15 18:25:18 +01:00
|
|
|
#include <hyperion/MessageForwarder.h>
|
|
|
|
|
2019-02-03 14:36:57 +01:00
|
|
|
// hyperion includes
|
2018-12-27 23:11:32 +01:00
|
|
|
#include <hyperion/Hyperion.h>
|
2016-02-15 18:25:18 +01:00
|
|
|
|
2019-02-03 14:36:57 +01:00
|
|
|
// utils includes
|
|
|
|
#include <utils/Logger.h>
|
|
|
|
|
|
|
|
// qt includes
|
|
|
|
#include <QTcpServer>
|
|
|
|
#include <QTcpSocket>
|
|
|
|
|
|
|
|
#include <flatbufserver/FlatBufferConnection.h>
|
|
|
|
|
|
|
|
MessageForwarder::MessageForwarder(Hyperion *hyperion)
|
2018-12-27 23:11:32 +01:00
|
|
|
: QObject()
|
|
|
|
, _hyperion(hyperion)
|
|
|
|
, _log(Logger::getInstance("NETFORWARDER"))
|
2019-02-03 14:36:57 +01:00
|
|
|
, _muxer(_hyperion->getMuxerInstance())
|
|
|
|
, _forwarder_enabled(true)
|
|
|
|
, _priority(140)
|
2016-02-15 18:25:18 +01:00
|
|
|
{
|
2018-12-27 23:11:32 +01:00
|
|
|
// get settings updates
|
|
|
|
connect(_hyperion, &Hyperion::settingsChanged, this, &MessageForwarder::handleSettingsUpdate);
|
2019-02-03 14:36:57 +01:00
|
|
|
|
|
|
|
// component changes
|
|
|
|
connect(_hyperion, &Hyperion::componentStateChanged, this, &MessageForwarder::componentStateChanged);
|
|
|
|
|
|
|
|
// connect with Muxer visible priority changes
|
|
|
|
connect(_muxer, &PriorityMuxer::visiblePriorityChanged, this, &MessageForwarder::handlePriorityChanges);
|
|
|
|
|
|
|
|
// init
|
|
|
|
handleSettingsUpdate(settings::NETFORWARD, _hyperion->getSetting(settings::NETFORWARD));
|
2016-02-15 18:25:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
MessageForwarder::~MessageForwarder()
|
|
|
|
{
|
2019-02-03 14:36:57 +01:00
|
|
|
while (!_forwardClients.isEmpty())
|
|
|
|
delete _forwardClients.takeFirst();
|
2016-02-15 18:25:18 +01:00
|
|
|
}
|
|
|
|
|
2019-02-03 14:36:57 +01:00
|
|
|
void MessageForwarder::handleSettingsUpdate(const settings::type &type, const QJsonDocument &config)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
|
|
|
if(type == settings::NETFORWARD)
|
|
|
|
{
|
|
|
|
// clear the current targets
|
|
|
|
_jsonSlaves.clear();
|
|
|
|
_protoSlaves.clear();
|
2019-02-03 14:36:57 +01:00
|
|
|
while (!_forwardClients.isEmpty())
|
|
|
|
delete _forwardClients.takeFirst();
|
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
// build new one
|
|
|
|
const QJsonObject &obj = config.object();
|
|
|
|
if ( !obj["json"].isNull() )
|
|
|
|
{
|
|
|
|
const QJsonArray & addr = obj["json"].toArray();
|
|
|
|
for (const auto& entry : addr)
|
|
|
|
{
|
|
|
|
addJsonSlave(entry.toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !obj["proto"].isNull() )
|
|
|
|
{
|
|
|
|
const QJsonArray & addr = obj["proto"].toArray();
|
|
|
|
for (const auto& entry : addr)
|
|
|
|
{
|
|
|
|
addProtoSlave(entry.toString());
|
|
|
|
}
|
|
|
|
}
|
2019-02-03 14:36:57 +01:00
|
|
|
|
|
|
|
if (!_jsonSlaves.isEmpty() && obj["enable"].toBool() && _forwarder_enabled)
|
|
|
|
{
|
|
|
|
InfoIf(obj["enable"].toBool(true), _log, "Forward now to json targets '%s'", QSTRING_CSTR(_jsonSlaves.join(", ")));
|
|
|
|
connect(_hyperion, &Hyperion::forwardJsonMessage, this, &MessageForwarder::forwardJsonMessage, Qt::UniqueConnection);
|
|
|
|
} else if (_jsonSlaves.isEmpty() || ! obj["enable"].toBool() || !_forwarder_enabled)
|
|
|
|
disconnect(_hyperion, &Hyperion::forwardJsonMessage, 0, 0);
|
|
|
|
|
|
|
|
if (!_protoSlaves.isEmpty() && obj["enable"].toBool() && _forwarder_enabled)
|
|
|
|
{
|
|
|
|
InfoIf(obj["enable"].toBool(true), _log, "Forward now to proto targets '%s'", QSTRING_CSTR(_protoSlaves.join(", ")));
|
|
|
|
connect(_hyperion, &Hyperion::forwardProtoMessage, this, &MessageForwarder::forwardProtoMessage, Qt::UniqueConnection);
|
|
|
|
} else if ( _protoSlaves.isEmpty() || ! obj["enable"].toBool() || !_forwarder_enabled)
|
|
|
|
disconnect(_hyperion, &Hyperion::forwardProtoMessage, 0, 0);
|
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
// update comp state
|
|
|
|
_hyperion->getComponentRegister().componentStateChanged(hyperion::COMP_FORWARDER, obj["enable"].toBool(true));
|
|
|
|
}
|
|
|
|
}
|
2016-02-15 18:25:18 +01:00
|
|
|
|
2019-02-03 14:36:57 +01:00
|
|
|
void MessageForwarder::componentStateChanged(const hyperion::Components component, bool enable)
|
|
|
|
{
|
|
|
|
if (component == hyperion::COMP_FORWARDER && _forwarder_enabled != enable)
|
|
|
|
{
|
|
|
|
_forwarder_enabled = enable;
|
|
|
|
handleSettingsUpdate(settings::NETFORWARD, _hyperion->getSetting(settings::NETFORWARD));
|
2019-02-05 19:55:48 +01:00
|
|
|
Info(_log, "Forwarder change state to %s", (_forwarder_enabled ? "enabled" : "disabled"));
|
2019-02-03 14:36:57 +01:00
|
|
|
_hyperion->getComponentRegister().componentStateChanged(component, _forwarder_enabled);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MessageForwarder::handlePriorityChanges(const quint8 &priority)
|
|
|
|
{
|
|
|
|
const QJsonObject obj = _hyperion->getSetting(settings::NETFORWARD).object();
|
|
|
|
if (priority != 0 && _forwarder_enabled && obj["enable"].toBool())
|
|
|
|
{
|
|
|
|
_protoSlaves.clear();
|
|
|
|
while (!_forwardClients.isEmpty())
|
|
|
|
delete _forwardClients.takeFirst();
|
|
|
|
|
|
|
|
hyperion::Components activePrio = _hyperion->getPriorityInfo(priority).componentId;
|
|
|
|
if (activePrio == hyperion::COMP_GRABBER || activePrio == hyperion::COMP_V4L)
|
|
|
|
{
|
|
|
|
if ( !obj["proto"].isNull() )
|
|
|
|
{
|
|
|
|
const QJsonArray & addr = obj["proto"].toArray();
|
|
|
|
for (const auto& entry : addr)
|
|
|
|
{
|
|
|
|
addProtoSlave(entry.toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
connect(_hyperion, &Hyperion::forwardProtoMessage, this, &MessageForwarder::forwardProtoMessage, Qt::UniqueConnection);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
disconnect(_hyperion, &Hyperion::forwardProtoMessage, 0, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-04 22:17:42 +01:00
|
|
|
void MessageForwarder::addJsonSlave(QString slave)
|
2016-02-15 18:25:18 +01:00
|
|
|
{
|
2017-03-04 22:17:42 +01:00
|
|
|
QStringList parts = slave.split(":");
|
2016-02-15 20:53:03 +01:00
|
|
|
if (parts.size() != 2)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
|
|
|
Error(_log, "Unable to parse address (%s)",QSTRING_CSTR(slave));
|
|
|
|
return;
|
|
|
|
}
|
2016-02-15 20:53:03 +01:00
|
|
|
|
|
|
|
bool ok;
|
2018-12-27 23:11:32 +01:00
|
|
|
parts[1].toUShort(&ok);
|
2016-02-15 20:53:03 +01:00
|
|
|
if (!ok)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
|
|
|
Error(_log, "Unable to parse port number (%s)",QSTRING_CSTR(parts[1]));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// verify loop with jsonserver
|
2019-02-03 14:36:57 +01:00
|
|
|
const QJsonObject &obj = _hyperion->getSetting(settings::JSONSERVER).object();
|
2018-12-27 23:11:32 +01:00
|
|
|
if(QHostAddress(parts[0]) == QHostAddress::LocalHost && parts[1].toInt() == obj["port"].toInt())
|
|
|
|
{
|
|
|
|
Error(_log, "Loop between JsonServer and Forwarder! (%s)",QSTRING_CSTR(slave));
|
|
|
|
return;
|
|
|
|
}
|
2016-02-15 20:53:03 +01:00
|
|
|
|
2019-02-03 14:36:57 +01:00
|
|
|
if (_forwarder_enabled)
|
|
|
|
_jsonSlaves << slave;
|
2016-02-15 18:25:18 +01:00
|
|
|
}
|
|
|
|
|
2017-03-04 22:17:42 +01:00
|
|
|
void MessageForwarder::addProtoSlave(QString slave)
|
2016-02-15 18:25:18 +01:00
|
|
|
{
|
2018-12-27 23:11:32 +01:00
|
|
|
QStringList parts = slave.split(":");
|
|
|
|
if (parts.size() != 2)
|
|
|
|
{
|
|
|
|
Error(_log, "Unable to parse address (%s)",QSTRING_CSTR(slave));
|
|
|
|
return;
|
|
|
|
}
|
2016-02-15 18:25:18 +01:00
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
bool ok;
|
|
|
|
parts[1].toUShort(&ok);
|
|
|
|
if (!ok)
|
|
|
|
{
|
|
|
|
Error(_log, "Unable to parse port number (%s)",QSTRING_CSTR(parts[1]));
|
|
|
|
return;
|
|
|
|
}
|
2016-02-15 20:53:03 +01:00
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
// verify loop with protoserver
|
2019-02-03 14:36:57 +01:00
|
|
|
const QJsonObject &obj = _hyperion->getSetting(settings::FLATBUFSERVER).object();
|
2018-12-27 23:11:32 +01:00
|
|
|
if(QHostAddress(parts[0]) == QHostAddress::LocalHost && parts[1].toInt() == obj["port"].toInt())
|
|
|
|
{
|
|
|
|
Error(_log, "Loop between ProtoServer and Forwarder! (%s)",QSTRING_CSTR(slave));
|
|
|
|
return;
|
|
|
|
}
|
2019-02-03 14:36:57 +01:00
|
|
|
|
|
|
|
if (_forwarder_enabled)
|
|
|
|
{
|
|
|
|
_protoSlaves << slave;
|
2019-02-05 19:55:48 +01:00
|
|
|
FlatBufferConnection* flatbuf = new FlatBufferConnection("Forwarder", slave.toLocal8Bit().constData(), _priority, false);
|
2019-02-03 14:36:57 +01:00
|
|
|
_forwardClients << flatbuf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MessageForwarder::forwardJsonMessage(const QJsonObject &message)
|
|
|
|
{
|
|
|
|
if (_forwarder_enabled)
|
|
|
|
{
|
|
|
|
QTcpSocket client;
|
|
|
|
for (int i=0; i<_jsonSlaves.size(); i++)
|
|
|
|
{
|
|
|
|
QStringList parts = _jsonSlaves.at(i).split(":");
|
|
|
|
client.connectToHost(QHostAddress(parts[0]), parts[1].toUShort());
|
|
|
|
if ( client.waitForConnected(500) )
|
|
|
|
{
|
|
|
|
sendJsonMessage(message,&client);
|
|
|
|
client.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-02-15 20:53:03 +01:00
|
|
|
}
|
2016-02-19 13:31:08 +01:00
|
|
|
|
2019-02-03 14:36:57 +01:00
|
|
|
void MessageForwarder::forwardProtoMessage(const Image<ColorRgb> &image)
|
2016-02-19 13:31:08 +01:00
|
|
|
{
|
2019-02-03 14:36:57 +01:00
|
|
|
if (_forwarder_enabled)
|
|
|
|
{
|
|
|
|
for (int i=0; i < _forwardClients.size(); i++)
|
|
|
|
_forwardClients.at(i)->setImage(image);
|
|
|
|
}
|
2016-02-19 13:31:08 +01:00
|
|
|
}
|
2016-09-07 20:10:37 +02:00
|
|
|
|
2019-02-03 14:36:57 +01:00
|
|
|
void MessageForwarder::sendJsonMessage(const QJsonObject &message, QTcpSocket *socket)
|
2016-09-07 20:10:37 +02:00
|
|
|
{
|
2019-02-03 14:36:57 +01:00
|
|
|
// 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 reply = QJsonDocument::fromJson(serializedReply ,&error);
|
|
|
|
|
|
|
|
if (error.error != QJsonParseError::NoError)
|
|
|
|
{
|
|
|
|
Error(_log, "Error while parsing reply: invalid json");
|
|
|
|
return;
|
|
|
|
}
|
2016-09-07 20:10:37 +02:00
|
|
|
}
|