2018-12-27 23:11:32 +01:00
|
|
|
#include "webserver/WebServer.h"
|
2020-09-07 21:34:14 +02:00
|
|
|
#include "HyperionConfig.h"
|
2018-12-27 23:11:32 +01:00
|
|
|
#include "StaticFileServing.h"
|
|
|
|
#include "QtHttpServer.h"
|
2018-12-30 22:07:53 +01:00
|
|
|
|
2018-12-28 18:12:45 +01:00
|
|
|
#include <QFileInfo>
|
|
|
|
#include <QJsonObject>
|
2018-12-27 23:11:32 +01:00
|
|
|
|
2018-12-30 22:07:53 +01:00
|
|
|
// bonjour
|
2020-05-12 19:51:19 +02:00
|
|
|
#ifdef ENABLE_AVAHI
|
2018-12-27 23:11:32 +01:00
|
|
|
#include <bonjour/bonjourserviceregister.h>
|
2020-05-12 19:51:19 +02:00
|
|
|
#endif
|
2018-12-30 22:07:53 +01:00
|
|
|
// netUtil
|
2018-12-28 18:12:45 +01:00
|
|
|
#include <utils/NetUtils.h>
|
2018-12-27 23:11:32 +01:00
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
WebServer::WebServer(const QJsonDocument& config, bool useSsl, QObject * parent)
|
2019-07-12 16:54:26 +02:00
|
|
|
: QObject(parent)
|
2018-12-30 22:07:53 +01:00
|
|
|
, _config(config)
|
2019-08-21 16:09:28 +02:00
|
|
|
, _useSsl(useSsl)
|
2018-12-27 23:11:32 +01:00
|
|
|
, _log(Logger::getInstance("WEBSERVER"))
|
2018-12-30 22:07:53 +01:00
|
|
|
, _server()
|
|
|
|
{
|
2019-07-12 16:54:26 +02:00
|
|
|
|
2018-12-30 22:07:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
WebServer::~WebServer()
|
|
|
|
{
|
|
|
|
stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebServer::initServer()
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
2020-03-11 22:18:08 +01:00
|
|
|
Debug(_log, "Initialize Webserver");
|
2018-12-30 22:07:53 +01:00
|
|
|
_server = new QtHttpServer (this);
|
2018-12-27 23:11:32 +01:00
|
|
|
_server->setServerName (QStringLiteral ("Hyperion Webserver"));
|
|
|
|
|
2019-08-21 16:09:28 +02:00
|
|
|
if(_useSsl)
|
|
|
|
{
|
|
|
|
_server->setUseSecure();
|
|
|
|
WEBSERVER_DEFAULT_PORT = 8092;
|
|
|
|
}
|
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
connect (_server, &QtHttpServer::started, this, &WebServer::onServerStarted);
|
|
|
|
connect (_server, &QtHttpServer::stopped, this, &WebServer::onServerStopped);
|
|
|
|
connect (_server, &QtHttpServer::error, this, &WebServer::onServerError);
|
|
|
|
|
|
|
|
// create StaticFileServing
|
2018-12-28 18:12:45 +01:00
|
|
|
_staticFileServing = new StaticFileServing (this);
|
2018-12-27 23:11:32 +01:00
|
|
|
connect(_server, &QtHttpServer::requestNeedsReply, _staticFileServing, &StaticFileServing::onRequestNeedsReply);
|
|
|
|
|
|
|
|
// init
|
2018-12-30 22:07:53 +01:00
|
|
|
handleSettingsUpdate(settings::WEBSERVER, _config);
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebServer::onServerStarted (quint16 port)
|
|
|
|
{
|
2020-07-12 09:19:59 +02:00
|
|
|
_inited = true;
|
2018-12-30 22:07:53 +01:00
|
|
|
|
2021-11-17 20:30:43 +00:00
|
|
|
Info(_log, "'%s' started on port %d",_server->getServerName().toStdString().c_str(), port);
|
|
|
|
|
2020-05-12 19:51:19 +02:00
|
|
|
#ifdef ENABLE_AVAHI
|
2018-12-28 18:12:45 +01:00
|
|
|
if(_serviceRegister == nullptr)
|
|
|
|
{
|
|
|
|
_serviceRegister = new BonjourServiceRegister(this);
|
|
|
|
_serviceRegister->registerService("_hyperiond-http._tcp", port);
|
|
|
|
}
|
|
|
|
else if( _serviceRegister->getPort() != port)
|
|
|
|
{
|
|
|
|
delete _serviceRegister;
|
|
|
|
_serviceRegister = new BonjourServiceRegister(this);
|
|
|
|
_serviceRegister->registerService("_hyperiond-http._tcp", port);
|
|
|
|
}
|
2020-05-12 19:51:19 +02:00
|
|
|
#endif
|
2018-12-30 22:07:53 +01:00
|
|
|
emit stateChange(true);
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
|
|
|
|
2019-07-21 15:03:50 +02:00
|
|
|
void WebServer::onServerStopped ()
|
|
|
|
{
|
2018-12-27 23:11:32 +01:00
|
|
|
Info(_log, "Stopped %s", _server->getServerName().toStdString().c_str());
|
2018-12-30 22:07:53 +01:00
|
|
|
emit stateChange(false);
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebServer::onServerError (QString msg)
|
|
|
|
{
|
|
|
|
Error(_log, "%s", msg.toStdString().c_str());
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void WebServer::handleSettingsUpdate(settings::type type, const QJsonDocument& config)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
|
|
|
if(type == settings::WEBSERVER)
|
|
|
|
{
|
2020-03-11 22:18:08 +01:00
|
|
|
Debug(_log, "Apply Webserver settings");
|
2018-12-27 23:11:32 +01:00
|
|
|
const QJsonObject& obj = config.object();
|
|
|
|
|
|
|
|
_baseUrl = obj["document_root"].toString(WEBSERVER_DEFAULT_PATH);
|
|
|
|
|
2018-12-30 22:07:53 +01:00
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
if ( (_baseUrl != ":/webconfig") && !_baseUrl.trimmed().isEmpty())
|
|
|
|
{
|
|
|
|
QFileInfo info(_baseUrl);
|
|
|
|
if (!info.exists() || !info.isDir())
|
|
|
|
{
|
|
|
|
Error(_log, "document_root '%s' is invalid", _baseUrl.toUtf8().constData());
|
|
|
|
_baseUrl = WEBSERVER_DEFAULT_PATH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
_baseUrl = WEBSERVER_DEFAULT_PATH;
|
|
|
|
|
|
|
|
Debug(_log, "Set document root to: %s", _baseUrl.toUtf8().constData());
|
|
|
|
_staticFileServing->setBaseUrl(_baseUrl);
|
|
|
|
|
2019-08-21 16:09:28 +02:00
|
|
|
// ssl different port
|
|
|
|
quint16 newPort = _useSsl ? obj["sslPort"].toInt(WEBSERVER_DEFAULT_PORT) : obj["port"].toInt(WEBSERVER_DEFAULT_PORT);
|
|
|
|
if(_port != newPort)
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
2019-08-21 16:09:28 +02:00
|
|
|
_port = newPort;
|
2018-12-27 23:11:32 +01:00
|
|
|
stop();
|
|
|
|
}
|
2018-12-28 18:12:45 +01:00
|
|
|
|
|
|
|
// eval if the port is available, will be incremented if not
|
2018-12-30 22:07:53 +01:00
|
|
|
if(!_server->isListening())
|
|
|
|
NetUtils::portAvailable(_port, _log);
|
|
|
|
|
2019-08-21 16:09:28 +02:00
|
|
|
// on ssl we want .key .cert and probably key password
|
|
|
|
if(_useSsl)
|
|
|
|
{
|
|
|
|
QString keyPath = obj["keyPath"].toString(WEBSERVER_DEFAULT_KEY_PATH);
|
|
|
|
QString crtPath = obj["crtPath"].toString(WEBSERVER_DEFAULT_CRT_PATH);
|
|
|
|
|
|
|
|
QSslKey currKey = _server->getPrivateKey();
|
|
|
|
QList<QSslCertificate> currCerts = _server->getCertificates();
|
|
|
|
|
|
|
|
// check keyPath
|
|
|
|
if ( (keyPath != WEBSERVER_DEFAULT_KEY_PATH) && !keyPath.trimmed().isEmpty())
|
|
|
|
{
|
|
|
|
QFileInfo kinfo(keyPath);
|
|
|
|
if (!kinfo.exists())
|
|
|
|
{
|
|
|
|
Error(_log, "No SSL key found at '%s' falling back to internal", keyPath.toUtf8().constData());
|
|
|
|
keyPath = WEBSERVER_DEFAULT_KEY_PATH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
keyPath = WEBSERVER_DEFAULT_KEY_PATH;
|
|
|
|
|
|
|
|
// check crtPath
|
|
|
|
if ( (crtPath != WEBSERVER_DEFAULT_CRT_PATH) && !crtPath.trimmed().isEmpty())
|
|
|
|
{
|
|
|
|
QFileInfo cinfo(crtPath);
|
|
|
|
if (!cinfo.exists())
|
|
|
|
{
|
|
|
|
Error(_log, "No SSL certificate found at '%s' falling back to internal", crtPath.toUtf8().constData());
|
|
|
|
crtPath = WEBSERVER_DEFAULT_CRT_PATH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
crtPath = WEBSERVER_DEFAULT_CRT_PATH;
|
|
|
|
|
|
|
|
// load and verify crt
|
|
|
|
QFile cfile(crtPath);
|
|
|
|
cfile.open(QIODevice::ReadOnly);
|
|
|
|
QList<QSslCertificate> validList;
|
|
|
|
QList<QSslCertificate> cList = QSslCertificate::fromDevice(&cfile, QSsl::Pem);
|
|
|
|
cfile.close();
|
|
|
|
|
|
|
|
// Filter for valid certs
|
|
|
|
for(const auto & entry : cList){
|
|
|
|
if(!entry.isNull() && QDateTime::currentDateTime().daysTo(entry.expiryDate()) > 0)
|
|
|
|
validList.append(entry);
|
|
|
|
else
|
|
|
|
Error(_log, "The provided SSL certificate is invalid/not supported/reached expiry date ('%s')", crtPath.toUtf8().constData());
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!validList.isEmpty()){
|
|
|
|
Debug(_log,"Setup SSL certificate");
|
|
|
|
_server->setCertificates(validList);
|
|
|
|
} else {
|
|
|
|
Error(_log, "No valid SSL certificate has been found ('%s')", crtPath.toUtf8().constData());
|
|
|
|
}
|
|
|
|
|
|
|
|
// load and verify key
|
|
|
|
QFile kfile(keyPath);
|
|
|
|
kfile.open(QIODevice::ReadOnly);
|
|
|
|
// The key should be RSA enrcrypted and PEM format, optional the passPhrase
|
|
|
|
QSslKey key(&kfile, QSsl::Rsa, QSsl::Pem, QSsl::PrivateKey, obj["keyPassPhrase"].toString().toUtf8());
|
|
|
|
kfile.close();
|
|
|
|
|
|
|
|
if(key.isNull()){
|
|
|
|
Error(_log, "The provided SSL key is invalid or not supported use RSA encrypt and PEM format ('%s')", keyPath.toUtf8().constData());
|
|
|
|
} else {
|
|
|
|
Debug(_log,"Setup private SSL key");
|
|
|
|
_server->setPrivateKey(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-28 18:12:45 +01:00
|
|
|
start();
|
2018-12-30 22:07:53 +01:00
|
|
|
emit portChanged(_port);
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebServer::start()
|
|
|
|
{
|
|
|
|
_server->start(_port);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WebServer::stop()
|
|
|
|
{
|
|
|
|
_server->stop();
|
|
|
|
}
|
2018-12-30 22:07:53 +01:00
|
|
|
|
|
|
|
void WebServer::setSSDPDescription(const QString & desc)
|
|
|
|
{
|
|
|
|
_staticFileServing->setSSDPDescription(desc);
|
|
|
|
}
|