forked from qt-creator/qt-creator
The heavy lifting was done by clazy. Change-Id: If3332a2b4a6d011d2cb74996f5dd750452093f31 Reviewed-by: Christian Stenger <christian.stenger@qt.io>
165 lines
5.7 KiB
C++
165 lines
5.7 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "directtunnel.h"
|
|
|
|
#include <ssh/sshconnection.h>
|
|
#include <ssh/sshdirecttcpiptunnel.h>
|
|
|
|
#include <QCoreApplication>
|
|
#include <QTcpServer>
|
|
#include <QTcpSocket>
|
|
#include <QTimer>
|
|
|
|
#include <cstdlib>
|
|
#include <iostream>
|
|
|
|
const QByteArray ServerDataPrefix("Received the following data: ");
|
|
const QByteArray TestData("Urgsblubb?");
|
|
|
|
using namespace QSsh;
|
|
|
|
DirectTunnel::DirectTunnel(const SshConnectionParameters ¶meters, QObject *parent)
|
|
: QObject(parent),
|
|
m_connection(new SshConnection(parameters, this)),
|
|
m_targetServer(new QTcpServer(this)),
|
|
m_expectingChannelClose(false)
|
|
{
|
|
connect(m_connection, &SshConnection::connected, this, &DirectTunnel::handleConnected);
|
|
connect(m_connection, &SshConnection::error, this, &DirectTunnel::handleConnectionError);
|
|
}
|
|
|
|
DirectTunnel::~DirectTunnel()
|
|
{
|
|
}
|
|
|
|
void DirectTunnel::run()
|
|
{
|
|
std::cout << "Connecting to SSH server..." << std::endl;
|
|
m_connection->connectToHost();
|
|
}
|
|
|
|
void DirectTunnel::handleConnectionError()
|
|
{
|
|
std::cerr << "SSH connection error: " << qPrintable(m_connection->errorString()) << std::endl;
|
|
emit finished(EXIT_FAILURE);
|
|
}
|
|
|
|
void DirectTunnel::handleConnected()
|
|
{
|
|
std::cout << "Opening server side..." << std::endl;
|
|
if (!m_targetServer->listen(QHostAddress::LocalHost)) {
|
|
std::cerr << "Error opening port: "
|
|
<< m_targetServer->errorString().toLocal8Bit().constData() << std::endl;
|
|
emit finished(EXIT_FAILURE);
|
|
return;
|
|
}
|
|
m_targetPort = m_targetServer->serverPort();
|
|
connect(m_targetServer, &QTcpServer::newConnection, this, &DirectTunnel::handleNewConnection);
|
|
|
|
m_tunnel = m_connection->createDirectTunnel(QLatin1String("localhost"), 1024, // made-up values
|
|
QLatin1String("localhost"), m_targetPort);
|
|
connect(m_tunnel.data(), &SshDirectTcpIpTunnel::initialized,
|
|
this, &DirectTunnel::handleInitialized);
|
|
connect(m_tunnel.data(), &SshDirectTcpIpTunnel::error,
|
|
this, &DirectTunnel::handleTunnelError);
|
|
connect(m_tunnel.data(), &QIODevice::readyRead, this, &DirectTunnel::handleServerData);
|
|
connect(m_tunnel.data(), &QIODevice::aboutToClose, this, &DirectTunnel::handleTunnelClosed);
|
|
|
|
std::cout << "Initializing tunnel..." << std::endl;
|
|
m_tunnel->initialize();
|
|
}
|
|
|
|
void DirectTunnel::handleInitialized()
|
|
{
|
|
std::cout << "Writing data into the tunnel..." << std::endl;
|
|
m_tunnel->write(TestData);
|
|
QTimer * const timeoutTimer = new QTimer(this);
|
|
connect(timeoutTimer, &QTimer::timeout, this, &DirectTunnel::handleTimeout);
|
|
timeoutTimer->start(10000);
|
|
}
|
|
|
|
void DirectTunnel::handleServerData()
|
|
{
|
|
m_dataReceivedFromServer += m_tunnel->readAll();
|
|
if (m_dataReceivedFromServer == ServerDataPrefix + TestData) {
|
|
std::cout << "Data exchange successful. Closing server socket..." << std::endl;
|
|
m_expectingChannelClose = true;
|
|
m_targetSocket->close();
|
|
}
|
|
}
|
|
|
|
void DirectTunnel::handleTunnelError(const QString &reason)
|
|
{
|
|
std::cerr << "Tunnel error: " << reason.toLocal8Bit().constData() << std::endl;
|
|
emit finished(EXIT_FAILURE);
|
|
}
|
|
|
|
void DirectTunnel::handleTunnelClosed()
|
|
{
|
|
if (m_expectingChannelClose) {
|
|
std::cout << "Successfully detected channel close." << std::endl;
|
|
std::cout << "Test finished successfully." << std::endl;
|
|
emit finished(EXIT_SUCCESS);
|
|
} else {
|
|
std::cerr << "Error: Remote host closed channel." << std::endl;
|
|
emit finished(EXIT_FAILURE);
|
|
}
|
|
}
|
|
|
|
void DirectTunnel::handleNewConnection()
|
|
{
|
|
m_targetSocket = m_targetServer->nextPendingConnection();
|
|
m_targetServer->close();
|
|
connect(m_targetSocket,
|
|
static_cast<void (QAbstractSocket::*)(QAbstractSocket::SocketError)>(&QAbstractSocket::error),
|
|
this, &DirectTunnel::handleSocketError);
|
|
connect(m_targetSocket, &QIODevice::readyRead, this, &DirectTunnel::handleClientData);
|
|
handleClientData();
|
|
}
|
|
|
|
void DirectTunnel::handleSocketError()
|
|
{
|
|
std::cerr << "Socket error: " << m_targetSocket->errorString().toLocal8Bit().constData()
|
|
<< std::endl;
|
|
emit finished(EXIT_FAILURE);
|
|
}
|
|
|
|
void DirectTunnel::handleClientData()
|
|
{
|
|
m_dataReceivedFromClient += m_targetSocket->readAll();
|
|
if (m_dataReceivedFromClient == TestData) {
|
|
std::cout << "Client data successfully received by server, now sending data to client..."
|
|
<< std::endl;
|
|
m_targetSocket->write(ServerDataPrefix + m_dataReceivedFromClient);
|
|
}
|
|
}
|
|
|
|
void DirectTunnel::handleTimeout()
|
|
{
|
|
std::cerr << "Error: Timeout waiting for test completion." << std::endl;
|
|
emit finished(EXIT_FAILURE);
|
|
}
|