/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "abstractprocessstep.h" #include "buildconfiguration.h" #include "buildstep.h" #include "ioutputparser.h" #include "project.h" #include "target.h" #include "task.h" #include #include #include #include #include using namespace ProjectExplorer; /*! \class ProjectExplorer::AbstractProcessStep \brief A convenience class, which can be used as a base class instead of BuildStep. It should be used as a base class if your buildstep just needs to run a process. Usage: \list \o Use processParameters() to configure the process you want to run (you need to do that before calling AbstractProcessStep::init()). \o Inside YourBuildStep::init() call AbstractProcessStep::init(). \o Inside YourBuildStep::run() call AbstractProcessStep::run(), which automatically starts the proces and by default adds the output on stdOut and stdErr to the OutputWindow. \o If you need to process the process output override stdOut() and/or stdErr. \endlist The two functions processStarted() and processFinished() are called after starting/finishing the process. By default they add a message to the output window. Use setEnabled() to control whether the BuildStep needs to run. (A disabled BuildStep immediately returns true, from the run function.) \sa ProjectExplorer::ProcessParameters */ /*! \fn void ProjectExplorer::AbstractProcessStep::setEnabled(bool b) \brief Enables or disables a BuildStep. Disabled BuildSteps immediately return true from their run method. Should be called from init() */ /*! \fn ProcessParameters *ProjectExplorer::AbstractProcessStep::processParameters() \brief Obtain a reference to the parameters for the actual process to run. Should be used in init() */ AbstractProcessStep::AbstractProcessStep(BuildStepList *bsl, const Core::Id id) : BuildStep(bsl, id), m_timer(0), m_futureInterface(0), m_ignoreReturnValue(false), m_process(0), m_eventLoop(0), m_outputParserChain(0) { } AbstractProcessStep::AbstractProcessStep(BuildStepList *bsl, AbstractProcessStep *bs) : BuildStep(bsl, bs), m_timer(0), m_futureInterface(0), m_ignoreReturnValue(bs->m_ignoreReturnValue), m_process(0), m_eventLoop(0), m_outputParserChain(0) { } AbstractProcessStep::~AbstractProcessStep() { delete m_process; delete m_timer; // do not delete m_futureInterface, we do not own it. delete m_outputParserChain; } /*! \brief Delete all existing output parsers and start a new chain with the given parser. Derived classes need to call this function. */ void AbstractProcessStep::setOutputParser(ProjectExplorer::IOutputParser *parser) { delete m_outputParserChain; m_outputParserChain = parser; if (m_outputParserChain) { connect(parser, SIGNAL(addOutput(QString,ProjectExplorer::BuildStep::OutputFormat)), this, SLOT(outputAdded(QString,ProjectExplorer::BuildStep::OutputFormat))); connect(parser, SIGNAL(addTask(ProjectExplorer::Task)), this, SLOT(taskAdded(ProjectExplorer::Task))); } } /*! \brief Append the given output parser to the existing chain of parsers. */ void AbstractProcessStep::appendOutputParser(ProjectExplorer::IOutputParser *parser) { if (!parser) return; QTC_ASSERT(m_outputParserChain, return); m_outputParserChain->appendOutputParser(parser); return; } ProjectExplorer::IOutputParser *AbstractProcessStep::outputParser() const { return m_outputParserChain; } bool AbstractProcessStep::ignoreReturnValue() { return m_ignoreReturnValue; } /*! \brief If ignoreReturnValue is set to true, then the abstractprocess step will return success even if the return value indicates otherwise. Should be called from init. */ void AbstractProcessStep::setIgnoreReturnValue(bool b) { m_ignoreReturnValue = b; } /*! \brief Reimplemented from BuildStep::init(). You need to call this from YourBuildStep::init() */ bool AbstractProcessStep::init() { return true; } /*! \brief Reimplemented from BuildStep::init(). You need to call this from YourBuildStep::run() */ void AbstractProcessStep::run(QFutureInterface &fi) { m_futureInterface = &fi; QDir wd(m_param.effectiveWorkingDirectory()); if (!wd.exists()) wd.mkpath(wd.absolutePath()); m_process = new Utils::QtcProcess(); #ifdef Q_OS_WIN m_process->setUseCtrlCStub(true); #endif m_process->setWorkingDirectory(wd.absolutePath()); m_process->setEnvironment(m_param.environment()); connect(m_process, SIGNAL(readyReadStandardOutput()), this, SLOT(processReadyReadStdOutput()), Qt::DirectConnection); connect(m_process, SIGNAL(readyReadStandardError()), this, SLOT(processReadyReadStdError()), Qt::DirectConnection); connect(m_process, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(slotProcessFinished(int,QProcess::ExitStatus)), Qt::DirectConnection); m_process->setCommand(m_param.effectiveCommand(), m_param.effectiveArguments()); m_process->start(); if (!m_process->waitForStarted()) { processStartupFailed(); delete m_process; m_process = 0; fi.reportResult(false); return; } processStarted(); m_timer = new QTimer(); connect(m_timer, SIGNAL(timeout()), this, SLOT(checkForCancel()), Qt::DirectConnection); m_timer->start(500); m_eventLoop = new QEventLoop; m_eventLoop->exec(); m_timer->stop(); delete m_timer; m_timer = 0; // The process has finished, leftover data is read in processFinished processFinished(m_process->exitCode(), m_process->exitStatus()); bool returnValue = processSucceeded(m_process->exitCode(), m_process->exitStatus()) || m_ignoreReturnValue; // Clean up output parsers if (m_outputParserChain) { delete m_outputParserChain; m_outputParserChain = 0; } delete m_process; m_process = 0; delete m_eventLoop; m_eventLoop = 0; fi.reportResult(returnValue); m_futureInterface = 0; return; } /*! \brief Called after the process is started. The default implementation adds a process started message to the output message */ void AbstractProcessStep::processStarted() { emit addOutput(tr("Starting: \"%1\" %2") .arg(QDir::toNativeSeparators(m_param.effectiveCommand()), m_param.prettyArguments()), BuildStep::MessageOutput); } /*! \brief Called after the process Finished. The default implementation adds a line to the output window */ void AbstractProcessStep::processFinished(int exitCode, QProcess::ExitStatus status) { QString command = QDir::toNativeSeparators(m_param.effectiveCommand()); if (status == QProcess::NormalExit && exitCode == 0) { emit addOutput(tr("The process \"%1\" exited normally.").arg(command), BuildStep::MessageOutput); } else if (status == QProcess::NormalExit) { emit addOutput(tr("The process \"%1\" exited with code %2.") .arg(command, QString::number(m_process->exitCode())), BuildStep::ErrorMessageOutput); } else { emit addOutput(tr("The process \"%1\" crashed.").arg(command), BuildStep::ErrorMessageOutput); } } /*! \brief Called if the process could not be started. By default adds a message to the output window. */ void AbstractProcessStep::processStartupFailed() { emit addOutput(tr("Could not start process \"%1\" %2") .arg(QDir::toNativeSeparators(m_param.effectiveCommand()), m_param.prettyArguments()), BuildStep::ErrorMessageOutput); } /*! \brief Called to test whether a prcess succeeded or not. */ bool AbstractProcessStep::processSucceeded(int exitCode, QProcess::ExitStatus status) { return exitCode == 0 && status == QProcess::NormalExit; } void AbstractProcessStep::processReadyReadStdOutput() { m_process->setReadChannel(QProcess::StandardOutput); while (m_process->canReadLine()) { QString line = QString::fromLocal8Bit(m_process->readLine()); stdOutput(line); } } /*! \brief Called for each line of output on stdOut(). The default implementation adds the line to the application output window. */ void AbstractProcessStep::stdOutput(const QString &line) { if (m_outputParserChain) m_outputParserChain->stdOutput(line); emit addOutput(line, BuildStep::NormalOutput, BuildStep::DontAppendNewline); } void AbstractProcessStep::processReadyReadStdError() { m_process->setReadChannel(QProcess::StandardError); while (m_process->canReadLine()) { QString line = QString::fromLocal8Bit(m_process->readLine()); stdError(line); } } /*! \brief Called for each line of output on StdErrror(). The default implementation adds the line to the application output window */ void AbstractProcessStep::stdError(const QString &line) { if (m_outputParserChain) m_outputParserChain->stdError(line); emit addOutput(line, BuildStep::ErrorOutput, BuildStep::DontAppendNewline); } void AbstractProcessStep::checkForCancel() { if (m_futureInterface->isCanceled() && m_timer->isActive()) { m_timer->stop(); m_process->terminate(); m_process->waitForFinished(5000); m_process->kill(); } } void AbstractProcessStep::taskAdded(const ProjectExplorer::Task &task) { // Do not bother to report issues if we do not care about the results of // the buildstep anyway: if (m_ignoreReturnValue) return; Task editable(task); QString filePath = task.file.toString(); if (!filePath.isEmpty() && !QDir::isAbsolutePath(filePath)) { // We have no save way to decide which file in which subfolder // is meant. Therefore we apply following heuristics: // 1. Check if file is unique in whole project // 2. Otherwise try again without any ../ // 3. give up. QList possibleFiles; QString fileName = QFileInfo(filePath).fileName(); foreach (const QString &file, project()->files(ProjectExplorer::Project::AllFiles)) { QFileInfo candidate(file); if (candidate.fileName() == fileName) possibleFiles << candidate; } if (possibleFiles.count() == 1) { editable.file = Utils::FileName(possibleFiles.first()); } else { // More then one filename, so do a better compare // Chop of any "../" while (filePath.startsWith(QLatin1String("../"))) filePath.remove(0, 3); int count = 0; QString possibleFilePath; foreach(const QFileInfo &fi, possibleFiles) { if (fi.filePath().endsWith(filePath)) { possibleFilePath = fi.filePath(); ++count; } } if (count == 1) editable.file = Utils::FileName::fromString(possibleFilePath); else qWarning() << "Could not find absolute location of file " << filePath; } } emit addTask(editable); } void AbstractProcessStep::outputAdded(const QString &string, ProjectExplorer::BuildStep::OutputFormat format) { emit addOutput(string, format, BuildStep::DontAppendNewline); } void AbstractProcessStep::slotProcessFinished(int, QProcess::ExitStatus) { QString line = QString::fromLocal8Bit(m_process->readAllStandardError()); if (!line.isEmpty()) stdError(line); line = QString::fromLocal8Bit(m_process->readAllStandardOutput()); if (!line.isEmpty()) stdOutput(line); m_eventLoop->exit(0); }