/************************************************************************** ** ** Copyright (C) 2012-2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the Qt Installer Framework. ** ** $QT_BEGIN_LICENSE:LGPL$ ** 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. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** **************************************************************************/ #include "utils.h" #include #include #include #include #include #if defined(Q_OS_WIN32) || defined(Q_OS_WINCE) # include "qt_windows.h" #endif #include #include #include #ifdef Q_OS_WIN #include // for Sleep #endif #ifdef Q_OS_UNIX #include #include #include #endif namespace { void sleepCopiedFromQTest(int ms) { if (ms < 0) return; #ifdef Q_OS_WIN Sleep(uint(ms)); #else struct timespec ts = { ms / 1000, (ms % 1000) * 1000 * 1000 }; nanosleep(&ts, NULL); #endif } } void QInstaller::uiDetachedWait(int ms) { QTime timer; timer.start(); do { QCoreApplication::processEvents(QEventLoop::AllEvents, ms); sleepCopiedFromQTest(10); } while (timer.elapsed() < ms); } static bool verb = false; void QInstaller::setVerbose(bool v) { verb = v; } bool QInstaller::isVerbose() { return verb; } #ifdef Q_OS_WIN class debugstream : public std::ostream { class buf : public std::stringbuf { public: buf() {} int sync() { std::string s = str(); if (s[s.length() - 1] == '\n' ) s[s.length() - 1] = '\0'; // remove \n std::cout << s << std::endl; str(std::string()); return 0; } }; public: debugstream() : std::ostream(&b) {} private: buf b; }; #endif std::ostream &QInstaller::stdverbose() { static std::fstream null; #ifdef Q_OS_WIN static debugstream stream; #else static std::ostream& stream = std::cout; #endif if (verb) return stream; return null; } std::ostream &QInstaller::operator<<(std::ostream &os, const QString &string) { return os << qPrintable(string); } QByteArray QInstaller::calculateHash(QIODevice *device, QCryptographicHash::Algorithm algo) { Q_ASSERT(device); QCryptographicHash hash(algo); static QByteArray buffer(1024 * 1024, '\0'); while (true) { const qint64 numRead = device->read(buffer.data(), buffer.size()); if (numRead <= 0) return hash.result(); hash.addData(buffer.constData(), numRead); } return QByteArray(); // never reached } QByteArray QInstaller::calculateHash(const QString &path, QCryptographicHash::Algorithm algo) { QFile file(path); if (!file.open(QIODevice::ReadOnly)) return QByteArray(); return calculateHash(&file, algo); } QString QInstaller::replaceVariables(const QHash &vars, const QString &str) { QString res; int pos = 0; while (true) { int pos1 = str.indexOf(QLatin1Char('@'), pos); if (pos1 == -1) break; int pos2 = str.indexOf(QLatin1Char('@'), pos1 + 1); if (pos2 == -1) break; res += str.mid(pos, pos1 - pos); QString name = str.mid(pos1 + 1, pos2 - pos1 - 1); res += vars.value(name); pos = pos2 + 1; } res += str.mid(pos); return res; } QString QInstaller::replaceWindowsEnvironmentVariables(const QString &str) { const QProcessEnvironment env = QProcessEnvironment::systemEnvironment(); QString res; int pos = 0; while (true) { int pos1 = str.indexOf(QLatin1Char( '%'), pos); if (pos1 == -1) break; int pos2 = str.indexOf(QLatin1Char( '%'), pos1 + 1); if (pos2 == -1) break; res += str.mid(pos, pos1 - pos); QString name = str.mid(pos1 + 1, pos2 - pos1 - 1); res += env.value(name); pos = pos2 + 1; } res += str.mid(pos); return res; } QInstaller::VerboseWriter::VerboseWriter(QObject *parent) : QObject(parent) { preFileBuffer.open(QIODevice::ReadWrite); stream.setDevice(&preFileBuffer); } QInstaller::VerboseWriter::~VerboseWriter() { stream.flush(); if (logFileName.isEmpty()) // binarycreator return; //if the installer installed nothing - there is no target directory - where the logfile can be saved if (!QFileInfo(logFileName).absoluteDir().exists()) return; QFile output(logFileName); if (output.open(QIODevice::ReadWrite | QIODevice::Append | QIODevice::Text)) { QString logInfo; logInfo += QLatin1String("*************************************"); logInfo += QLatin1String("Invoked:") + QDateTime::currentDateTime().toString(); output.write(logInfo.toLocal8Bit()); output.write(preFileBuffer.data()); output.close(); } stream.setDevice(0); } void QInstaller::VerboseWriter::setOutputStream(const QString &fileName) { logFileName = fileName; } Q_GLOBAL_STATIC(QInstaller::VerboseWriter, verboseWriter) QInstaller::VerboseWriter *QInstaller::VerboseWriter::instance() { return verboseWriter(); } QInstaller::VerboseWriter &QInstaller::verbose() { return *verboseWriter(); } #ifdef Q_OS_WIN // taken from qcoreapplication_p.h template static QVector qWinCmdLine(Char *cmdParam, int length, int &argc) { QVector argv(8); Char *p = cmdParam; Char *p_end = p + length; argc = 0; while (*p && p < p_end) { // parse cmd line arguments while (QChar((short)(*p)).isSpace()) // skip white space p++; if (*p && p < p_end) { // arg starts int quote; Char *start, *r; if (*p == Char('\"') || *p == Char('\'')) { // " or ' quote quote = *p; start = ++p; } else { quote = 0; start = p; } r = start; while (*p && p < p_end) { if (quote) { if (*p == quote) { p++; if (QChar((short)(*p)).isSpace()) break; quote = 0; } } if (*p == '\\') { // escape char? p++; if (*p == Char('\"') || *p == Char('\'')) ; // yes else p--; // treat \ literally } else { if (!quote && (*p == Char('\"') || *p == Char('\''))) { // " or ' quote quote = *p++; continue; } else if (QChar((short)(*p)).isSpace() && !quote) break; } if (*p) *r++ = *p++; } if (*p && p < p_end) p++; *r = Char('\0'); if (argc >= (int)argv.size()-1) // expand array argv.resize(argv.size()*2); argv[argc++] = start; } } argv[argc] = 0; return argv; } QStringList QInstaller::parseCommandLineArgs(int argc, char **argv) { Q_UNUSED(argc) Q_UNUSED(argv) QStringList arguments; QString cmdLine = QString::fromWCharArray(GetCommandLine()); QVector args = qWinCmdLine((wchar_t *)cmdLine.utf16(), cmdLine.length(), argc); for (int a = 0; a < argc; ++a) arguments << QString::fromWCharArray(args[a]); return arguments; } #else QStringList QInstaller::parseCommandLineArgs(int argc, char **argv) { QStringList arguments; for (int a = 0; a < argc; ++a) arguments << QString::fromLocal8Bit(argv[a]); return arguments; } #endif #ifdef Q_OS_WIN // taken from qprocess_win.cpp static QString qt_create_commandline(const QString &program, const QStringList &arguments) { QString args; if (!program.isEmpty()) { QString programName = program; if (!programName.startsWith(QLatin1Char('\"')) && !programName.endsWith(QLatin1Char('\"')) && programName.contains(QLatin1Char(' '))) { programName = QLatin1Char('\"') + programName + QLatin1Char('\"'); } programName.replace(QLatin1Char('/'), QLatin1Char('\\')); // add the program as the first arg ... it works better args = programName + QLatin1Char(' '); } for (int i = 0; i < arguments.size(); ++i) { QString tmp = arguments.at(i); // in the case of \" already being in the string the \ must also be escaped tmp.replace(QLatin1String("\\\""), QLatin1String("\\\\\"")); // escape a single " because the arguments will be parsed tmp.replace(QLatin1Char('\"'), QLatin1String("\\\"")); if (tmp.isEmpty() || tmp.contains(QLatin1Char(' ')) || tmp.contains(QLatin1Char('\t'))) { // The argument must not end with a \ since this would be interpreted // as escaping the quote -- rather put the \ behind the quote: e.g. // rather use "foo"\ than "foo\" QString endQuote(QLatin1Char('\"')); int i = tmp.length(); while (i > 0 && tmp.at(i - 1) == QLatin1Char('\\')) { --i; endQuote += QLatin1Char('\\'); } args += QLatin1String(" \"") + tmp.left(i) + endQuote; } else { args += QLatin1Char(' ') + tmp; } } return args; } QString QInstaller::createCommandline(const QString &program, const QStringList &arguments) { return qt_create_commandline(program, arguments); } #endif