164 lines
7.9 KiB
C++
164 lines
7.9 KiB
C++
// SPDX-License-Identifier: LGPL-3.0-or-later
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
*
|
|
* nymea-remoteproxy
|
|
* Tunnel proxy server for the nymea remote access
|
|
*
|
|
* Copyright (C) 2013 - 2024, nymea GmbH
|
|
* Copyright (C) 2024 - 2025, chargebyte austria GmbH
|
|
*
|
|
* This file is part of nymea-remoteproxy.
|
|
*
|
|
* nymea-remoteproxy is free software: you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public License
|
|
* as published by the Free Software Foundation, either version 3
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* nymea-remoteproxy is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with nymea-remoteproxy. If not, see <https://www.gnu.org/licenses/>.
|
|
*
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
#include "tunnelproxyhandler.h"
|
|
|
|
#include "engine.h"
|
|
#include "jsontypes.h"
|
|
#include "loggingcategories.h"
|
|
|
|
#include "tunnelproxy/tunnelproxyserver.h"
|
|
|
|
namespace remoteproxy {
|
|
|
|
TunnelProxyHandler::TunnelProxyHandler(QObject *parent) : JsonHandler(parent)
|
|
{
|
|
// Methods
|
|
QVariantMap params; QVariantMap returns;
|
|
|
|
// Server
|
|
params.clear(); returns.clear();
|
|
setDescription("RegisterServer", "Register a new TunnelProxy server on this instance. Multiple TunnelProxy clients can be connected to the registered server on success.");
|
|
params.insert("serverName", JsonTypes::basicTypeToString(JsonTypes::String));
|
|
params.insert("serverUuid", JsonTypes::basicTypeToString(JsonTypes::Uuid));
|
|
setParams("RegisterServer", params);
|
|
returns.insert("tunnelProxyError", JsonTypes::tunnelProxyErrorRef());
|
|
returns.insert("slipEnabled", JsonTypes::basicTypeToString(JsonTypes::Bool));
|
|
setReturns("RegisterServer", returns);
|
|
|
|
params.clear(); returns.clear();
|
|
setDescription("DisconnectClient", "A registered server can ask the remote proxy connection to disconnect a client for whatever reason.");
|
|
params.insert("socketAddress", JsonTypes::basicTypeToString(JsonTypes::UInt));
|
|
setParams("DisconnectClient", params);
|
|
returns.insert("tunnelProxyError", JsonTypes::tunnelProxyErrorRef());
|
|
setReturns("DisconnectClient", returns);
|
|
|
|
params.clear(); returns.clear();
|
|
setDescription("Ping", "In order to keep a connection alive when no client is connected, this Ping method can be used. The sent timestamp will be returned as sent in the response for speed measuements on the client side.");
|
|
params.insert("timestamp", JsonTypes::basicTypeToString(JsonTypes::UInt));
|
|
setParams("Ping", params);
|
|
returns.insert("timestamp", JsonTypes::basicTypeToString(JsonTypes::UInt));
|
|
setReturns("Ping", returns);
|
|
|
|
// Client
|
|
params.clear(); returns.clear();
|
|
setDescription("RegisterClient", "Register a new TunnelProxy client on TunnelProxy server with the given serverUuid. "
|
|
"On success, the remote connection has been accepted and any further data will come from the connected server.");
|
|
params.insert("clientName", JsonTypes::basicTypeToString(JsonTypes::String));
|
|
params.insert("clientUuid", JsonTypes::basicTypeToString(JsonTypes::Uuid));
|
|
params.insert("serverUuid", JsonTypes::basicTypeToString(JsonTypes::Uuid));
|
|
setParams("RegisterClient", params);
|
|
returns.insert("tunnelProxyError", JsonTypes::tunnelProxyErrorRef());
|
|
setReturns("RegisterClient", returns);
|
|
|
|
// Notifications
|
|
|
|
// Server
|
|
params.clear(); returns.clear();
|
|
setDescription("ClientConnected", "Emitted whenever a new client has been connected to a registered server. "
|
|
"Only tunnel proxy clients registered as server will receive this notification. The socket address will be used for framing.");
|
|
params.insert("clientName", JsonTypes::basicTypeToString(JsonTypes::String));
|
|
params.insert("clientUuid", JsonTypes::basicTypeToString(JsonTypes::String));
|
|
params.insert("clientPeerAddress", JsonTypes::basicTypeToString(JsonTypes::String));
|
|
params.insert("socketAddress", JsonTypes::basicTypeToString(JsonTypes::UInt));
|
|
setParams("ClientConnected", params);
|
|
|
|
params.clear(); returns.clear();
|
|
setDescription("ClientDisconnected", "Emitted whenever a new client has been connected to a registered server. "
|
|
"Only tunnel proxy clients registered as server will receive this notification.");
|
|
params.insert("socketAddress", JsonTypes::basicTypeToString(JsonTypes::UInt));
|
|
setParams("ClientDisconnected", params);
|
|
}
|
|
|
|
QString TunnelProxyHandler::name() const
|
|
{
|
|
return "TunnelProxy";
|
|
}
|
|
|
|
JsonReply *TunnelProxyHandler::RegisterServer(const QVariantMap ¶ms, TransportClient *transportClient)
|
|
{
|
|
qCDebug(dcJsonRpc()) << name() << "register server" << params << transportClient;
|
|
QUuid serverUuid = params.value("serverUuid").toUuid();
|
|
TunnelProxyServer::TunnelProxyError error = TunnelProxyServer::TunnelProxyErrorNoError;
|
|
if (serverUuid.isNull()) {
|
|
qCWarning(dcJsonRpc()) << "Invalid uuid received" << params.value("serverUuid").toString() << serverUuid;
|
|
error = TunnelProxyServer::TunnelProxyErrorInvalidUuid;
|
|
} else {
|
|
QString serverName = params.value("serverName").toString();
|
|
error = Engine::instance()->tunnelProxyServer()->registerServer(transportClient->clientId(), serverUuid, serverName);
|
|
}
|
|
|
|
QVariantMap response;
|
|
response.insert("tunnelProxyError", JsonTypes::tunnelProxyErrorToString(error));
|
|
response.insert("slipEnabled", error == TunnelProxyServer::TunnelProxyErrorNoError);
|
|
return createReply("RegisterServer", response);
|
|
}
|
|
|
|
JsonReply *TunnelProxyHandler::DisconnectClient(const QVariantMap ¶ms, TransportClient *transportClient)
|
|
{
|
|
qCDebug(dcJsonRpc()) << name() << "disconnect client requested" << params << transportClient;
|
|
quint16 socketAddress = static_cast<quint16>(params.value("socketAddress").toUInt());
|
|
TunnelProxyServer::TunnelProxyError error = Engine::instance()->tunnelProxyServer()->disconnectClient(transportClient->clientId(), socketAddress);
|
|
|
|
QVariantMap response;
|
|
response.insert("tunnelProxyError", JsonTypes::tunnelProxyErrorToString(error));
|
|
return createReply("DisconnectClient", response);
|
|
}
|
|
|
|
JsonReply *TunnelProxyHandler::Ping(const QVariantMap ¶ms, TransportClient *transportClient)
|
|
{
|
|
qCDebug(dcJsonRpc()) << name() << "ping received" << params << transportClient;
|
|
|
|
QVariantMap response;
|
|
response.insert("timestamp", params.value("timestamp"));
|
|
return createReply("Ping", response);
|
|
}
|
|
|
|
JsonReply *TunnelProxyHandler::RegisterClient(const QVariantMap ¶ms, TransportClient *transportClient)
|
|
{
|
|
qCDebug(dcJsonRpc()) << name() << "register client" << params << transportClient;
|
|
QString clientName = params.value("clientName").toString();
|
|
QUuid clientUuid = params.value("clientUuid").toUuid();
|
|
QUuid serverUuid = params.value("serverUuid").toUuid();
|
|
TunnelProxyServer::TunnelProxyError error = TunnelProxyServer::TunnelProxyErrorNoError;
|
|
if (serverUuid.isNull()) {
|
|
qCWarning(dcJsonRpc()) << "Invalid server uuid received" << params.value("serverUuid").toString() << serverUuid;
|
|
error = TunnelProxyServer::TunnelProxyErrorInvalidUuid;
|
|
} else if (clientUuid.isNull()) {
|
|
qCWarning(dcJsonRpc()) << "Invalid client uuid received" << params.value("clientUuid").toString() << clientUuid;
|
|
error = TunnelProxyServer::TunnelProxyErrorInvalidUuid;
|
|
} else {
|
|
error = Engine::instance()->tunnelProxyServer()->registerClient(transportClient->clientId(), clientUuid, clientName, serverUuid);
|
|
}
|
|
|
|
QVariantMap response;
|
|
response.insert("tunnelProxyError", JsonTypes::tunnelProxyErrorToString(error));
|
|
return createReply("RegisterClient", response);
|
|
}
|
|
|
|
}
|