forked from qt-creator/qt-creator
Add static PatchTool::confirmPatching() and reuse it in two places. Use Tr::tr() inside PatchTool. Change-Id: I70779619dbb58ab6e46a585bbeff51588ccb2f53 Reviewed-by: Orgad Shaneh <orgads@gmail.com>
138 lines
4.9 KiB
C++
138 lines
4.9 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0+ OR GPL-3.0 WITH Qt-GPL-exception-1.0
|
|
|
|
#include "coreplugintr.h"
|
|
#include "icore.h"
|
|
#include "messagemanager.h"
|
|
#include "patchtool.h"
|
|
|
|
#include <utils/environment.h>
|
|
#include <utils/qtcprocess.h>
|
|
|
|
#include <QMessageBox>
|
|
|
|
using namespace Utils;
|
|
|
|
namespace Core {
|
|
|
|
const char settingsGroupC[] = "General";
|
|
const char patchCommandKeyC[] = "PatchCommand";
|
|
const char patchCommandDefaultC[] = "patch";
|
|
|
|
FilePath PatchTool::patchCommand()
|
|
{
|
|
QSettings *s = ICore::settings();
|
|
|
|
s->beginGroup(settingsGroupC);
|
|
const FilePath command = FilePath::fromVariant(s->value(patchCommandKeyC, patchCommandDefaultC));
|
|
s->endGroup();
|
|
|
|
return command;
|
|
}
|
|
|
|
void PatchTool::setPatchCommand(const FilePath &newCommand)
|
|
{
|
|
Utils::QtcSettings *s = ICore::settings();
|
|
s->beginGroup(settingsGroupC);
|
|
s->setValueWithDefault(patchCommandKeyC, newCommand.toVariant(), QVariant(QString(patchCommandDefaultC)));
|
|
s->endGroup();
|
|
}
|
|
|
|
bool PatchTool::confirmPatching(QWidget *parent, PatchAction patchAction)
|
|
{
|
|
const QString title = patchAction == PatchAction::Apply ? Tr::tr("Apply Chunk")
|
|
: Tr::tr("Revert Chunk");
|
|
const QString question = patchAction == PatchAction::Apply
|
|
? Tr::tr("Would you like to apply the chunk?")
|
|
: Tr::tr("Would you like to revert the chunk?");
|
|
return QMessageBox::question(parent, title, question, QMessageBox::Yes | QMessageBox::No)
|
|
== QMessageBox::Yes;
|
|
}
|
|
|
|
static bool runPatchHelper(const QByteArray &input, const FilePath &workingDirectory,
|
|
int strip, PatchAction patchAction, bool withCrlf)
|
|
{
|
|
const FilePath patch = PatchTool::patchCommand();
|
|
if (patch.isEmpty()) {
|
|
MessageManager::writeDisrupting(Tr::tr("There is no patch-command configured in "
|
|
"the general \"Environment\" settings."));
|
|
return false;
|
|
}
|
|
|
|
if (!patch.exists() && !patch.searchInPath().exists()) {
|
|
MessageManager::writeDisrupting(Tr::tr("The patch-command configured in the general "
|
|
"\"Environment\" settings does not exist."));
|
|
return false;
|
|
}
|
|
|
|
QtcProcess patchProcess;
|
|
if (!workingDirectory.isEmpty())
|
|
patchProcess.setWorkingDirectory(workingDirectory);
|
|
Environment env = Environment::systemEnvironment();
|
|
env.setupEnglishOutput();
|
|
patchProcess.setEnvironment(env);
|
|
QStringList args;
|
|
// Add argument 'apply' when git is used as patch command since git 2.5/Windows
|
|
// no longer ships patch.exe.
|
|
if (patch.endsWith("git") || patch.endsWith("git.exe"))
|
|
args << "apply";
|
|
|
|
if (strip >= 0)
|
|
args << ("-p" + QString::number(strip));
|
|
if (patchAction == PatchAction::Revert)
|
|
args << "-R";
|
|
if (withCrlf)
|
|
args << "--binary";
|
|
MessageManager::writeDisrupting(Tr::tr("Running in %1: %2 %3")
|
|
.arg(workingDirectory.toUserOutput(), patch.toUserOutput(), args.join(' ')));
|
|
patchProcess.setCommand({patch, args});
|
|
patchProcess.setWriteData(input);
|
|
patchProcess.start();
|
|
if (!patchProcess.waitForStarted()) {
|
|
MessageManager::writeFlashing(Tr::tr("Unable to launch \"%1\": %2")
|
|
.arg(patch.toUserOutput(), patchProcess.errorString()));
|
|
return false;
|
|
}
|
|
|
|
QByteArray stdOut;
|
|
QByteArray stdErr;
|
|
if (!patchProcess.readDataFromProcess(&stdOut, &stdErr)) {
|
|
patchProcess.stop();
|
|
patchProcess.waitForFinished();
|
|
MessageManager::writeFlashing(Tr::tr("A timeout occurred running \"%1\"")
|
|
.arg(patch.toUserOutput()));
|
|
return false;
|
|
|
|
}
|
|
if (!stdOut.isEmpty()) {
|
|
if (stdOut.contains("(different line endings)") && !withCrlf) {
|
|
QByteArray crlfInput = input;
|
|
crlfInput.replace('\n', "\r\n");
|
|
return runPatchHelper(crlfInput, workingDirectory, strip, patchAction, true);
|
|
} else {
|
|
MessageManager::writeFlashing(QString::fromLocal8Bit(stdOut));
|
|
}
|
|
}
|
|
if (!stdErr.isEmpty())
|
|
MessageManager::writeFlashing(QString::fromLocal8Bit(stdErr));
|
|
|
|
if (patchProcess.exitStatus() != QProcess::NormalExit) {
|
|
MessageManager::writeFlashing(Tr::tr("\"%1\" crashed.").arg(patch.toUserOutput()));
|
|
return false;
|
|
}
|
|
if (patchProcess.exitCode() != 0) {
|
|
MessageManager::writeFlashing(Tr::tr("\"%1\" failed (exit code %2).")
|
|
.arg(patch.toUserOutput()).arg(patchProcess.exitCode()));
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool PatchTool::runPatch(const QByteArray &input, const FilePath &workingDirectory,
|
|
int strip, PatchAction patchAction)
|
|
{
|
|
return runPatchHelper(input, workingDirectory, strip, patchAction, false);
|
|
}
|
|
|
|
} // namespace Core
|