forked from qt-creator/qt-creator
Since we also license under GPL-3.0 WITH Qt-GPL-exception-1.0,
this applies only to a hypothetical newer version of GPL, that doesn't
exist yet. If such a version emerges, we can still decide to relicense...
While at it, replace (deprecated) GPL-3.0 with more explicit GPL-3.0-only
Change was done by running
find . -type f -exec perl -pi -e "s/LicenseRef-Qt-Commercial OR GPL-3.0\+ OR GPL-3.0 WITH Qt-GPL-exception-1.0/LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0/g" {} \;
Change-Id: I5097e6ce8d10233993ee30d7e25120e2659eb10b
Reviewed-by: Eike Ziller <eike.ziller@qt.io>
330 lines
11 KiB
C++
330 lines
11 KiB
C++
// Copyright (C) 2016 BlackBerry Limited. All rights reserved.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
#include "qnxdeployqtlibrariesdialog.h"
|
|
|
|
#include "qnxconstants.h"
|
|
#include "qnxqtversion.h"
|
|
#include "qnxtr.h"
|
|
|
|
#include <projectexplorer/deployablefile.h>
|
|
#include <projectexplorer/devicesupport/idevice.h>
|
|
#include <qtsupport/qtversionmanager.h>
|
|
#include <remotelinux/genericdirectuploadservice.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/layoutbuilder.h>
|
|
#include <utils/qtcassert.h>
|
|
|
|
#include <QComboBox>
|
|
#include <QDialog>
|
|
#include <QDir>
|
|
#include <QLabel>
|
|
#include <QLineEdit>
|
|
#include <QMessageBox>
|
|
#include <QPlainTextEdit>
|
|
#include <QProgressBar>
|
|
#include <QPushButton>
|
|
#include <QSpacerItem>
|
|
#include <QVariant>
|
|
|
|
using namespace ProjectExplorer;
|
|
using namespace QtSupport;
|
|
using namespace RemoteLinux;
|
|
using namespace Utils;
|
|
|
|
namespace Qnx::Internal {
|
|
|
|
QnxDeployQtLibrariesDialog::QnxDeployQtLibrariesDialog(const IDevice::ConstPtr &device,
|
|
QWidget *parent)
|
|
: QDialog(parent)
|
|
, m_device(device)
|
|
{
|
|
setWindowTitle(Tr::tr("Deploy Qt to QNX Device"));
|
|
|
|
m_qtLibraryCombo = new QComboBox(this);
|
|
const QList<QtVersion*> qtVersions = QtVersionManager::sortVersions(
|
|
QtVersionManager::versions(QtVersion::isValidPredicate(
|
|
equal(&QtVersion::type, QString::fromLatin1(Constants::QNX_QNX_QT)))));
|
|
for (QtVersion *v : qtVersions)
|
|
m_qtLibraryCombo->addItem(v->displayName(), v->uniqueId());
|
|
|
|
m_deployButton = new QPushButton(Tr::tr("Deploy"), this);
|
|
m_deployButton->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed);
|
|
|
|
m_basePathLabel = new QLabel(this);
|
|
|
|
m_remoteDirectory = new QLineEdit(this);
|
|
m_remoteDirectory->setText(QLatin1String("/qt"));
|
|
|
|
m_deployProgress = new QProgressBar(this);
|
|
m_deployProgress->setValue(0);
|
|
m_deployProgress->setTextVisible(true);
|
|
|
|
m_deployLogWindow = new QPlainTextEdit(this);
|
|
|
|
m_closeButton = new QPushButton(Tr::tr("Close"), this);
|
|
|
|
m_uploadService = new GenericDirectUploadService(this);
|
|
m_uploadService->setDevice(m_device);
|
|
|
|
using namespace Layouting;
|
|
|
|
Column {
|
|
Form {
|
|
Tr::tr("Qt library to deploy:"), m_qtLibraryCombo, m_deployButton, br,
|
|
Tr::tr("Remote directory:"), m_basePathLabel, m_remoteDirectory, br
|
|
},
|
|
m_deployProgress,
|
|
m_deployLogWindow,
|
|
Row { st, m_closeButton }
|
|
}.attachTo(this);
|
|
|
|
connect(m_uploadService, &AbstractRemoteLinuxDeployService::progressMessage,
|
|
this, &QnxDeployQtLibrariesDialog::updateProgress);
|
|
connect(m_uploadService, &AbstractRemoteLinuxDeployService::progressMessage,
|
|
m_deployLogWindow, &QPlainTextEdit::appendPlainText);
|
|
connect(m_uploadService, &AbstractRemoteLinuxDeployService::errorMessage,
|
|
m_deployLogWindow, &QPlainTextEdit::appendPlainText);
|
|
connect(m_uploadService, &AbstractRemoteLinuxDeployService::warningMessage,
|
|
this, [this](const QString &message) {
|
|
if (!message.contains("stat:"))
|
|
m_deployLogWindow->appendPlainText(message);
|
|
});
|
|
connect(m_uploadService, &AbstractRemoteLinuxDeployService::stdOutData,
|
|
m_deployLogWindow, &QPlainTextEdit::appendPlainText);
|
|
connect(m_uploadService, &AbstractRemoteLinuxDeployService::stdErrData,
|
|
m_deployLogWindow, &QPlainTextEdit::appendPlainText);
|
|
connect(m_uploadService, &AbstractRemoteLinuxDeployService::finished,
|
|
this, &QnxDeployQtLibrariesDialog::handleUploadFinished);
|
|
|
|
connect(&m_checkDirProcess, &QtcProcess::done,
|
|
this, &QnxDeployQtLibrariesDialog::handleCheckDirDone);
|
|
connect(&m_removeDirProcess, &QtcProcess::done,
|
|
this, &QnxDeployQtLibrariesDialog::handleRemoveDirDone);
|
|
|
|
connect(m_deployButton, &QAbstractButton::clicked,
|
|
this, &QnxDeployQtLibrariesDialog::deployLibraries);
|
|
connect(m_closeButton, &QAbstractButton::clicked,
|
|
this, &QWidget::close);
|
|
}
|
|
|
|
QnxDeployQtLibrariesDialog::~QnxDeployQtLibrariesDialog() = default;
|
|
|
|
int QnxDeployQtLibrariesDialog::execAndDeploy(int qtVersionId, const QString &remoteDirectory)
|
|
{
|
|
m_remoteDirectory->setText(remoteDirectory);
|
|
m_qtLibraryCombo->setCurrentIndex(m_qtLibraryCombo->findData(qtVersionId));
|
|
|
|
deployLibraries();
|
|
return exec();
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::closeEvent(QCloseEvent *event)
|
|
{
|
|
// A disabled Deploy button indicates the upload is still running
|
|
if (!m_deployButton->isEnabled()) {
|
|
const int answer = QMessageBox::question(this, windowTitle(),
|
|
Tr::tr("Closing the dialog will stop the deployment. Are you sure you want to do this?"),
|
|
QMessageBox::Yes | QMessageBox::No);
|
|
if (answer == QMessageBox::No)
|
|
event->ignore();
|
|
else if (answer == QMessageBox::Yes)
|
|
m_uploadService->stop();
|
|
}
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::deployLibraries()
|
|
{
|
|
QTC_ASSERT(m_state == Inactive, return);
|
|
|
|
if (m_remoteDirectory->text().isEmpty()) {
|
|
QMessageBox::warning(this, windowTitle(),
|
|
Tr::tr("Please input a remote directory to deploy to."));
|
|
return;
|
|
}
|
|
|
|
QTC_ASSERT(!m_device.isNull(), return);
|
|
|
|
m_progressCount = 0;
|
|
m_deployProgress->setValue(0);
|
|
m_remoteDirectory->setEnabled(false);
|
|
m_deployButton->setEnabled(false);
|
|
m_qtLibraryCombo->setEnabled(false);
|
|
m_deployLogWindow->clear();
|
|
|
|
startCheckDirProcess();
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::startUpload()
|
|
{
|
|
QTC_CHECK(m_state == CheckingRemoteDirectory || m_state == RemovingRemoteDirectory);
|
|
|
|
m_state = Uploading;
|
|
|
|
QList<DeployableFile> filesToUpload = gatherFiles();
|
|
|
|
m_deployProgress->setRange(0, filesToUpload.count());
|
|
|
|
m_uploadService->setDeployableFiles(filesToUpload);
|
|
m_uploadService->start();
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::updateProgress(const QString &progressMessage)
|
|
{
|
|
QTC_CHECK(m_state == Uploading);
|
|
|
|
const int progress = progressMessage.count("sftp> put") + progressMessage.count("sftp> ln -s");
|
|
if (progress != 0) {
|
|
m_progressCount += progress;
|
|
m_deployProgress->setValue(m_progressCount);
|
|
}
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::handleUploadFinished()
|
|
{
|
|
m_remoteDirectory->setEnabled(true);
|
|
m_deployButton->setEnabled(true);
|
|
m_qtLibraryCombo->setEnabled(true);
|
|
|
|
m_state = Inactive;
|
|
}
|
|
|
|
QList<DeployableFile> QnxDeployQtLibrariesDialog::gatherFiles()
|
|
{
|
|
QList<DeployableFile> result;
|
|
|
|
const int qtVersionId = m_qtLibraryCombo->itemData(m_qtLibraryCombo->currentIndex()).toInt();
|
|
|
|
auto qtVersion = dynamic_cast<const QnxQtVersion *>(QtVersionManager::version(qtVersionId));
|
|
|
|
QTC_ASSERT(qtVersion, return result);
|
|
|
|
if (HostOsInfo::isWindowsHost()) {
|
|
result.append(gatherFiles(qtVersion->libraryPath().toString(),
|
|
QString(),
|
|
QStringList() << QLatin1String("*.so.?")));
|
|
result.append(gatherFiles(qtVersion->libraryPath().toString() + QLatin1String("/fonts")));
|
|
} else {
|
|
result.append(gatherFiles(qtVersion->libraryPath().toString()));
|
|
}
|
|
|
|
result.append(gatherFiles(qtVersion->pluginPath().toString()));
|
|
result.append(gatherFiles(qtVersion->importsPath().toString()));
|
|
result.append(gatherFiles(qtVersion->qmlPath().toString()));
|
|
return result;
|
|
}
|
|
|
|
QList<DeployableFile> QnxDeployQtLibrariesDialog::gatherFiles(
|
|
const QString &dirPath, const QString &baseDirPath, const QStringList &nameFilters)
|
|
{
|
|
QList<DeployableFile> result;
|
|
if (dirPath.isEmpty())
|
|
return result;
|
|
|
|
static const QStringList unusedDirs = {"include", "mkspecs", "cmake", "pkgconfig"};
|
|
const QString dp = dirPath.endsWith('/') ? dirPath.left(dirPath.size() - 1) : dirPath;
|
|
if (unusedDirs.contains(dp))
|
|
return result;
|
|
|
|
const QDir dir(dirPath);
|
|
QFileInfoList list = dir.entryInfoList(nameFilters,
|
|
QDir::Dirs | QDir::Files | QDir::NoDotAndDotDot);
|
|
|
|
for (auto &fileInfo : list) {
|
|
if (fileInfo.isDir()) {
|
|
result.append(gatherFiles(fileInfo.absoluteFilePath(), baseDirPath.isEmpty() ?
|
|
dirPath : baseDirPath));
|
|
} else {
|
|
static const QStringList unusedSuffixes = {"cmake", "la", "prl", "a", "pc"};
|
|
if (unusedSuffixes.contains(fileInfo.suffix()))
|
|
continue;
|
|
|
|
QString remoteDir;
|
|
if (baseDirPath.isEmpty()) {
|
|
remoteDir = fullRemoteDirectory() + QLatin1Char('/') +
|
|
QFileInfo(dirPath).baseName();
|
|
} else {
|
|
QDir baseDir(baseDirPath);
|
|
baseDir.cdUp();
|
|
remoteDir = fullRemoteDirectory() + QLatin1Char('/') +
|
|
baseDir.relativeFilePath(dirPath);
|
|
}
|
|
result.append(DeployableFile(FilePath::fromString(fileInfo.absoluteFilePath()),
|
|
remoteDir));
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
QString QnxDeployQtLibrariesDialog::fullRemoteDirectory() const
|
|
{
|
|
return m_remoteDirectory->text();
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::startCheckDirProcess()
|
|
{
|
|
QTC_CHECK(m_state == Inactive);
|
|
m_state = CheckingRemoteDirectory;
|
|
m_deployLogWindow->appendPlainText(Tr::tr("Checking existence of \"%1\"")
|
|
.arg(fullRemoteDirectory()));
|
|
m_checkDirProcess.setCommand({m_device->filePath("test"),
|
|
{"-d", fullRemoteDirectory()}});
|
|
m_checkDirProcess.start();
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::startRemoveDirProcess()
|
|
{
|
|
QTC_CHECK(m_state == CheckingRemoteDirectory);
|
|
m_state = RemovingRemoteDirectory;
|
|
m_deployLogWindow->appendPlainText(Tr::tr("Removing \"%1\"").arg(fullRemoteDirectory()));
|
|
m_removeDirProcess.setCommand({m_device->filePath("rm"),
|
|
{"-rf", fullRemoteDirectory()}});
|
|
m_removeDirProcess.start();
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::handleCheckDirDone()
|
|
{
|
|
QTC_CHECK(m_state == CheckingRemoteDirectory);
|
|
if (handleError(m_checkDirProcess))
|
|
return;
|
|
|
|
if (m_checkDirProcess.exitCode() == 0) { // Directory exists
|
|
const int answer = QMessageBox::question(this, windowTitle(),
|
|
Tr::tr("The remote directory \"%1\" already exists.\n"
|
|
"Deploying to that directory will remove any files already present.\n\n"
|
|
"Are you sure you want to continue?").arg(fullRemoteDirectory()),
|
|
QMessageBox::Yes | QMessageBox::No);
|
|
if (answer == QMessageBox::Yes)
|
|
startRemoveDirProcess();
|
|
else
|
|
handleUploadFinished();
|
|
} else {
|
|
startUpload();
|
|
}
|
|
}
|
|
|
|
void QnxDeployQtLibrariesDialog::handleRemoveDirDone()
|
|
{
|
|
QTC_CHECK(m_state == RemovingRemoteDirectory);
|
|
if (handleError(m_removeDirProcess))
|
|
return;
|
|
|
|
QTC_ASSERT(m_removeDirProcess.exitCode() == 0, return);
|
|
startUpload();
|
|
}
|
|
|
|
// Returns true if the error appeared, false when finished with success
|
|
bool QnxDeployQtLibrariesDialog::handleError(const QtcProcess &process)
|
|
{
|
|
if (process.result() == ProcessResult::FinishedWithSuccess)
|
|
return false;
|
|
|
|
m_deployLogWindow->appendPlainText(Tr::tr("Connection failed: %1").arg(process.errorString()));
|
|
handleUploadFinished();
|
|
return true;
|
|
}
|
|
|
|
} // Qnx::Internal
|