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);
 | 
						|
}
 |