Move unarchiving code out of the plugin dialog code

Change-Id: I2b6510ae527d57a06692336cfd7b0434cdcbda51
Reviewed-by: Alessandro Portale <alessandro.portale@qt.io>
This commit is contained in:
Eike Ziller
2020-05-27 14:08:49 +02:00
parent 9ee693ee22
commit cc8148606a
6 changed files with 185 additions and 94 deletions

View File

@@ -10,6 +10,7 @@ add_qtc_library(Utils
algorithm.h algorithm.h
ansiescapecodehandler.cpp ansiescapecodehandler.h ansiescapecodehandler.cpp ansiescapecodehandler.h
appmainwindow.cpp appmainwindow.h appmainwindow.cpp appmainwindow.h
archive.cpp archive.h
basetreeview.cpp basetreeview.h basetreeview.cpp basetreeview.h
benchmarker.cpp benchmarker.h benchmarker.cpp benchmarker.h
buildablehelperlibrary.cpp buildablehelperlibrary.h buildablehelperlibrary.cpp buildablehelperlibrary.h

124
src/libs/utils/archive.cpp Normal file
View File

@@ -0,0 +1,124 @@
/****************************************************************************
**
** Copyright (C) 2020 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 "archive.h"
#include "algorithm.h"
#include "checkablemessagebox.h"
#include "environment.h"
#include "mimetypes/mimedatabase.h"
#include "qtcassert.h"
#include "synchronousprocess.h"
#include <QDir>
#include <QPushButton>
namespace {
struct Tool
{
Utils::FilePath executable;
QStringList arguments;
};
Utils::optional<Tool> unzipTool(const Utils::FilePath &src, const Utils::FilePath &dest)
{
const Utils::FilePath unzip = Utils::Environment::systemEnvironment().searchInPath(
Utils::HostOsInfo::withExecutableSuffix("unzip"));
if (!unzip.isEmpty())
return Tool{unzip, {"-o", src.toString(), "-d", dest.toString()}};
const Utils::FilePath sevenzip = Utils::Environment::systemEnvironment().searchInPath(
Utils::HostOsInfo::withExecutableSuffix("7z"));
if (!sevenzip.isEmpty())
return Tool{sevenzip, {"x", QString("-o") + dest.toString(), "-y", src.toString()}};
const Utils::FilePath cmake = Utils::Environment::systemEnvironment().searchInPath(
Utils::HostOsInfo::withExecutableSuffix("cmake"));
if (!cmake.isEmpty())
return Tool{cmake, {"-E", "tar", "xvf", src.toString()}};
return {};
}
} // namespace
namespace Utils {
bool Archive::supportsFile(const FilePath &filePath, QString *reason)
{
const QList<MimeType> mimeType = mimeTypesForFileName(filePath.toString());
if (!anyOf(mimeType, [](const MimeType &mt) { return mt.inherits("application/zip"); })) {
if (reason)
*reason = tr("File format not supported.");
return false;
}
if (!unzipTool({}, {})) {
if (reason)
*reason = tr("Could not find unzip, 7z, or cmake executable in PATH.");
return false;
}
return true;
}
bool Archive::unarchive(const FilePath &src, const FilePath &dest, QWidget *parent)
{
const Utils::optional<Tool> tool = unzipTool(src, dest);
QTC_ASSERT(tool, return false);
const QString workingDirectory = dest.toFileInfo().absoluteFilePath();
QDir(workingDirectory).mkpath(".");
CheckableMessageBox box(parent);
box.setIcon(QMessageBox::Information);
box.setWindowTitle(tr("Unzipping File"));
box.setText(tr("Unzipping \"%1\" to \"%2\".").arg(src.toUserOutput(), dest.toUserOutput()));
box.setStandardButtons(QDialogButtonBox::Ok | QDialogButtonBox::Cancel);
box.button(QDialogButtonBox::Ok)->setEnabled(false);
box.setCheckBoxVisible(false);
box.setDetailedText(
tr("Running %1\nin \"%2\".\n\n", "Running <cmd> in <workingdirectory>")
.arg(CommandLine(tool->executable, tool->arguments).toUserOutput(), workingDirectory));
QProcess process;
process.setProcessChannelMode(QProcess::MergedChannels);
QObject::connect(&process, &QProcess::readyReadStandardOutput, &box, [&box, &process]() {
box.setDetailedText(box.detailedText() + QString::fromUtf8(process.readAllStandardOutput()));
});
QObject::connect(&process,
QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),
[&box](int, QProcess::ExitStatus) {
box.button(QDialogButtonBox::Ok)->setEnabled(true);
box.button(QDialogButtonBox::Cancel)->setEnabled(false);
});
QObject::connect(&box, &QMessageBox::rejected, &process, [&process] {
SynchronousProcess::stopProcess(process);
});
process.setProgram(tool->executable.toString());
process.setArguments(tool->arguments);
process.setWorkingDirectory(workingDirectory);
process.start(QProcess::ReadOnly);
box.exec();
return process.exitStatus() == QProcess::NormalExit && process.exitCode() == 0;
}
} // namespace Utils

42
src/libs/utils/archive.h Normal file
View File

@@ -0,0 +1,42 @@
/****************************************************************************
**
** Copyright (C) 2020 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.
**
****************************************************************************/
#pragma once
#include "utils_global.h"
#include "fileutils.h"
namespace Utils {
class QTCREATOR_UTILS_EXPORT Archive
{
Q_DECLARE_TR_FUNCTIONS(Utils::Archive)
public:
static bool supportsFile(const FilePath &filePath, QString *reason = nullptr);
static bool unarchive(const FilePath &src, const FilePath &dest, QWidget *parent);
};
} // namespace Utils

View File

@@ -133,7 +133,8 @@ SOURCES += \
$$PWD/namevaluevalidator.cpp \ $$PWD/namevaluevalidator.cpp \
$$PWD/camelcasecursor.cpp \ $$PWD/camelcasecursor.cpp \
$$PWD/infolabel.cpp \ $$PWD/infolabel.cpp \
$$PWD/overlaywidget.cpp $$PWD/overlaywidget.cpp \
$$PWD/archive.cpp
HEADERS += \ HEADERS += \
$$PWD/environmentfwd.h \ $$PWD/environmentfwd.h \
@@ -281,7 +282,8 @@ HEADERS += \
$$PWD/namevaluevalidator.h \ $$PWD/namevaluevalidator.h \
$$PWD/camelcasecursor.h \ $$PWD/camelcasecursor.h \
$$PWD/infolabel.h \ $$PWD/infolabel.h \
$$PWD/overlaywidget.h $$PWD/overlaywidget.h \
$$PWD/archive.h
FORMS += $$PWD/filewizardpage.ui \ FORMS += $$PWD/filewizardpage.ui \
$$PWD/projectintropage.ui \ $$PWD/projectintropage.ui \

View File

@@ -44,6 +44,8 @@ Project {
"ansiescapecodehandler.h", "ansiescapecodehandler.h",
"appmainwindow.cpp", "appmainwindow.cpp",
"appmainwindow.h", "appmainwindow.h",
"archive.cpp",
"archive.h",
"basetreeview.cpp", "basetreeview.cpp",
"basetreeview.h", "basetreeview.h",
"benchmarker.cpp", "benchmarker.cpp",

View File

@@ -37,15 +37,10 @@
#include <extensionsystem/pluginspec.h> #include <extensionsystem/pluginspec.h>
#include <extensionsystem/pluginview.h> #include <extensionsystem/pluginview.h>
#include <utils/algorithm.h> #include <utils/archive.h>
#include <utils/checkablemessagebox.h>
#include <utils/environment.h>
#include <utils/fancylineedit.h> #include <utils/fancylineedit.h>
#include <utils/infolabel.h> #include <utils/infolabel.h>
#include <utils/mimetypes/mimedatabase.h>
#include <utils/pathchooser.h> #include <utils/pathchooser.h>
#include <utils/qtcassert.h>
#include <utils/synchronousprocess.h>
#include <utils/wizard.h> #include <utils/wizard.h>
#include <utils/wizardpage.h> #include <utils/wizardpage.h>
@@ -72,45 +67,13 @@ static bool s_isRestartRequired = false;
const char kPath[] = "Path"; const char kPath[] = "Path";
const char kApplicationInstall[] = "ApplicationInstall"; const char kApplicationInstall[] = "ApplicationInstall";
static bool hasLibSuffix(const QString &path) static bool hasLibSuffix(const FilePath &path)
{ {
return (HostOsInfo().isWindowsHost() && path.endsWith(".dll", Qt::CaseInsensitive)) return (HostOsInfo().isWindowsHost() && path.endsWith(".dll"))
|| (HostOsInfo().isLinuxHost() && QFileInfo(path).completeSuffix().startsWith(".so")) || (HostOsInfo().isLinuxHost() && path.toFileInfo().completeSuffix().startsWith(".so"))
|| (HostOsInfo().isMacHost() && path.endsWith(".dylib")); || (HostOsInfo().isMacHost() && path.endsWith(".dylib"));
} }
static bool isZipFile(const QString &path)
{
const QList<MimeType> mimeType = mimeTypesForFileName(path);
return anyOf(mimeType, [](const MimeType &mt) { return mt.inherits("application/zip"); });
}
struct Tool
{
FilePath executable;
QStringList arguments;
};
static Utils::optional<Tool> unzipTool(const FilePath &src, const FilePath &dest)
{
const FilePath unzip = Utils::Environment::systemEnvironment().searchInPath(
Utils::HostOsInfo::withExecutableSuffix("unzip"));
if (!unzip.isEmpty())
return Tool{unzip, {"-o", src.toString(), "-d", dest.toString()}};
const FilePath sevenzip = Utils::Environment::systemEnvironment().searchInPath(
Utils::HostOsInfo::withExecutableSuffix("7z"));
if (!sevenzip.isEmpty())
return Tool{sevenzip, {"x", QString("-o") + dest.toString(), "-y", src.toString()}};
const FilePath cmake = Utils::Environment::systemEnvironment().searchInPath(
Utils::HostOsInfo::withExecutableSuffix("cmake"));
if (!cmake.isEmpty())
return Tool{cmake, {"-E", "tar", "xvf", src.toString()}};
return {};
}
class SourcePage : public WizardPage class SourcePage : public WizardPage
{ {
public: public:
@@ -149,21 +112,17 @@ public:
bool isComplete() const bool isComplete() const
{ {
const QString path = field(kPath).toString(); const auto path = FilePath::fromVariant(field(kPath));
if (!QFile::exists(path)) { if (!QFile::exists(path.toString())) {
m_info->setText(PluginDialog::tr("File does not exist.")); m_info->setText(PluginDialog::tr("File does not exist."));
return false; return false;
} }
if (hasLibSuffix(path)) if (hasLibSuffix(path))
return true; return true;
if (!isZipFile(path)) { QString error;
m_info->setText(PluginDialog::tr("File format not supported.")); if (!Archive::supportsFile(path, &error)) {
return false; m_info->setText(error);
}
if (!unzipTool({}, {})) {
m_info->setText(
PluginDialog::tr("Could not find unzip, 7z, or cmake executable in PATH."));
return false; return false;
} }
return true; return true;
@@ -367,45 +326,6 @@ static bool copyPluginFile(const FilePath &src, const FilePath &dest)
return true; return true;
} }
static bool unzip(const FilePath &src, const FilePath &dest)
{
const Utils::optional<Tool> tool = unzipTool(src, dest);
QTC_ASSERT(tool, return false);
const QString workingDirectory = dest.toFileInfo().absoluteFilePath();
QDir(workingDirectory).mkpath(".");
CheckableMessageBox box(ICore::dialogParent());
box.setIcon(QMessageBox::Information);
box.setWindowTitle(PluginDialog::tr("Unzipping File"));
box.setText(PluginDialog::tr("Unzipping \"%1\" to \"%2\".")
.arg(src.toUserOutput(), dest.toUserOutput()));
box.setStandardButtons(QDialogButtonBox::Ok | QDialogButtonBox::Cancel);
box.button(QDialogButtonBox::Ok)->setEnabled(false);
box.setCheckBoxVisible(false);
box.setDetailedText(
PluginDialog::tr("Running %1\nin \"%2\".\n\n", "Running <cmd> in <workingdirectory>")
.arg(CommandLine(tool->executable, tool->arguments).toUserOutput(), workingDirectory));
QProcess process;
process.setProcessChannelMode(QProcess::MergedChannels);
QObject::connect(&process, &QProcess::readyReadStandardOutput, &box, [&box, &process]() {
box.setDetailedText(box.detailedText() + QString::fromUtf8(process.readAllStandardOutput()));
});
QObject::connect(&process,
QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),
[&box](int, QProcess::ExitStatus) {
box.button(QDialogButtonBox::Ok)->setEnabled(true);
box.button(QDialogButtonBox::Cancel)->setEnabled(false);
});
QObject::connect(&box, &QMessageBox::rejected, &process, [&process] {
SynchronousProcess::stopProcess(process);
});
process.setProgram(tool->executable.toString());
process.setArguments(tool->arguments);
process.setWorkingDirectory(workingDirectory);
process.start(QProcess::ReadOnly);
box.exec();
return process.exitStatus() == QProcess::NormalExit && process.exitCode() == 0;
}
void PluginDialog::showInstallWizard() void PluginDialog::showInstallWizard()
{ {
Wizard wizard(ICore::dialogParent()); Wizard wizard(ICore::dialogParent());
@@ -423,11 +343,11 @@ void PluginDialog::showInstallWizard()
if (wizard.exec()) { if (wizard.exec()) {
const FilePath path = pluginFilePath(&wizard); const FilePath path = pluginFilePath(&wizard);
const FilePath installPath = pluginInstallPath(&wizard); const FilePath installPath = pluginInstallPath(&wizard);
if (hasLibSuffix(path.toString())) { if (hasLibSuffix(path)) {
if (copyPluginFile(path, installPath)) if (copyPluginFile(path, installPath))
updateRestartRequired(); updateRestartRequired();
} else if (isZipFile(path.toString())) { } else if (Archive::supportsFile(path)) {
if (unzip(path, installPath)) if (Archive::unarchive(path, installPath, ICore::dialogParent()))
updateRestartRequired(); updateRestartRequired();
} }
} }