aboutsummaryrefslogtreecommitdiffstats
path: root/src/lib/corelib/buildgraph/command.cpp
diff options
context:
space:
mode:
authorChristian Kandeler <christian.kandeler@digia.com>2014-01-09 17:50:40 +0100
committerJoerg Bornemann <joerg.bornemann@digia.com>2014-01-10 18:11:22 +0100
commit81af9acaa295a574c1cb5e6714725197dac7f530 (patch)
treecc8c94467f49a7d267e5249f624874feecc7eed4 /src/lib/corelib/buildgraph/command.cpp
parent2fe25eb3f20ffb4e58cb559f2bcb9950c963290a (diff)
Move Qt profile setup into a dedicated library.
Otherwise all changes to the implementation will have to be duplicated in IDEs. Change-Id: I61e6d4fa1ee9b724eb5d9de9f233dc915a6c8bc3 Reviewed-by: Joerg Bornemann <joerg.bornemann@digia.com>
Diffstat (limited to 'src/lib/corelib/buildgraph/command.cpp')
-rw-r--r--src/lib/corelib/buildgraph/command.cpp287
1 files changed, 287 insertions, 0 deletions
diff --git a/src/lib/corelib/buildgraph/command.cpp b/src/lib/corelib/buildgraph/command.cpp
new file mode 100644
index 000000000..8e30906ee
--- /dev/null
+++ b/src/lib/corelib/buildgraph/command.cpp
@@ -0,0 +1,287 @@
+/****************************************************************************
+**
+** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
+** Contact: http://www.qt-project.org/legal
+**
+** This file is part of the Qt Build Suite.
+**
+** 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 "command.h"
+#include <logging/translator.h>
+#include <tools/qbsassert.h>
+#include <tools/hostosinfo.h>
+
+#include <QScriptEngine>
+#include <QScriptValueIterator>
+#include <QSet>
+
+namespace qbs {
+namespace Internal {
+
+AbstractCommand::AbstractCommand()
+ : m_description(defaultDescription()),
+ m_highlight(defaultHighLight()),
+ m_silent(defaultIsSilent())
+{
+}
+
+AbstractCommand::~AbstractCommand()
+{
+}
+
+AbstractCommand *AbstractCommand::createByType(AbstractCommand::CommandType commandType)
+{
+ switch (commandType) {
+ case AbstractCommand::ProcessCommandType:
+ return new ProcessCommand;
+ case AbstractCommand::JavaScriptCommandType:
+ return new JavaScriptCommand;
+ }
+ qFatal("%s: Invalid command type %d", Q_FUNC_INFO, commandType);
+ return 0;
+}
+
+bool AbstractCommand::equals(const AbstractCommand *other) const
+{
+ return m_description == other->m_description && m_highlight == other->m_highlight
+ && m_silent == other->m_silent && type() == other->type();
+}
+
+void AbstractCommand::fillFromScriptValue(const QScriptValue *scriptValue, const CodeLocation &codeLocation)
+{
+ m_description = scriptValue->property("description").toString();
+ m_highlight = scriptValue->property("highlight").toString();
+ m_silent = scriptValue->property("silent").toBool();
+ m_codeLocation = codeLocation;
+}
+
+void AbstractCommand::load(QDataStream &s)
+{
+ s >> m_description >> m_highlight >> m_silent >> m_codeLocation;
+}
+
+void AbstractCommand::store(QDataStream &s)
+{
+ s << m_description << m_highlight << m_silent << m_codeLocation;
+}
+
+static QScriptValue js_CommandBase(QScriptContext *context, QScriptEngine *engine)
+{
+ QScriptValue cmd = context->thisObject();
+ QBS_ASSERT(context->isCalledAsConstructor(), cmd = engine->newObject());
+ cmd.setProperty("description", engine->toScriptValue(AbstractCommand::defaultDescription()));
+ cmd.setProperty("highlight", engine->toScriptValue(AbstractCommand::defaultHighLight()));
+ cmd.setProperty("silent", engine->toScriptValue(AbstractCommand::defaultIsSilent()));
+ return cmd;
+}
+
+static QScriptValue js_Command(QScriptContext *context, QScriptEngine *engine)
+{
+ if (Q_UNLIKELY(!context->isCalledAsConstructor()))
+ return context->throwError(Tr::tr("Command constructor called without new."));
+
+ static ProcessCommand commandPrototype;
+
+ QScriptValue program = context->argument(0);
+ if (program.isUndefined())
+ program = engine->toScriptValue(commandPrototype.program());
+ QScriptValue arguments = context->argument(1);
+ if (arguments.isUndefined())
+ arguments = engine->toScriptValue(commandPrototype.arguments());
+ QScriptValue cmd = js_CommandBase(context, engine);
+ cmd.setProperty("className", engine->toScriptValue(QString("Command")));
+ cmd.setProperty("program", program);
+ cmd.setProperty("arguments", arguments);
+ cmd.setProperty("workingDir", engine->toScriptValue(commandPrototype.workingDir()));
+ cmd.setProperty("maxExitCode", engine->toScriptValue(commandPrototype.maxExitCode()));
+ cmd.setProperty("stdoutFilterFunction", engine->toScriptValue(commandPrototype.stdoutFilterFunction()));
+ cmd.setProperty("stderrFilterFunction", engine->toScriptValue(commandPrototype.stderrFilterFunction()));
+ cmd.setProperty("responseFileThreshold", engine->toScriptValue(commandPrototype.responseFileThreshold()));
+ cmd.setProperty("responseFileUsagePrefix", engine->toScriptValue(commandPrototype.responseFileUsagePrefix()));
+ cmd.setProperty("environment", engine->toScriptValue(commandPrototype.environment().toStringList()));
+ return cmd;
+}
+
+
+void ProcessCommand::setupForJavaScript(QScriptValue targetObject)
+{
+ QBS_CHECK(targetObject.isObject());
+ QScriptValue ctor = targetObject.engine()->newFunction(js_Command, 2);
+ targetObject.setProperty("Command", ctor);
+}
+
+ProcessCommand::ProcessCommand()
+ : m_maxExitCode(0)
+ , m_responseFileThreshold(HostOsInfo::isWindowsHost() ? 32000 : -1)
+{
+}
+
+void ProcessCommand::getEnvironmentFromList(const QStringList &envList)
+{
+ m_environment.clear();
+ foreach (const QString &env, envList) {
+ const int equalsIndex = env.indexOf(QLatin1Char('='));
+ if (equalsIndex <= 0 || equalsIndex == env.count() - 1)
+ continue;
+ const QString &var = env.left(equalsIndex);
+ const QString &value = env.mid(equalsIndex + 1);
+ m_environment.insert(var, value);
+ }
+}
+
+bool ProcessCommand::equals(const AbstractCommand *otherAbstractCommand) const
+{
+ if (!AbstractCommand::equals(otherAbstractCommand))
+ return false;
+ const ProcessCommand * const other = static_cast<const ProcessCommand *>(otherAbstractCommand);
+ return m_program == other->m_program
+ && m_arguments == other->m_arguments
+ && m_workingDir == other->m_workingDir
+ && m_maxExitCode == other->m_maxExitCode
+ && m_stdoutFilterFunction == other->m_stdoutFilterFunction
+ && m_stderrFilterFunction == other->m_stderrFilterFunction
+ && m_responseFileThreshold == other->m_responseFileThreshold
+ && m_responseFileUsagePrefix == other->m_responseFileUsagePrefix
+ && m_environment == other->m_environment;
+}
+
+void ProcessCommand::fillFromScriptValue(const QScriptValue *scriptValue, const CodeLocation &codeLocation)
+{
+ AbstractCommand::fillFromScriptValue(scriptValue, codeLocation);
+ m_program = scriptValue->property("program").toString();
+ m_arguments = scriptValue->property("arguments").toVariant().toStringList();
+ m_workingDir = scriptValue->property("workingDirectory").toString();
+ m_maxExitCode = scriptValue->property("maxExitCode").toInt32();
+ m_stdoutFilterFunction = scriptValue->property("stdoutFilterFunction").toString();
+ m_stderrFilterFunction = scriptValue->property("stderrFilterFunction").toString();
+ m_responseFileThreshold = scriptValue->property("responseFileThreshold").toInt32();
+ m_responseFileUsagePrefix = scriptValue->property("responseFileUsagePrefix").toString();
+ QStringList envList = scriptValue->property(QLatin1String("environment")).toVariant()
+ .toStringList();
+ getEnvironmentFromList(envList);
+}
+
+void ProcessCommand::load(QDataStream &s)
+{
+ AbstractCommand::load(s);
+ QStringList envList;
+ s >> m_program
+ >> m_arguments
+ >> envList
+ >> m_workingDir
+ >> m_maxExitCode
+ >> m_stdoutFilterFunction
+ >> m_stderrFilterFunction
+ >> m_responseFileThreshold
+ >> m_responseFileUsagePrefix;
+ getEnvironmentFromList(envList);
+}
+
+void ProcessCommand::store(QDataStream &s)
+{
+ AbstractCommand::store(s);
+ s << m_program
+ << m_arguments
+ << m_environment.toStringList()
+ << m_workingDir
+ << m_maxExitCode
+ << m_stdoutFilterFunction
+ << m_stderrFilterFunction
+ << m_responseFileThreshold
+ << m_responseFileUsagePrefix;
+}
+
+static QScriptValue js_JavaScriptCommand(QScriptContext *context, QScriptEngine *engine)
+{
+ if (Q_UNLIKELY(!context->isCalledAsConstructor()))
+ return context->throwError(Tr::tr("JavaScriptCommand constructor called without new."));
+ if (Q_UNLIKELY(context->argumentCount() != 0)) {
+ return context->throwError(QScriptContext::SyntaxError,
+ "JavaScriptCommand c'tor doesn't take arguments.");
+ }
+
+ static JavaScriptCommand commandPrototype;
+ QScriptValue cmd = js_CommandBase(context, engine);
+ cmd.setProperty("className", engine->toScriptValue(QString("JavaScriptCommand")));
+ cmd.setProperty("sourceCode", engine->toScriptValue(commandPrototype.sourceCode()));
+ return cmd;
+}
+
+void JavaScriptCommand::setupForJavaScript(QScriptValue targetObject)
+{
+ QBS_CHECK(targetObject.isObject());
+ QScriptValue ctor = targetObject.engine()->newFunction(js_JavaScriptCommand, 0);
+ targetObject.setProperty("JavaScriptCommand", ctor);
+}
+
+JavaScriptCommand::JavaScriptCommand()
+{
+}
+
+bool JavaScriptCommand::equals(const AbstractCommand *otherAbstractCommand) const
+{
+ if (!AbstractCommand::equals(otherAbstractCommand))
+ return false;
+ const JavaScriptCommand * const other
+ = static_cast<const JavaScriptCommand *>(otherAbstractCommand);
+ return m_sourceCode == other->m_sourceCode
+ && m_properties == other->m_properties;
+}
+
+void JavaScriptCommand::fillFromScriptValue(const QScriptValue *scriptValue, const CodeLocation &codeLocation)
+{
+ AbstractCommand::fillFromScriptValue(scriptValue, codeLocation);
+ QScriptValue sourceCode = scriptValue->property("sourceCode");
+ if (sourceCode.isFunction())
+ m_sourceCode = "(" + sourceCode.toString() + ")()";
+ else
+ m_sourceCode = sourceCode.toString();
+ static QSet<QString> predefinedProperties = QSet<QString>()
+ << "description" << "highlight" << "silent" << "className" << "sourceCode";
+
+ QScriptValueIterator it(*scriptValue);
+ while (it.hasNext()) {
+ it.next();
+ if (predefinedProperties.contains(it.name()))
+ continue;
+ m_properties.insert(it.name(), it.value().toVariant());
+ }
+}
+
+void JavaScriptCommand::load(QDataStream &s)
+{
+ AbstractCommand::load(s);
+ s >> m_sourceCode
+ >> m_properties;
+}
+
+void JavaScriptCommand::store(QDataStream &s)
+{
+ AbstractCommand::store(s);
+ s << m_sourceCode
+ << m_properties;
+}
+
+} // namespace Internal
+} // namespace qbs