/****************************************************************************** * Copyright 2007 by Aaron Seigo <aseigo@kde.org> * * * * This library is free software; you can redistribute it and/or * * modify it under the terms of the GNU Library General Public * * License as published by the Free Software Foundation; either * * version 2 of the License, or (at your option) any later version. * * * * This library is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * * Library General Public License for more details. * * * * You should have received a copy of the GNU Library General Public License * * along with this library; see the file COPYING.LIB. If not, write to * * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * * Boston, MA 02110-1301, USA. * *******************************************************************************/ #include "packagestructure.h" #include <QDir> #include <QMap> #include <QFileInfo> #include <kconfiggroup.h> #include <kstandarddirs.h> #include <kservicetypetrader.h> #include <kurl.h> #include <ktemporaryfile.h> #include <ktempdir.h> #include <kzip.h> #include <kio/netaccess.h> #include <kio/job.h> #include "package.h" #include "private/packages_p.h" #include "theme.h" namespace Plasma { class ContentStructure { public: ContentStructure() : directory(false), required(false) { } ContentStructure(const ContentStructure &other) { path = other.path; name = other.name; mimetypes = other.mimetypes; directory = other.directory; required = other.required; } QString path; QString name; QStringList mimetypes; bool directory : 1; bool required : 1; }; class PackageStructurePrivate { public: PackageStructurePrivate() : metadata(0), externalPaths(false) { } ~PackageStructurePrivate() { delete metadata; } void createPackageMetadata(const QString &path); static QHash<QString, PackageStructure::Ptr> structures; QString type; QString path; QString contentsPrefix; QString packageRoot; QString servicePrefix; QMap<QByteArray, ContentStructure> contents; QStringList mimetypes; PackageMetadata *metadata; bool externalPaths; }; QHash<QString, PackageStructure::Ptr> PackageStructurePrivate::structures; PackageStructure::PackageStructure(QObject *parent, const QString &type) : QObject(parent), d(new PackageStructurePrivate) { d->type = type; d->contentsPrefix = "contents/"; d->packageRoot = "plasma/plasmoids/"; d->servicePrefix = "plasma-applet-"; } PackageStructure::~PackageStructure() { delete d; } PackageStructure::Ptr PackageStructure::load(const QString &packageFormat) { if (packageFormat.isEmpty()) { return Ptr(new PackageStructure()); } PackageStructure::Ptr structure = PackageStructurePrivate::structures[packageFormat]; if (structure) { return structure; } if (packageFormat == "Plasma/Applet") { structure = defaultPackageStructure(AppletComponent); structure->d->type = "Plasma/Applet"; } else if (packageFormat == "Plasma/DataEngine") { structure = defaultPackageStructure(DataEngineComponent); structure->d->type = "Plasma/DataEngine"; } else if (packageFormat == "Plasma/Runner") { structure = defaultPackageStructure(RunnerComponent); structure->d->type = "Plasma/Runner"; } else if (packageFormat == "Plasma/Theme") { structure = Theme::packageStructure(); structure->d->type = "Plasma/Theme"; } if (structure) { PackageStructurePrivate::structures[packageFormat] = structure; return structure; } // first we check for plugins in sycoca QString constraint = QString("[X-KDE-PluginInfo-Name] == '%1'").arg(packageFormat); KService::List offers = KServiceTypeTrader::self()->query("Plasma/PackageStructure", constraint); QVariantList args; QString error; foreach (const KService::Ptr &offer, offers) { PackageStructure::Ptr structure( offer->createInstance<Plasma::PackageStructure>(0, args, &error)); if (structure) { return structure; } kDebug() << "Couldn't load PackageStructure for" << packageFormat << "! reason given: " << error; } // if that didn't give us any love, then we try to load from a config file structure = new PackageStructure(); QString configPath("plasma/packageformats/%1rc"); configPath = KStandardDirs::locate("data", configPath.arg(packageFormat)); if (!configPath.isEmpty()) { KConfig config(configPath); structure->read(&config); PackageStructurePrivate::structures[packageFormat] = structure; return structure; } // try to load from absolute file path KUrl url(packageFormat); if (url.isLocalFile()) { KConfig config(KIO::NetAccess::mostLocalUrl(url, NULL).toLocalFile(), KConfig::SimpleConfig); structure->read(&config); PackageStructurePrivate::structures[structure->type()] = structure; } else { KTemporaryFile tmp; if (tmp.open()) { KIO::Job *job = KIO::file_copy(url, KUrl(tmp.fileName()), -1, KIO::Overwrite | KIO::HideProgressInfo); if (job->exec()) { KConfig config(tmp.fileName(), KConfig::SimpleConfig); structure->read(&config); PackageStructurePrivate::structures[structure->type()] = structure; } } } return structure; } PackageStructure &PackageStructure::operator=(const PackageStructure &rhs) { if (this == &rhs) { return *this; } *d = *rhs.d; return *this; } QString PackageStructure::type() const { return d->type; } QList<const char*> PackageStructure::directories() const { QList<const char*> dirs; QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constBegin(); while (it != d->contents.constEnd()) { if (it.value().directory) { dirs << it.key(); } ++it; } return dirs; } QList<const char*> PackageStructure::requiredDirectories() const { QList<const char*> dirs; QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constBegin(); while (it != d->contents.constEnd()) { if (it.value().directory && it.value().required) { dirs << it.key(); } ++it; } return dirs; } QList<const char*> PackageStructure::files() const { QList<const char*> files; QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constBegin(); while (it != d->contents.constEnd()) { if (!it.value().directory) { files << it.key(); } ++it; } return files; } QList<const char*> PackageStructure::requiredFiles() const { QList<const char*> files; QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constBegin(); while (it != d->contents.constEnd()) { if (!it.value().directory && it.value().required) { files << it.key(); } ++it; } return files; } QStringList PackageStructure::entryList(const char *key) { QString p = path(key); if (p.isEmpty()) { return QStringList(); } QDir dir(d->path + d->contentsPrefix + p); if (dir.exists()) { if (d->externalPaths) { return dir.entryList(QDir::Files | QDir::Readable); } // ensure that we don't return files outside of our base path // due to symlink or ../ games QString canonicalized = dir.canonicalPath(); if (canonicalized.startsWith(d->path)) { return dir.entryList(QDir::Files | QDir::Readable); } } return QStringList(); } void PackageStructure::addDirectoryDefinition(const char *key, const QString &path, const QString &name) { ContentStructure s; s.name = name; s.path = path; s.directory = true; d->contents[key] = s; } void PackageStructure::addFileDefinition(const char *key, const QString &path, const QString &name) { ContentStructure s; s.name = name; s.path = path; s.directory = false; d->contents[key] = s; } QString PackageStructure::path(const char *key) const { //kDebug() << "looking for" << key; QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constFind(key); if (it == d->contents.constEnd()) { return QString(); } //kDebug() << "found" << key << "and the value is" << it.value().path; return it.value().path; } QString PackageStructure::name(const char *key) const { QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constFind(key); if (it == d->contents.constEnd()) { return QString(); } return it.value().name; } void PackageStructure::setRequired(const char *key, bool required) { QMap<QByteArray, ContentStructure>::iterator it = d->contents.find(key); if (it == d->contents.end()) { return; } it.value().required = required; } bool PackageStructure::isRequired(const char *key) const { QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constFind(key); if (it == d->contents.constEnd()) { return false; } return it.value().required; } void PackageStructure::setDefaultMimetypes(QStringList mimetypes) { d->mimetypes = mimetypes; } void PackageStructure::setMimetypes(const char *key, QStringList mimetypes) { QMap<QByteArray, ContentStructure>::iterator it = d->contents.find(key); if (it == d->contents.end()) { return; } it.value().mimetypes = mimetypes; } QStringList PackageStructure::mimetypes(const char *key) const { QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constFind(key); if (it == d->contents.constEnd()) { return QStringList(); } if (it.value().mimetypes.isEmpty()) { return d->mimetypes; } return it.value().mimetypes; } void PackageStructure::setPath(const QString &path) { QDir dir(path); QString basePath = dir.canonicalPath(); bool valid = QFile::exists(basePath); if (valid) { QFileInfo info(basePath); if (info.isDir()) { basePath.append(QDir::separator()); } //kDebug() << "basePath is" << basePath; } else { kDebug() << path << "invalid, basePath is" << basePath; return; } if (d->path == basePath) { return; } d->path = basePath; delete d->metadata; d->metadata = 0; pathChanged(); } QString PackageStructure::path() const { return d->path; } void PackageStructure::pathChanged() { // default impl does nothing, this is a hook for subclasses. } void PackageStructure::read(const KConfigBase *config) { d->contents.clear(); d->mimetypes.clear(); d->type = config->group("").readEntry("Type", QString()); QStringList groups = config->groupList(); foreach (const QString &group, groups) { QByteArray key = group.toAscii(); KConfigGroup entry = config->group(group); QString path = entry.readEntry("Path", QString()); QString name = entry.readEntry("Name", QString()); QStringList mimetypes = entry.readEntry("Mimetypes", QStringList()); bool directory = entry.readEntry("Directory", false); bool required = entry.readEntry("Required", false); if (directory) { addDirectoryDefinition(key, path, name); } else { addFileDefinition(key, path, name); } setMimetypes(key, mimetypes); setRequired(key, required); } } void PackageStructure::write(KConfigBase *config) const { config->group("").writeEntry("Type", type()); QMap<QByteArray, ContentStructure>::const_iterator it = d->contents.constBegin(); while (it != d->contents.constEnd()) { KConfigGroup group = config->group(it.key()); group.writeEntry("Path", it.value().path); group.writeEntry("Name", it.value().name); if (!it.value().mimetypes.isEmpty()) { group.writeEntry("Mimetypes", it.value().mimetypes); } if (it.value().directory) { group.writeEntry("Directory", true); } if (it.value().required) { group.writeEntry("Required", true); } ++it; } } QString PackageStructure::contentsPrefix() const { return d->contentsPrefix; } void PackageStructure::setContentsPrefix(const QString &prefix) { d->contentsPrefix = prefix; } bool PackageStructure::installPackage(const QString &package, const QString &packageRoot) { return Package::installPackage(package, packageRoot, d->servicePrefix); } bool PackageStructure::uninstallPackage(const QString &packageName, const QString &packageRoot) { return Package::uninstallPackage(packageName, packageRoot, d->servicePrefix); } void PackageStructure::createNewWidgetBrowser(QWidget *parent) { Q_UNUSED(parent) emit newWidgetBrowserFinished(); } QString PackageStructure::defaultPackageRoot() const { return d->packageRoot; } QString PackageStructure::servicePrefix() const { return d->servicePrefix; } void PackageStructure::setDefaultPackageRoot(const QString &packageRoot) { d->packageRoot = packageRoot; } void PackageStructure::setServicePrefix(const QString &servicePrefix) { d->servicePrefix = servicePrefix; } void PackageStructurePrivate::createPackageMetadata(const QString &path) { if (metadata) { delete metadata; metadata = 0; } QString metadataPath(path + "/metadata.desktop"); if (!QFile::exists(metadataPath)) { metadataPath.clear(); kWarning() << "No metadata file in the package, expected it at:" << metadataPath; } metadata = new PackageMetadata(metadataPath); } PackageMetadata PackageStructure::metadata() { if (!d->metadata && !d->path.isEmpty()) { QFileInfo fileInfo(d->path); if (fileInfo.isDir()) { d->createPackageMetadata(d->path); } else if (fileInfo.exists()) { KZip archive(d->path); if (archive.open(QIODevice::ReadOnly)) { const KArchiveDirectory *source = archive.directory(); KTempDir tempdir; source->copyTo(tempdir.name()); d->createPackageMetadata(tempdir.name()); } else { kWarning() << "Could not open package file:" << d->path; } } } if (!d->metadata) { d->metadata = new PackageMetadata(); } return *d->metadata; } bool PackageStructure::allowExternalPaths() const { return d->externalPaths; } void PackageStructure::setAllowExternalPaths(bool allow) { d->externalPaths = allow; } } // Plasma namespace #include "packagestructure.moc"