Files
qt-creator/src/plugins/perforce/perforcechecker.cpp
Tobias Hunger 8b46136b60 Perforce: Modernize
* use pragma once
* remove debug code
* Remove some "slots"
* Use Qt5 style connects (almost everywhere)

Change-Id: Iaa9c3bbacbb2be3c96502c52cc169ad4ec7015a5
Reviewed-by: Orgad Shaneh <orgads@gmail.com>
2016-02-10 13:32:50 +00:00

216 lines
6.4 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 "perforcechecker.h"
#include <utils/qtcassert.h>
#include <utils/synchronousprocess.h>
#include <QRegExp>
#include <QTimer>
#include <QFileInfo>
#include <QDir>
#include <QApplication>
#include <QCursor>
namespace Perforce {
namespace Internal {
PerforceChecker::PerforceChecker(QObject *parent) : QObject(parent)
{
connect(&m_process, static_cast<void (QProcess::*)(QProcess::ProcessError)>(&QProcess::error),
this, &PerforceChecker::slotError);
connect(&m_process, static_cast<void (QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished),
this, &PerforceChecker::slotFinished);
}
PerforceChecker::~PerforceChecker()
{
m_process.kill();
m_process.waitForFinished();
resetOverrideCursor();
}
bool PerforceChecker::isRunning() const
{
return m_process.state() == QProcess::Running;
}
bool PerforceChecker::waitForFinished(int msec)
{
return m_process.waitForFinished(msec);
}
void PerforceChecker::resetOverrideCursor()
{
if (m_isOverrideCursor) {
QApplication::restoreOverrideCursor();
m_isOverrideCursor = false;
}
}
void PerforceChecker::start(const QString &binary, const QString &workingDirectory,
const QStringList &basicArgs,
int timeoutMS)
{
if (isRunning()) {
emitFailed(QLatin1String("Internal error: process still running"));
return;
}
if (binary.isEmpty()) {
emitFailed(tr("No executable specified"));
return;
}
m_binary = binary;
QStringList args = basicArgs;
args << QLatin1String("client") << QLatin1String("-o");
if (!workingDirectory.isEmpty())
m_process.setWorkingDirectory(workingDirectory);
m_process.start(m_binary, args);
m_process.closeWriteChannel();
// Timeout handling
m_timeOutMS = timeoutMS;
m_timedOut = false;
if (timeoutMS > 0)
QTimer::singleShot(m_timeOutMS, this, &PerforceChecker::slotTimeOut);
// Cursor
if (m_useOverideCursor) {
m_isOverrideCursor = true;
QApplication::setOverrideCursor(QCursor(Qt::BusyCursor));
}
}
void PerforceChecker::slotTimeOut()
{
if (!isRunning())
return;
m_timedOut = true;
Utils::SynchronousProcess::stopProcess(m_process);
emitFailed(tr("\"%1\" timed out after %2 ms.").
arg(m_binary).arg(m_timeOutMS));
}
void PerforceChecker::slotError(QProcess::ProcessError error)
{
if (m_timedOut)
return;
switch (error) {
case QProcess::FailedToStart:
emitFailed(tr("Unable to launch \"%1\": %2").
arg(QDir::toNativeSeparators(m_binary), m_process.errorString()));
break;
case QProcess::Crashed: // Handled elsewhere
case QProcess::Timedout:
break;
case QProcess::ReadError:
case QProcess::WriteError:
case QProcess::UnknownError:
Utils::SynchronousProcess::stopProcess(m_process);
break;
}
}
void PerforceChecker::slotFinished(int exitCode, QProcess::ExitStatus exitStatus)
{
if (m_timedOut)
return;
switch (exitStatus) {
case QProcess::CrashExit:
emitFailed(tr("\"%1\" crashed.").arg(QDir::toNativeSeparators(m_binary)));
break;
case QProcess::NormalExit:
if (exitCode) {
const QString stdErr = QString::fromLocal8Bit(m_process.readAllStandardError());
emitFailed(tr("\"%1\" terminated with exit code %2: %3").
arg(QDir::toNativeSeparators(m_binary)).arg(exitCode).arg(stdErr));
} else {
parseOutput(QString::fromLocal8Bit(m_process.readAllStandardOutput()));
}
break;
}
}
// Parse p4 client output for the top level
static inline QString clientRootFromOutput(const QString &in)
{
QRegExp regExp(QLatin1String("(\\n|\\r\\n|\\r)Root:\\s*(.*)(\\n|\\r\\n|\\r)"));
QTC_ASSERT(regExp.isValid(), return QString());
regExp.setMinimal(true);
// Normalize slashes and capitalization of Windows drive letters for caching.
if (regExp.indexIn(in) != -1)
return QFileInfo(regExp.cap(2).trimmed()).absoluteFilePath();
return QString();
}
void PerforceChecker::parseOutput(const QString &response)
{
if (!response.contains(QLatin1String("View:")) && !response.contains(QLatin1String("//depot/"))) {
emitFailed(tr("The client does not seem to contain any mapped files."));
return;
}
const QString repositoryRoot = clientRootFromOutput(response);
if (repositoryRoot.isEmpty()) {
//: Unable to determine root of the p4 client installation
emitFailed(tr("Unable to determine the client root."));
return;
}
// Check existence. No precise check here, might be a symlink
if (QFileInfo::exists(repositoryRoot)) {
emitSucceeded(repositoryRoot);
} else {
emitFailed(tr("The repository \"%1\" does not exist.").
arg(QDir::toNativeSeparators(repositoryRoot)));
}
}
void PerforceChecker::emitFailed(const QString &m)
{
resetOverrideCursor();
emit failed(m);
}
void PerforceChecker::emitSucceeded(const QString &m)
{
resetOverrideCursor();
emit succeeded(m);
}
bool PerforceChecker::useOverideCursor() const
{
return m_useOverideCursor;
}
void PerforceChecker::setUseOverideCursor(bool v)
{
m_useOverideCursor = v;
}
}
}