2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
**
|
2010-03-05 11:25:49 +01:00
|
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-06-17 00:01:27 +10:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Commercial Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
|
|
** accordance with the Qt Commercial License Agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and Nokia.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** 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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** If you are unsure which license is appropriate for your use, please
|
2009-08-14 09:30:56 +02:00
|
|
|
** contact the sales department at http://qt.nokia.com/contact.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 16:19:05 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "environment.h"
|
|
|
|
|
|
|
|
#include <QtCore/QProcess>
|
|
|
|
#include <QtCore/QDir>
|
|
|
|
#include <QtCore/QString>
|
|
|
|
|
2010-09-23 10:35:23 +02:00
|
|
|
using namespace Utils;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
QList<EnvironmentItem> EnvironmentItem::fromStringList(QStringList list)
|
|
|
|
{
|
|
|
|
QList<EnvironmentItem> result;
|
|
|
|
foreach (const QString &string, list) {
|
|
|
|
int pos = string.indexOf(QLatin1Char('='));
|
2008-12-09 11:07:24 +01:00
|
|
|
if (pos == -1) {
|
2010-02-01 12:43:56 +01:00
|
|
|
EnvironmentItem item(string, QString());
|
2008-12-02 12:01:29 +01:00
|
|
|
item.unset = true;
|
|
|
|
result.append(item);
|
|
|
|
} else {
|
|
|
|
EnvironmentItem item(string.left(pos), string.mid(pos+1));
|
|
|
|
result.append(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList EnvironmentItem::toStringList(QList<EnvironmentItem> list)
|
|
|
|
{
|
|
|
|
QStringList result;
|
|
|
|
foreach (const EnvironmentItem &item, list) {
|
2008-12-09 11:07:24 +01:00
|
|
|
if (item.unset)
|
2008-12-02 12:01:29 +01:00
|
|
|
result << QString(item.name);
|
|
|
|
else
|
|
|
|
result << QString(item.name + '=' + item.value);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Environment::Environment()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Environment::Environment(QStringList env)
|
|
|
|
{
|
2008-12-09 11:07:24 +01:00
|
|
|
foreach (const QString &s, env) {
|
2008-12-02 12:01:29 +01:00
|
|
|
int i = s.indexOf("=");
|
2008-12-09 11:07:24 +01:00
|
|
|
if (i >= 0) {
|
2008-12-02 12:01:29 +01:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
m_values.insert(s.left(i).toUpper(), s.mid(i+1));
|
|
|
|
#else
|
|
|
|
m_values.insert(s.left(i), s.mid(i+1));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-13 17:25:09 +01:00
|
|
|
QStringList Environment::toStringList() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
QStringList result;
|
2009-01-13 17:25:09 +01:00
|
|
|
const QMap<QString, QString>::const_iterator end = m_values.constEnd();
|
|
|
|
for (QMap<QString, QString>::const_iterator it = m_values.constBegin(); it != end; ++it) {
|
|
|
|
QString entry = it.key();
|
|
|
|
entry += QLatin1Char('=');
|
|
|
|
entry += it.value();
|
|
|
|
result.push_back(entry);
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Environment::set(const QString &key, const QString &value)
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
QString _key = key.toUpper();
|
|
|
|
#else
|
|
|
|
const QString &_key = key;
|
|
|
|
#endif
|
|
|
|
m_values.insert(_key, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Environment::unset(const QString &key)
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
QString _key = key.toUpper();
|
|
|
|
#else
|
|
|
|
const QString &_key = key;
|
|
|
|
#endif
|
|
|
|
m_values.remove(_key);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Environment::appendOrSet(const QString &key, const QString &value, const QString &sep)
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
QString _key = key.toUpper();
|
|
|
|
#else
|
|
|
|
const QString &_key = key;
|
|
|
|
#endif
|
2010-02-17 17:38:48 +01:00
|
|
|
QMap<QString, QString>::iterator it = m_values.find(key);
|
|
|
|
if (it == m_values.end()) {
|
2008-12-02 12:01:29 +01:00
|
|
|
m_values.insert(_key, value);
|
|
|
|
} else {
|
2010-02-17 17:38:48 +01:00
|
|
|
// Append unless it is already there
|
|
|
|
const QString toAppend = sep + value;
|
|
|
|
if (!it.value().endsWith(toAppend))
|
|
|
|
it.value().append(toAppend);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Environment::prependOrSet(const QString&key, const QString &value, const QString &sep)
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
QString _key = key.toUpper();
|
|
|
|
#else
|
|
|
|
const QString &_key = key;
|
|
|
|
#endif
|
2010-02-17 17:38:48 +01:00
|
|
|
QMap<QString, QString>::iterator it = m_values.find(key);
|
|
|
|
if (it == m_values.end()) {
|
2008-12-02 12:01:29 +01:00
|
|
|
m_values.insert(_key, value);
|
|
|
|
} else {
|
2010-02-17 17:38:48 +01:00
|
|
|
// Prepend unless it is already there
|
|
|
|
const QString toPrepend = value + sep;
|
|
|
|
if (!it.value().startsWith(toPrepend))
|
|
|
|
it.value().prepend(toPrepend);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Environment::appendOrSetPath(const QString &value)
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_WIN
|
2010-02-01 12:43:56 +01:00
|
|
|
const QChar sep = QLatin1Char(';');
|
2008-12-02 12:01:29 +01:00
|
|
|
#else
|
2010-02-01 12:43:56 +01:00
|
|
|
const QChar sep = QLatin1Char(':');
|
2008-12-02 12:01:29 +01:00
|
|
|
#endif
|
2010-02-01 12:43:56 +01:00
|
|
|
appendOrSet(QLatin1String("PATH"), QDir::toNativeSeparators(value), QString(sep));
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Environment::prependOrSetPath(const QString &value)
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_WIN
|
2010-02-01 12:43:56 +01:00
|
|
|
const QChar sep = QLatin1Char(';');
|
2008-12-02 12:01:29 +01:00
|
|
|
#else
|
2010-02-01 12:43:56 +01:00
|
|
|
const QChar sep = QLatin1Char(':');
|
2008-12-02 12:01:29 +01:00
|
|
|
#endif
|
2010-02-01 12:43:56 +01:00
|
|
|
prependOrSet(QLatin1String("PATH"), QDir::toNativeSeparators(value), QString(sep));
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Environment Environment::systemEnvironment()
|
|
|
|
{
|
|
|
|
return Environment(QProcess::systemEnvironment());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Environment::clear()
|
|
|
|
{
|
|
|
|
m_values.clear();
|
|
|
|
}
|
|
|
|
|
2010-09-23 13:08:03 +02:00
|
|
|
QString Environment::searchInPath(const QString &executable,
|
2010-10-18 11:10:55 +02:00
|
|
|
const QStringList &additionalDirs) const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-10-18 11:10:55 +02:00
|
|
|
QStringList execs;
|
|
|
|
execs << executable;
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
// Check all the executable extensions on windows:
|
|
|
|
QStringList extensions = value(QLatin1String("PATHEXT")).split(QLatin1Char(';'));
|
|
|
|
|
|
|
|
// .exe.bat is legal (and run when starting new.exe), so always go through the complete list once:
|
|
|
|
foreach (const QString &ext, extensions)
|
|
|
|
execs << executable + ext.toLower();
|
|
|
|
#endif
|
|
|
|
return searchInPath(execs, additionalDirs);
|
|
|
|
}
|
2010-09-23 13:08:03 +02:00
|
|
|
|
2010-10-18 11:10:55 +02:00
|
|
|
QString Environment::searchInPath(const QStringList &executables,
|
|
|
|
const QStringList &additionalDirs) const
|
|
|
|
{
|
|
|
|
foreach (const QString &executable, executables) {
|
|
|
|
QString exec = expandVariables(executable);
|
2010-09-23 13:08:03 +02:00
|
|
|
|
2010-10-18 11:10:55 +02:00
|
|
|
if (exec.isEmpty())
|
2010-09-23 13:08:03 +02:00
|
|
|
continue;
|
|
|
|
|
2010-10-18 11:10:55 +02:00
|
|
|
QFileInfo baseFi(exec);
|
|
|
|
if (baseFi.isAbsolute() && baseFi.exists())
|
|
|
|
return QDir::toNativeSeparators(exec);
|
|
|
|
|
|
|
|
// Check in directories:
|
|
|
|
foreach (const QString &dir, additionalDirs) {
|
|
|
|
if (dir.isEmpty())
|
|
|
|
continue;
|
|
|
|
QFileInfo fi(dir + QLatin1Char('/') + exec);
|
|
|
|
if (fi.isFile() && fi.isExecutable())
|
|
|
|
return fi.absoluteFilePath();
|
|
|
|
}
|
2010-09-23 13:08:03 +02:00
|
|
|
|
2010-10-18 11:10:55 +02:00
|
|
|
// Check in path:
|
|
|
|
const QChar slash = QLatin1Char('/');
|
|
|
|
if (exec.indexOf(slash) != -1)
|
|
|
|
continue;
|
|
|
|
foreach (const QString &p, path()) {
|
|
|
|
QString fp = p;
|
|
|
|
fp += slash;
|
|
|
|
fp += exec;
|
|
|
|
const QFileInfo fi(fp);
|
|
|
|
if (fi.exists())
|
|
|
|
return fi.absoluteFilePath();
|
|
|
|
}
|
2010-09-23 13:08:03 +02:00
|
|
|
}
|
2010-02-02 17:09:41 +01:00
|
|
|
return QString();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QStringList Environment::path() const
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_WIN
|
2010-02-01 12:43:56 +01:00
|
|
|
const QChar sep = QLatin1Char(';');
|
2008-12-02 12:01:29 +01:00
|
|
|
#else
|
2010-02-01 12:43:56 +01:00
|
|
|
const QChar sep = QLatin1Char(':');
|
2008-12-02 12:01:29 +01:00
|
|
|
#endif
|
2010-04-26 16:33:18 +02:00
|
|
|
return m_values.value(QLatin1String("PATH")).split(sep, QString::SkipEmptyParts);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString Environment::value(const QString &key) const
|
|
|
|
{
|
|
|
|
return m_values.value(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Environment::key(Environment::const_iterator it) const
|
|
|
|
{
|
|
|
|
return it.key();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Environment::value(Environment::const_iterator it) const
|
|
|
|
{
|
|
|
|
return it.value();
|
|
|
|
}
|
|
|
|
|
|
|
|
Environment::const_iterator Environment::constBegin() const
|
|
|
|
{
|
|
|
|
return m_values.constBegin();
|
|
|
|
}
|
|
|
|
|
|
|
|
Environment::const_iterator Environment::constEnd() const
|
|
|
|
{
|
|
|
|
return m_values.constEnd();
|
|
|
|
}
|
|
|
|
|
2010-10-18 21:14:45 +02:00
|
|
|
Environment::const_iterator Environment::constFind(const QString &name) const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
QMap<QString, QString>::const_iterator it = m_values.constFind(name);
|
2008-12-09 11:07:24 +01:00
|
|
|
if (it == m_values.constEnd())
|
2008-12-02 12:01:29 +01:00
|
|
|
return constEnd();
|
|
|
|
else
|
|
|
|
return it;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Environment::size() const
|
|
|
|
{
|
|
|
|
return m_values.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Environment::modify(const QList<EnvironmentItem> & list)
|
|
|
|
{
|
|
|
|
Environment resultEnvironment = *this;
|
|
|
|
foreach (const EnvironmentItem &item, list) {
|
2008-12-09 11:07:24 +01:00
|
|
|
if (item.unset) {
|
2008-12-02 12:01:29 +01:00
|
|
|
resultEnvironment.unset(item.name);
|
|
|
|
} else {
|
|
|
|
// TODO use variable expansion
|
|
|
|
QString value = item.value;
|
2008-12-09 11:07:24 +01:00
|
|
|
for (int i=0; i < value.size(); ++i) {
|
|
|
|
if (value.at(i) == QLatin1Char('$')) {
|
|
|
|
if ((i + 1) < value.size()) {
|
2008-12-02 12:01:29 +01:00
|
|
|
const QChar &c = value.at(i+1);
|
|
|
|
int end = -1;
|
|
|
|
if (c == '(')
|
|
|
|
end = value.indexOf(')', i);
|
2008-12-09 11:07:24 +01:00
|
|
|
else if (c == '{')
|
2008-12-02 12:01:29 +01:00
|
|
|
end = value.indexOf('}', i);
|
2008-12-09 11:07:24 +01:00
|
|
|
if (end != -1) {
|
2008-12-02 12:01:29 +01:00
|
|
|
const QString &name = value.mid(i+2, end-i-2);
|
2010-10-18 21:14:45 +02:00
|
|
|
Environment::const_iterator it = constFind(name);
|
2008-12-09 11:07:24 +01:00
|
|
|
if (it != constEnd())
|
2008-12-02 12:01:29 +01:00
|
|
|
value.replace(i, end-i+1, it.value());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
resultEnvironment.set(item.name, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*this = resultEnvironment;
|
|
|
|
}
|
|
|
|
|
2010-04-19 18:13:23 +02:00
|
|
|
bool Environment::hasKey(const QString &key)
|
|
|
|
{
|
|
|
|
return m_values.contains(key);
|
|
|
|
}
|
|
|
|
|
2010-01-11 10:25:15 +01:00
|
|
|
bool Environment::operator!=(const Environment &other) const
|
2009-03-19 18:15:33 +01:00
|
|
|
{
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
|
2010-01-11 10:25:15 +01:00
|
|
|
bool Environment::operator==(const Environment &other) const
|
2009-03-19 18:15:33 +01:00
|
|
|
{
|
|
|
|
return m_values == other.m_values;
|
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
QStringList Environment::parseCombinedArgString(const QString &program)
|
|
|
|
{
|
|
|
|
QStringList args;
|
|
|
|
QString tmp;
|
|
|
|
int quoteCount = 0;
|
|
|
|
bool inQuote = false;
|
|
|
|
|
|
|
|
// handle quoting. tokens can be surrounded by double quotes
|
|
|
|
// "hello world". three consecutive double quotes represent
|
|
|
|
// the quote character itself.
|
|
|
|
for (int i = 0; i < program.size(); ++i) {
|
|
|
|
if (program.at(i) == QLatin1Char('"')) {
|
|
|
|
++quoteCount;
|
|
|
|
if (quoteCount == 3) {
|
|
|
|
// third consecutive quote
|
|
|
|
quoteCount = 0;
|
|
|
|
tmp += program.at(i);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (quoteCount) {
|
|
|
|
if (quoteCount == 1)
|
|
|
|
inQuote = !inQuote;
|
|
|
|
quoteCount = 0;
|
|
|
|
}
|
|
|
|
if (!inQuote && program.at(i).isSpace()) {
|
|
|
|
if (!tmp.isEmpty()) {
|
|
|
|
args += tmp;
|
|
|
|
tmp.clear();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
tmp += program.at(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!tmp.isEmpty())
|
|
|
|
args += tmp;
|
|
|
|
return args;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Environment::joinArgumentList(const QStringList &arguments)
|
|
|
|
{
|
|
|
|
QString result;
|
2010-02-01 12:43:56 +01:00
|
|
|
const QChar doubleQuote = QLatin1Char('"');
|
2008-12-09 11:07:24 +01:00
|
|
|
foreach (QString arg, arguments) {
|
2008-12-02 12:01:29 +01:00
|
|
|
if (!result.isEmpty())
|
|
|
|
result += QLatin1Char(' ');
|
2010-02-01 12:43:56 +01:00
|
|
|
arg.replace(QString(doubleQuote), QLatin1String("\"\"\""));
|
|
|
|
if (arg.contains(QLatin1Char(' '))) {
|
|
|
|
arg.insert(0, doubleQuote);
|
|
|
|
arg += doubleQuote;
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
result += arg;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2010-10-06 16:11:07 +02:00
|
|
|
enum State { BASE, VARIABLE, OPTIONALVARIABLEBRACE, STRING };
|
2010-09-22 15:10:20 +02:00
|
|
|
|
|
|
|
/** Expand environment variables in a string.
|
|
|
|
*
|
|
|
|
* Environment variables are accepted in the following forms:
|
|
|
|
* $SOMEVAR, ${SOMEVAR} and %SOMEVAR%.
|
|
|
|
*
|
|
|
|
* Strings enclosed in '"' characters do not get varaibles
|
2010-10-06 16:11:07 +02:00
|
|
|
* substituted.
|
2010-09-22 15:10:20 +02:00
|
|
|
*/
|
|
|
|
QString Environment::expandVariables(const QString &input) const
|
|
|
|
{
|
|
|
|
QString result;
|
|
|
|
QString variable;
|
|
|
|
QChar endVariable;
|
|
|
|
State state = BASE;
|
|
|
|
|
|
|
|
int length = input.count();
|
|
|
|
for (int i = 0; i < length; ++i) {
|
|
|
|
QChar c = input.at(i);
|
|
|
|
if (state == BASE) {
|
2010-10-06 16:11:07 +02:00
|
|
|
if (c == '$') {
|
2010-09-22 15:10:20 +02:00
|
|
|
state = OPTIONALVARIABLEBRACE;
|
|
|
|
variable.clear();
|
|
|
|
endVariable = QChar(0);
|
|
|
|
} else if (c == '%') {
|
|
|
|
state = VARIABLE;
|
|
|
|
variable.clear();
|
|
|
|
endVariable = '%';
|
|
|
|
} else if (c == '\"') {
|
|
|
|
state = STRING;
|
|
|
|
result += c;
|
|
|
|
} else {
|
|
|
|
result += c;
|
|
|
|
}
|
|
|
|
} else if (state == VARIABLE) {
|
|
|
|
if (c == endVariable) {
|
|
|
|
result += value(variable);
|
|
|
|
state = BASE;
|
|
|
|
} else if (c.isLetterOrNumber() || c == '_') {
|
|
|
|
variable += c;
|
|
|
|
} else {
|
|
|
|
result += value(variable);
|
|
|
|
result += c;
|
|
|
|
state = BASE;
|
|
|
|
}
|
|
|
|
} else if (state == OPTIONALVARIABLEBRACE) {
|
|
|
|
if (c == '{')
|
|
|
|
endVariable = '}';
|
|
|
|
else
|
|
|
|
variable = c;
|
|
|
|
state = VARIABLE;
|
|
|
|
} else if (state == STRING) {
|
2010-10-06 16:11:07 +02:00
|
|
|
if (c == '\"') {
|
2010-09-22 15:10:20 +02:00
|
|
|
state = BASE;
|
|
|
|
result += c;
|
|
|
|
} else {
|
|
|
|
result += c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (state == VARIABLE)
|
|
|
|
result += value(variable);
|
|
|
|
return result;
|
|
|
|
}
|
2010-09-23 13:08:03 +02:00
|
|
|
|
|
|
|
QStringList Environment::expandVariables(const QStringList &variables) const
|
|
|
|
{
|
|
|
|
QStringList results;
|
|
|
|
foreach (const QString & i, variables)
|
|
|
|
results << expandVariables(i);
|
|
|
|
return results;
|
|
|
|
}
|