summaryrefslogtreecommitdiffstats
path: root/tools/getrepositorycontent/main.cpp
diff options
context:
space:
mode:
authorTim Jenssen <tim.jenssen@digia.com>2013-04-09 10:21:54 +0200
committerTim Jenssen <tim.jenssen@digia.com>2013-04-09 11:03:50 +0200
commitac2319b4f0e67da02880bdf1f8832cf3f527c252 (patch)
treedc8011d6ca728d70c9bffc6df784d370d7f3605b /tools/getrepositorycontent/main.cpp
parent3ba4a78b39609c4568d6357b092790a78f919fe9 (diff)
create getrepositorycontent tool
It is a very useful tool to get repositories locally for testing. Change-Id: Ia83841c3e5c96ab369c35580dd98141cf22ed643 Reviewed-by: Karsten Heimrich <karsten.heimrich@digia.com>
Diffstat (limited to 'tools/getrepositorycontent/main.cpp')
-rw-r--r--tools/getrepositorycontent/main.cpp450
1 files changed, 450 insertions, 0 deletions
diff --git a/tools/getrepositorycontent/main.cpp b/tools/getrepositorycontent/main.cpp
new file mode 100644
index 000000000..66a2ec43f
--- /dev/null
+++ b/tools/getrepositorycontent/main.cpp
@@ -0,0 +1,450 @@
+/**************************************************************************
+**
+** Copyright (C) 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 "downloader.h"
+#include "domnodedebugstreamoperator.h"
+
+#include <globals.h>
+#include <init.h>
+#include <fileutils.h>
+#include <lib7z_facade.h>
+#include <utils.h>
+
+#include <QCoreApplication>
+#include <QFile>
+#include <QUrl>
+#include <QString>
+#include <QDomDocument>
+#include <QDomElement>
+#include <QDomNodeList>
+#include <QStringList>
+#include <QDebug>
+#include <QFileInfo>
+#include <QDir>
+#include <QDirIterator>
+#include <QDebug>
+
+#include <iostream>
+
+static void printUsage()
+{
+ const QString appName = QFileInfo( QCoreApplication::applicationFilePath() ).fileName();
+ std::cout << "Usage: " << qPrintable(appName)
+ << " --url <repository_url> --repository <empty_repository_dir> --packages <empty_packages_dir>" << std::endl;
+ std::cout << " --url URL to fetch all the content from." << std::endl;
+ std::cout << " --repository Target directory for the repository content." << std::endl;
+ std::cout << " --packages The packages target directory where it creates the needed content to create new installers or repositories." << std::endl;
+ std::cout << " --clean Removes all the content if there is an existing repository or packages dir" << std::endl;
+
+ std::cout << "Example:" << std::endl;
+ std::cout << " " << qPrintable(appName) << " --url http://www.example.com/repository/" <<
+ " --repository repository --packages packages" << std::endl;
+}
+
+// this should be a new class which uses XmlStreamReader instead of DomDocument
+// should be implicit shared, see repository class
+// maybe we can use some code from persistentdata in qtcreator
+class ComponentData {
+ public:
+ ComponentData() {}
+ ComponentData(const QString &/*xmlData*/) {}
+
+ QVariant attributeValue(const QString &key, const QString &attribute, const QVariant &defaultValue = QVariant()) {
+ Q_UNUSED(key)
+ Q_UNUSED(attribute)
+ return defaultValue;
+ }
+
+ QVariant value(const QString &key, const QVariant &defaultValue = QVariant()) {
+ Q_UNUSED(defaultValue)
+ // just use the quick dirty hack added members
+ if (key == QLatin1String("Script"))
+ return m_script;
+ if (key == QLatin1String("Version"))
+ return m_version;
+ return QVariant();
+ }
+ QString textValue(const QString &key, const QString &defaultValue = QString()) {
+ return value(key, defaultValue).toString();
+ }
+
+ // quick dirty hack added public members
+ public:
+ QDomDocument m_packageXml;
+ QStringList m_downloadDownloadableArchives;
+ QString m_script;
+ QString m_version;
+};
+
+static void downloadFile(const QUrl &source, const QString &target)
+{
+ QEventLoop downloadEventLoop;
+ Downloader downloader(source, target);
+ QObject::connect(&downloader, SIGNAL(finished()), &downloadEventLoop, SLOT(quit()));
+ downloader.run();
+ downloadEventLoop.exec();
+}
+
+QHash<QString, ComponentData> downLoadRepository(const QString &repositoryUrl, const QString &repositoryTarget)
+{
+ const QString updatesXmlFileName = QLatin1String("Updates.xml");
+ QHash<QString, ComponentData> componentDataHash;
+
+ const QUrl updatesXmlUrl(QString::fromLatin1("%1/%2").arg(repositoryUrl, updatesXmlFileName));
+
+ downloadFile(updatesXmlUrl, QDir(repositoryTarget).filePath(updatesXmlFileName));
+
+ QFile updatesFile(QDir(repositoryTarget).filePath(updatesXmlFileName));
+ if (!updatesFile.exists()) {
+ qDebug() << "could not download the file:" << updatesXmlUrl.toString();
+ return componentDataHash;
+ } else
+ qDebug() << "file downloaded to location:" << QDir(repositoryTarget).filePath(updatesXmlFileName);
+ if (!updatesFile.open(QIODevice::ReadOnly)) {
+ qDebug() << QString::fromLatin1("Could not open Updates.xml for reading: %1").arg(updatesFile
+ .errorString()) ;
+ return componentDataHash;
+ }
+
+ QStringList ignoreTagList;
+ ignoreTagList << QLatin1String("Name");
+ ignoreTagList << QLatin1String("ReleaseDate");
+ ignoreTagList << QLatin1String("SHA1");
+ ignoreTagList << QLatin1String("UpdateFile");
+ QStringList fileTagList;
+ fileTagList << QLatin1String("DownloadableArchives");
+
+
+ QDomDocument updatesXml;
+ QString error;
+ int line = 0;
+ int column = 0;
+ if (!updatesXml.setContent( &updatesFile, &error, &line, &column)) {
+ qWarning() << QString::fromLatin1("Could not parse component index: %1:%2: %3")
+ .arg(QString::number(line), QString::number(column), error);
+ return componentDataHash;
+ }
+
+ QDomNode packageUpdateDomNode = updatesXml.firstChildElement(QLatin1String("Updates")).firstChildElement(
+ QLatin1String("PackageUpdate"));
+ while (!packageUpdateDomNode.isNull()) {
+
+ if (packageUpdateDomNode.nodeName() == QLatin1String("PackageUpdate")) {
+ QDomNode packageUpdateEntry = packageUpdateDomNode.firstChild();
+ QString currentPackageName;
+ ComponentData currentComponentData;
+ // creating the package.xml for later use
+ QDomElement currentNewPackageElement = currentComponentData.m_packageXml.createElement(
+ QLatin1String("Package"));
+ while (!packageUpdateEntry.isNull()) {
+ // do name first before ignore filters the name out
+ if (packageUpdateEntry.nodeName() == QLatin1String("Name")) {
+ currentPackageName = packageUpdateEntry.toElement().text();
+ }
+ if (ignoreTagList.contains(packageUpdateEntry.nodeName())) {
+ packageUpdateEntry = packageUpdateEntry.nextSibling();
+ continue;
+ }
+ if (packageUpdateEntry.nodeName() == QLatin1String("Script")) {
+ currentComponentData.m_script = packageUpdateEntry.toElement().text();
+ }
+ if (packageUpdateEntry.nodeName() == QLatin1String("Version")) {
+ currentComponentData.m_version = packageUpdateEntry.toElement().text();
+ currentComponentData.m_downloadDownloadableArchives.append(
+ currentComponentData.m_version + QLatin1String("meta.7z"));
+ }
+
+ if (packageUpdateEntry.nodeName() == QLatin1String("DownloadableArchives")) {
+ QStringList tDownloadList = packageUpdateEntry.toElement().text().split(
+ QInstaller::commaRegExp(), QString::SkipEmptyParts);
+ foreach (const QString &download, tDownloadList) {
+ currentComponentData.m_downloadDownloadableArchives.append(
+ currentComponentData.m_version + download);
+ currentComponentData.m_downloadDownloadableArchives.append(
+ currentComponentData.m_version + download + QLatin1String(".sha1"));
+ }
+ }
+
+ currentNewPackageElement.appendChild(packageUpdateEntry.cloneNode(true));
+ packageUpdateEntry = packageUpdateEntry.nextSibling();
+ }
+ currentComponentData.m_packageXml.appendChild(currentNewPackageElement);
+ Q_ASSERT(!currentComponentData.m_packageXml.toString().isEmpty());
+ componentDataHash.insert(currentPackageName, currentComponentData);
+ } else {
+ qWarning() << QString::fromLatin1("Unknown elment '%1'").arg(packageUpdateDomNode.nodeName(),
+ QFileInfo(updatesXmlFileName).absoluteFilePath());
+ }
+ packageUpdateDomNode = packageUpdateDomNode.nextSibling();
+ }
+
+ QHashIterator<QString, ComponentData> itComponentData(componentDataHash);
+ while (itComponentData.hasNext()) {
+ itComponentData.next();
+ QString componentDirectory = QDir(repositoryTarget).filePath(itComponentData.key());
+ if (!QDir().mkpath(componentDirectory))
+ qWarning() << "couldn't create:" << componentDirectory;
+
+ foreach (const QString &download, itComponentData.value().m_downloadDownloadableArchives) {
+ const QString fileTarget(componentDirectory + QDir::separator() + download);
+ const QUrl downloadUrl(repositoryUrl + QLatin1String("/") + itComponentData.key() + QLatin1String("/") + download);
+ downloadFile(downloadUrl, fileTarget);
+ }
+ }
+ return componentDataHash;
+}
+
+bool extractFile(const QString &source, const QString &target)
+{
+ if (!Lib7z::isSupportedArchive(source)) {
+ qWarning() << source << "is not a supported archive";
+ }
+
+ QFile archive(source);
+ if (archive.open(QIODevice::ReadOnly)) {
+ try {
+ Lib7z::extractArchive(&archive, target);
+ } catch (const Lib7z::SevenZipException& e) {
+ qWarning() << QString::fromLatin1("Error while extracting %1: %2.").arg(source, e.message());
+ return false;
+ } catch (...) {
+ qWarning() << QString::fromLatin1("Unknown exception caught while extracting %1.").arg(source);
+ return false;
+ }
+ } else {
+ qWarning() << QString::fromLatin1("Could not open %1 for reading: %2.").arg(
+ target, archive.errorString());
+ return false;
+ }
+ return true;
+}
+
+
+int main(int argc, char *argv[])
+{
+ QCoreApplication app(argc, argv);
+
+ // init installer to have the 7z lib initialized
+ QInstaller::init();
+ // with the installer messagehandler we need to enable verbose to see QDebugs
+ QInstaller::setVerbose(true);
+
+ QString repositoryUrl;
+ QString repositoryTarget;
+ QString packageDirectoryTarget;
+ bool clean = false;
+ QStringList args = app.arguments();
+ QStringList::const_iterator itArgument = args.constBegin();
+ itArgument++; // ignore the first one
+ if (itArgument == args.constEnd()) {
+ printUsage();
+ return 0;
+ }
+
+ for (; itArgument != args.constEnd(); ++itArgument) {
+ if (*itArgument == QString::fromLatin1("-h") || *itArgument == QString::fromLatin1("--help")) {
+ printUsage();
+ return 0;
+ } else if (*itArgument == QString::fromLatin1("--clean")) {
+ clean = true;
+ } else if (*itArgument == QString::fromLatin1("-u") || *itArgument == QString::fromLatin1("--url")) {
+ ++itArgument;
+ if (itArgument == args.end()) {
+ printUsage();
+ return -1;
+ } else {
+ repositoryUrl = *itArgument;
+ }
+ } else if (*itArgument == QString::fromLatin1("-r") || *itArgument == QString::fromLatin1("--repository")) {
+ ++itArgument;
+ if (itArgument == args.end()) {
+ printUsage();
+ return -1;
+ } else {
+ repositoryTarget = *itArgument;
+ }
+ } else if (*itArgument == QString::fromLatin1("-p") || *itArgument == QString::fromLatin1("--packages")) {
+ ++itArgument;
+ if (itArgument == args.end()) {
+ printUsage();
+ return -1;
+ } else {
+ packageDirectoryTarget = *itArgument;
+ }
+ } else {
+ qWarning() << QString::fromLatin1("Argument '%1' is unknown").arg(*itArgument);
+ printUsage();
+ return 0;
+ }
+ }
+ if (repositoryTarget.isEmpty() || packageDirectoryTarget.isEmpty()) {
+ printUsage();
+ return 0;
+ } else {
+ // resolve pathes
+ repositoryTarget = QFileInfo(repositoryTarget).absoluteFilePath();
+ packageDirectoryTarget = QFileInfo(packageDirectoryTarget).absoluteFilePath();
+ }
+
+ foreach (const QString &target, QStringList() << repositoryTarget << packageDirectoryTarget) {
+ if (QFileInfo(target).exists()) {
+ if (clean) {
+ qDebug() << "removing directory:" << target;
+ QInstaller::removeDirectory(target, true);
+ } else if (!QDir(target).entryInfoList(QDir::AllEntries | QDir::NoDotAndDotDot).isEmpty()){
+ qWarning() << QString::fromLatin1("The directory '%1' needs to be empty or just "
+ "add the --clean argument.").arg(target);
+ return EXIT_FAILURE;
+ }
+ }
+
+ while (!QDir().mkpath(target)) {
+ qWarning() << QString::fromLatin1("Could not create %1").arg(target);
+ }
+ }
+
+ QHash<QString, ComponentData> componentDataHash;
+ componentDataHash = downLoadRepository(repositoryUrl, repositoryTarget);
+
+ // maybe in that case we should download the meta data to temp and
+ // get the downdloadable archives information from there later
+ if (packageDirectoryTarget.isEmpty())
+ return EXIT_SUCCESS;
+
+ QDirIterator itRepositoryFile(repositoryTarget, QDir::Files, QDirIterator::Subdirectories);
+ while (itRepositoryFile.hasNext()) {
+ QString currentFile = itRepositoryFile.next();
+
+ QString componentSubdirectoryName = itRepositoryFile.filePath();
+
+ QString normalizedRepositoryTarget = repositoryTarget;
+ normalizedRepositoryTarget.replace(QLatin1Char('\\'), QLatin1Char('/'));
+
+ componentSubdirectoryName.remove(repositoryTarget);
+ componentSubdirectoryName.remove(normalizedRepositoryTarget);
+
+ QString componentPackageDir = QFileInfo(packageDirectoryTarget + QDir::separator() + componentSubdirectoryName).absolutePath();
+ QString absoluteSourceFilePath = itRepositoryFile.filePath();
+
+ if (currentFile.endsWith(QLatin1String("meta.7z"))) {
+ QString absolutTargetPath = QFileInfo(
+ packageDirectoryTarget + QDir::separator()).absolutePath();
+
+ extractFile(absoluteSourceFilePath, absolutTargetPath);
+ QInstaller::moveDirectoryContents(componentPackageDir,
+ componentPackageDir + QDir::separator() + QLatin1String("meta"));
+ } else if (!currentFile.endsWith(QLatin1String("Updates.xml")) && !currentFile.endsWith(QLatin1String(".sha1"))){
+ QString pathToTarget = componentPackageDir + QDir::separator() + QLatin1String("data");
+ QString target = QDir(pathToTarget).absoluteFilePath(itRepositoryFile.fileName());
+ QDir().mkpath(pathToTarget);
+ QFile file;
+ if (!file.copy(absoluteSourceFilePath, target)) {
+ qWarning() << QString::fromLatin1("copy file %1 to %2 wasn't working %3").arg(
+ absoluteSourceFilePath, target, file.errorString());
+ }
+ }
+ }
+
+
+ QHashIterator<QString, ComponentData> itComponentData(componentDataHash);
+ while (itComponentData.hasNext()) {
+ itComponentData.next();
+ if (itComponentData.value().m_script.isEmpty())
+ continue;
+
+ QString componentScript = QFileInfo(QString::fromLatin1("%1/%2/meta/%3").arg(
+ packageDirectoryTarget, itComponentData.key(), itComponentData.value().m_script)).absoluteFilePath();
+
+ QString packagesXml = QFileInfo(QString::fromLatin1("%1/%2/meta/packages.xml").arg(
+ packageDirectoryTarget, itComponentData.key())).absoluteFilePath();
+
+ QFile packagesXmlFile(packagesXml);
+
+ if (!packagesXmlFile.open(QIODevice::WriteOnly | QIODevice::Text)) {
+ qWarning() << QString::fromLatin1("Failed to open '%1' for writing. %2").arg(
+ packagesXml, packagesXmlFile.errorString());
+ return EXIT_FAILURE;
+ }
+
+ if (itComponentData.value().m_packageXml.toString().isEmpty()) {
+ qWarning() << "No xml data found in component:" << itComponentData.key();
+ return EXIT_FAILURE;
+ }
+
+ QTextStream stream(&packagesXmlFile);
+ stream << itComponentData.value().m_packageXml.toString(4);
+ packagesXmlFile.close();
+
+ QString dataPackagesPath = QFileInfo(QString::fromLatin1("%1/%2/data").arg(
+ packageDirectoryTarget, itComponentData.key())).absoluteFilePath();
+ QDir().mkpath(dataPackagesPath);
+
+ QString dataRepositoryPath = QFileInfo(repositoryTarget + QDir::separator() + itComponentData.key()).absoluteFilePath();
+ QDir().mkpath(dataRepositoryPath);
+
+ QFile componentScriptFile(componentScript);
+ if (!componentScriptFile.open(QIODevice::ReadOnly | QIODevice::Text)) {
+ qDebug() << QString::fromLatin1("Can not read %1 %2").arg(componentScript, componentScriptFile.errorString());
+ continue;
+ }
+
+ QString sevenZString;
+ QTextStream in(&componentScriptFile);
+ in.setCodec("UTF-8");
+ while (!in.atEnd()) {
+ QString line = in.readLine();
+ if (line.contains(QLatin1String(".7z"))) {
+ int firstPosition = line.indexOf(QLatin1String("\""));
+ QString subString = line.right(line.count() - firstPosition - 1); //-1 means "
+ //qDebug() << subString;
+ int secondPosition = subString.indexOf(QLatin1String("\""));
+ sevenZString = subString.left(secondPosition);
+ QUrl downloadUrl((QStringList() << repositoryUrl << itComponentData.key() << itComponentData.value().m_version + sevenZString).join(QLatin1String("/")));
+ QString localRepositoryTarget = dataRepositoryPath + QLatin1Char('/') + itComponentData.value().m_version + sevenZString;
+ downloadFile(downloadUrl, localRepositoryTarget);
+ downloadFile(downloadUrl.toString() + QLatin1String(".sha1"), localRepositoryTarget + QLatin1String(".sha1"));
+ QFile::copy(localRepositoryTarget, dataPackagesPath + QLatin1Char('/') + sevenZString);
+ }
+ }
+ }
+
+ return 0;
+}