289b9ac1eb
Review: http://reviewboard.kde.org/r/1984/ svn path=/trunk/KDE/kdelibs/; revision=1042086
579 lines
16 KiB
C++
579 lines
16 KiB
C++
/******************************************************************************
|
|
* 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/Wallpaper") {
|
|
structure = defaultPackageStructure(WallpaperComponent);
|
|
structure->d->type = "Plasma/Wallpaper";
|
|
} 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)
|
|
{
|
|
KUrl url(path);
|
|
QDir dir(url.toLocalFile());
|
|
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"
|
|
|