2007-06-17 00:25:16 +00:00
|
|
|
/******************************************************************************
|
2007-08-06 11:20:02 +00:00
|
|
|
* Copyright 2007 by Aaron Seigo <aseigo@kde.org> *
|
|
|
|
* Copyright 2007 by Riccardo Iaconelli <riccardo@kde.org> *
|
2007-06-17 00:25:16 +00:00
|
|
|
* *
|
|
|
|
* 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. *
|
|
|
|
*******************************************************************************/
|
|
|
|
|
2007-07-20 08:06:27 +00:00
|
|
|
#include "package.h"
|
|
|
|
|
2007-06-17 00:25:16 +00:00
|
|
|
#include <QDir>
|
|
|
|
#include <QFile>
|
|
|
|
|
|
|
|
#include <KArchiveDirectory>
|
|
|
|
#include <KArchiveEntry>
|
2007-07-20 04:07:00 +00:00
|
|
|
#include <KComponentData>
|
2008-02-29 08:14:32 +00:00
|
|
|
#include <KDesktopFile>
|
2007-06-17 00:25:16 +00:00
|
|
|
#include <KIO/FileCopyJob>
|
|
|
|
#include <KIO/Job>
|
2007-07-20 04:07:00 +00:00
|
|
|
#include <KPluginInfo>
|
2007-06-17 00:25:16 +00:00
|
|
|
#include <KStandardDirs>
|
2007-12-02 12:04:57 +00:00
|
|
|
#include <KTempDir>
|
2007-07-20 04:23:27 +00:00
|
|
|
#include <KTemporaryFile>
|
2007-06-17 00:25:16 +00:00
|
|
|
#include <KZip>
|
2007-10-05 22:21:25 +00:00
|
|
|
#include <KDebug>
|
2007-06-17 00:25:16 +00:00
|
|
|
|
|
|
|
#include "packagemetadata.h"
|
|
|
|
|
|
|
|
namespace Plasma
|
|
|
|
{
|
|
|
|
|
|
|
|
class Package::Private
|
|
|
|
{
|
|
|
|
public:
|
2008-02-26 04:07:07 +00:00
|
|
|
Private(const PackageStructure::Ptr st, const QString& p)
|
2007-06-17 00:25:16 +00:00
|
|
|
: structure(st),
|
|
|
|
basePath(p),
|
2007-12-02 12:04:57 +00:00
|
|
|
valid(QFile::exists(basePath)),
|
|
|
|
metadata(0)
|
2007-06-17 00:25:16 +00:00
|
|
|
{
|
|
|
|
if (valid && basePath[basePath.length() - 1] != '/') {
|
|
|
|
basePath.append('/');
|
|
|
|
}
|
2007-12-02 12:04:57 +00:00
|
|
|
}
|
2008-02-26 04:07:07 +00:00
|
|
|
|
2007-12-02 12:04:57 +00:00
|
|
|
~Private()
|
|
|
|
{
|
|
|
|
delete metadata;
|
2007-06-17 00:25:16 +00:00
|
|
|
}
|
|
|
|
|
2008-02-28 03:53:26 +00:00
|
|
|
PackageStructure::Ptr structure;
|
2007-06-17 00:25:16 +00:00
|
|
|
QString basePath;
|
|
|
|
bool valid;
|
2007-12-02 12:04:57 +00:00
|
|
|
PackageMetadata *metadata;
|
2007-06-17 00:25:16 +00:00
|
|
|
};
|
|
|
|
|
2008-02-28 03:53:26 +00:00
|
|
|
Package::Package(const QString& packageRoot, const QString& package, PackageStructure::Ptr structure)
|
2007-07-04 12:55:07 +00:00
|
|
|
: d(new Private(structure, packageRoot + '/' + package))
|
2007-06-17 00:25:16 +00:00
|
|
|
{
|
2008-02-28 03:53:26 +00:00
|
|
|
structure->setPath(d->basePath);
|
2007-06-17 00:25:16 +00:00
|
|
|
}
|
|
|
|
|
2008-02-28 03:53:26 +00:00
|
|
|
Package::Package(const QString &packagePath, PackageStructure::Ptr structure)
|
2007-11-26 19:16:40 +00:00
|
|
|
: d(new Private(structure, packagePath))
|
|
|
|
{
|
2008-02-28 03:53:26 +00:00
|
|
|
structure->setPath(d->basePath);
|
2007-11-26 19:16:40 +00:00
|
|
|
}
|
|
|
|
|
2007-06-17 00:25:16 +00:00
|
|
|
Package::~Package()
|
|
|
|
{
|
2007-10-27 14:27:25 +00:00
|
|
|
delete d;
|
2007-06-17 00:25:16 +00:00
|
|
|
}
|
|
|
|
|
2007-07-23 07:26:28 +00:00
|
|
|
bool Package::isValid() const
|
2007-07-20 03:21:40 +00:00
|
|
|
{
|
|
|
|
if (!d->valid) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-02-26 04:07:07 +00:00
|
|
|
foreach (const char *dir, d->structure->requiredDirectories()) {
|
2008-02-29 08:14:32 +00:00
|
|
|
if (!QFile::exists(d->basePath + d->structure->contentsPrefix() + d->structure->path(dir))) {
|
2007-12-02 12:04:57 +00:00
|
|
|
kWarning(505) << "Could not find required directory" << dir;
|
2007-07-20 03:21:40 +00:00
|
|
|
d->valid = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-02-26 04:07:07 +00:00
|
|
|
foreach (const char *file, d->structure->requiredFiles()) {
|
2008-02-29 08:14:32 +00:00
|
|
|
if (!QFile::exists(d->basePath + d->structure->contentsPrefix() + d->structure->path(file))) {
|
2008-01-18 00:10:27 +00:00
|
|
|
kWarning(505) << "Could not find required file" << file << ", look in"
|
2008-02-29 08:14:32 +00:00
|
|
|
<< d->basePath + d->structure->contentsPrefix() + d->structure->path(file) << endl;
|
2007-07-20 03:21:40 +00:00
|
|
|
d->valid = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-07-23 07:26:28 +00:00
|
|
|
QString Package::filePath(const char* fileType, const QString& filename) const
|
2007-06-17 00:25:16 +00:00
|
|
|
{
|
|
|
|
if (!d->valid) {
|
2008-02-29 08:14:32 +00:00
|
|
|
kDebug() << "package is not valid";
|
2007-06-17 00:25:16 +00:00
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2008-02-26 04:07:07 +00:00
|
|
|
QString path = d->structure->path(fileType);
|
2007-07-20 03:21:40 +00:00
|
|
|
|
|
|
|
if (path.isEmpty()) {
|
2008-02-29 08:14:32 +00:00
|
|
|
kDebug() << "no matching path came of it";
|
2007-07-20 03:21:40 +00:00
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2008-02-29 08:14:32 +00:00
|
|
|
path.prepend(d->basePath + d->structure->contentsPrefix());
|
2007-11-03 12:22:21 +00:00
|
|
|
|
2007-07-20 03:21:40 +00:00
|
|
|
if (!filename.isEmpty()) {
|
2007-06-17 00:25:16 +00:00
|
|
|
path.append("/").append(filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (QFile::exists(path)) {
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2008-02-29 08:14:32 +00:00
|
|
|
kDebug() << path << "does not exist";
|
2007-06-17 00:25:16 +00:00
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2007-07-23 07:26:28 +00:00
|
|
|
QString Package::filePath(const char* fileType) const
|
2007-06-17 00:25:16 +00:00
|
|
|
{
|
|
|
|
return filePath(fileType, QString());
|
|
|
|
}
|
|
|
|
|
2007-07-23 07:26:28 +00:00
|
|
|
QStringList Package::entryList(const char* fileType) const
|
2007-06-17 00:25:16 +00:00
|
|
|
{
|
|
|
|
if (!d->valid) {
|
|
|
|
return QStringList();
|
|
|
|
}
|
|
|
|
|
2008-02-26 04:07:07 +00:00
|
|
|
QString path = d->structure->path(fileType);
|
2007-06-17 00:25:16 +00:00
|
|
|
if (path.isEmpty()) {
|
|
|
|
return QStringList();
|
|
|
|
}
|
|
|
|
|
2008-02-29 08:14:32 +00:00
|
|
|
QDir dir(d->basePath + d->structure->contentsPrefix() + path);
|
2007-06-17 00:25:16 +00:00
|
|
|
|
|
|
|
if (!dir.exists()) {
|
|
|
|
return QStringList();
|
|
|
|
}
|
|
|
|
|
|
|
|
return dir.entryList(QDir::Files | QDir::Readable);
|
|
|
|
}
|
|
|
|
|
2008-02-26 04:07:07 +00:00
|
|
|
const PackageMetadata* Package::metadata() const
|
2007-12-02 12:04:57 +00:00
|
|
|
{
|
2008-02-28 03:53:26 +00:00
|
|
|
//FIXME: this only works for native plasma packges; should fall back to... PackageStructure?
|
2007-12-02 12:04:57 +00:00
|
|
|
if (!d->metadata) {
|
|
|
|
d->metadata = new PackageMetadata(d->basePath + "metadata.desktop");
|
|
|
|
}
|
|
|
|
return d->metadata;
|
|
|
|
}
|
|
|
|
|
2008-02-28 03:53:26 +00:00
|
|
|
const QString Package::path() const
|
|
|
|
{
|
|
|
|
return d->basePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
const PackageStructure::Ptr Package::structure() const
|
|
|
|
{
|
|
|
|
return d->structure;
|
|
|
|
}
|
|
|
|
|
2007-06-17 00:25:16 +00:00
|
|
|
//TODO: provide a version of this that allows one to ask for certain types of packages, etc?
|
2007-11-04 23:38:07 +00:00
|
|
|
// should we be using KService here instead/as well?
|
2007-06-17 00:25:16 +00:00
|
|
|
QStringList Package::knownPackages(const QString& packageRoot) // static
|
|
|
|
{
|
|
|
|
QDir dir(packageRoot);
|
|
|
|
|
|
|
|
if (!dir.exists()) {
|
|
|
|
return QStringList();
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList packages;
|
|
|
|
|
2007-12-03 03:43:30 +00:00
|
|
|
foreach (const QString& sdir, dir.entryList(QDir::AllDirs | QDir::Readable)) {
|
|
|
|
QString metadata = packageRoot + '/' + sdir + "/metadata.desktop";
|
2007-06-17 00:25:16 +00:00
|
|
|
if (QFile::exists(metadata)) {
|
|
|
|
PackageMetadata m(metadata);
|
2008-03-02 10:44:12 +00:00
|
|
|
packages << m.pluginName();
|
2007-06-17 00:25:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return packages;
|
|
|
|
}
|
|
|
|
|
2007-07-20 04:07:00 +00:00
|
|
|
bool Package::installPackage(const QString& package,
|
|
|
|
const QString& packageRoot) // static
|
2007-06-17 00:25:16 +00:00
|
|
|
{
|
|
|
|
//TODO: report *what* failed if something does fail
|
|
|
|
QDir root(packageRoot);
|
|
|
|
|
|
|
|
if (!root.exists()) {
|
|
|
|
KStandardDirs::makeDir(packageRoot);
|
|
|
|
if (!root.exists()) {
|
2007-12-02 12:04:57 +00:00
|
|
|
kWarning(505) << "Could not create package root directory:" << packageRoot;
|
2007-06-17 00:25:16 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-30 22:27:32 +00:00
|
|
|
QFileInfo fileInfo(package);
|
|
|
|
if (!fileInfo.exists()) {
|
2007-12-02 12:04:57 +00:00
|
|
|
kWarning(505) << "No such file:" << package;
|
2007-06-17 00:25:16 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-03-30 22:27:32 +00:00
|
|
|
QString path;
|
|
|
|
KTempDir tempdir;
|
|
|
|
bool archivedPackage = false;
|
2007-06-17 00:25:16 +00:00
|
|
|
|
2008-03-30 22:27:32 +00:00
|
|
|
if (fileInfo.isDir()) {
|
|
|
|
// we have a directory, so let's just install what is in there
|
|
|
|
path = package;
|
2007-06-17 00:25:16 +00:00
|
|
|
|
2008-03-30 22:27:32 +00:00
|
|
|
// make sure we end in a slash!
|
|
|
|
if (path[path.size() - 1] != '/') {
|
|
|
|
path.append('/');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
KZip archive(package);
|
|
|
|
if (!archive.open(QIODevice::ReadOnly)) {
|
|
|
|
kWarning(505) << "Could not open package file:" << package;
|
|
|
|
return false;
|
|
|
|
}
|
2007-06-17 00:25:16 +00:00
|
|
|
|
2008-03-30 22:27:32 +00:00
|
|
|
archivedPackage = true;
|
|
|
|
const KArchiveDirectory* source = archive.directory();
|
|
|
|
const KArchiveEntry* metadata = source->entry("metadata.desktop");
|
2007-06-17 00:25:16 +00:00
|
|
|
|
2008-03-30 22:27:32 +00:00
|
|
|
if (!metadata) {
|
|
|
|
kWarning(505) << "No metadata file in package" << package;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
path = tempdir.name();
|
|
|
|
source->copyTo(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString metadataPath = path + "metadata.desktop";
|
2007-06-17 00:25:16 +00:00
|
|
|
if (!QFile::exists(metadataPath)) {
|
2007-12-02 12:04:57 +00:00
|
|
|
kWarning(505) << "No metadata file in package" << package;
|
2007-06-17 00:25:16 +00:00
|
|
|
return false;
|
|
|
|
}
|
2008-03-30 22:27:32 +00:00
|
|
|
|
2007-06-17 00:25:16 +00:00
|
|
|
PackageMetadata meta(metadataPath);
|
2008-03-02 10:44:12 +00:00
|
|
|
QString targetName = meta.pluginName();
|
2007-06-17 00:25:16 +00:00
|
|
|
|
|
|
|
if (targetName.isEmpty()) {
|
2008-03-02 10:44:12 +00:00
|
|
|
kWarning(505) << "Package plugin name not specified";
|
2007-06-17 00:25:16 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-07-04 12:55:07 +00:00
|
|
|
targetName = packageRoot + '/' + targetName;
|
2007-06-17 00:25:16 +00:00
|
|
|
if (QFile::exists(targetName)) {
|
2007-12-02 12:04:57 +00:00
|
|
|
kWarning(505) << targetName << "already exists";
|
2007-06-17 00:25:16 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-03-30 22:27:32 +00:00
|
|
|
KIO::FileCopyJob* job(0);
|
|
|
|
|
|
|
|
if (archivedPackage) {
|
|
|
|
// it's in a temp dir, so just move it over.
|
|
|
|
job = KIO::file_move(path, targetName, -1, KIO::HideProgressInfo);
|
|
|
|
} else {
|
|
|
|
// it's a directory containing the stuff, so copy the contents rather
|
|
|
|
// than move them
|
|
|
|
job = KIO::file_copy(path, targetName, -1, KIO::HideProgressInfo);
|
|
|
|
}
|
2007-06-17 00:25:16 +00:00
|
|
|
|
2007-07-20 04:07:00 +00:00
|
|
|
if (!job->exec()) {
|
2007-12-02 12:04:57 +00:00
|
|
|
kWarning(505) << "Could not move package to destination:" << targetName;
|
2007-06-17 00:25:16 +00:00
|
|
|
return false;
|
|
|
|
}
|
2008-02-27 00:51:04 +00:00
|
|
|
|
2008-03-30 22:27:32 +00:00
|
|
|
if (archivedPackage) {
|
|
|
|
// no need to remove the temp dir (which has been successfully moved if it's an archive)
|
|
|
|
tempdir.setAutoRemove(false);
|
|
|
|
}
|
2007-06-17 00:25:16 +00:00
|
|
|
|
2007-07-20 04:07:00 +00:00
|
|
|
// and now we register it as a service =)
|
|
|
|
targetName.append("/metadata.desktop");
|
2008-02-29 08:14:32 +00:00
|
|
|
KConfigGroup cg = KDesktopFile(targetName).desktopGroup();
|
2008-02-27 00:51:04 +00:00
|
|
|
|
|
|
|
// should not installing it as a service disqualify it?
|
|
|
|
// i don't think so since KServiceTypeTrader may not be
|
|
|
|
// used by the installing app in any case, and the
|
|
|
|
// package is properly installed - aseigo
|
2008-02-29 08:14:32 +00:00
|
|
|
|
|
|
|
//TODO: reduce code duplication with registerPackage below
|
|
|
|
|
|
|
|
QFile icon(packageRoot + cg.readEntry("Icon"));
|
|
|
|
if (icon.exists()) {
|
|
|
|
QString installedIcon("plasma_applet_" + meta.pluginName() + cg.readEntry("Icon"));
|
|
|
|
meta.write(targetName, installedIcon);
|
|
|
|
installedIcon = KStandardDirs::locateLocal("icon", installedIcon);
|
|
|
|
job = KIO::file_copy(icon.fileName(), installedIcon, -1, KIO::HideProgressInfo);
|
|
|
|
job->exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString serviceName(KGlobal::mainComponent().componentName() + "_plasma_applet_" + meta.pluginName());
|
|
|
|
QString service = KStandardDirs::locateLocal("services", serviceName + ".desktop");
|
|
|
|
job = KIO::file_copy(targetName, service, -1, KIO::HideProgressInfo);
|
|
|
|
job->exec();
|
|
|
|
|
2008-02-27 00:51:04 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-02-29 08:14:32 +00:00
|
|
|
bool Package::registerPackage(const PackageMetadata &data, const QString &iconPath)
|
2008-02-27 00:51:04 +00:00
|
|
|
{
|
2008-02-29 08:14:32 +00:00
|
|
|
QString serviceName("plasma-applet-" + data.pluginName());
|
|
|
|
QString service = KStandardDirs::locateLocal("services", serviceName + ".desktop");
|
2007-07-20 04:07:00 +00:00
|
|
|
|
2008-02-29 08:14:32 +00:00
|
|
|
if (data.pluginName().isEmpty()) {
|
2008-02-27 00:51:04 +00:00
|
|
|
return false;
|
2007-07-20 04:07:00 +00:00
|
|
|
}
|
|
|
|
|
2008-02-29 08:14:32 +00:00
|
|
|
data.write(service);
|
|
|
|
|
|
|
|
KDesktopFile config(service);
|
|
|
|
KConfigGroup cg = config.desktopGroup();
|
|
|
|
cg.writeEntry("Type", "Service");
|
2008-03-05 21:12:33 +00:00
|
|
|
//TODO do we really like to just install all packages as applet/containment? Probably
|
|
|
|
//it would make sense to let the packages themself decide what they are.
|
|
|
|
cg.writeEntry("X-KDE-ServiceTypes", "Plasma/Applet,Plasma/Containment");
|
2008-02-29 08:14:32 +00:00
|
|
|
cg.writeEntry("X-KDE-PluginInfo-EnabledByDefault", true);
|
|
|
|
|
|
|
|
QFile icon(iconPath);
|
|
|
|
if (icon.exists()) {
|
|
|
|
//FIXME: the '/' search will break on non-UNIX. do we care?
|
|
|
|
QString installedIcon("plasma_applet_" + data.pluginName() +
|
|
|
|
iconPath.right(iconPath.length() - iconPath.lastIndexOf("/")));
|
|
|
|
cg.writeEntry("Icon", installedIcon);
|
|
|
|
installedIcon = KStandardDirs::locateLocal("icon", installedIcon);
|
|
|
|
KIO::FileCopyJob *job = KIO::file_copy(iconPath, installedIcon, -1, KIO::HideProgressInfo);
|
|
|
|
job->exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2007-06-17 00:25:16 +00:00
|
|
|
}
|
|
|
|
|
2007-12-02 12:04:57 +00:00
|
|
|
bool Package::createPackage(const PackageMetadata &metadata,
|
|
|
|
const QString &source,
|
|
|
|
const QString &destination,
|
|
|
|
const QString &icon) // static
|
2007-07-20 04:23:27 +00:00
|
|
|
{
|
|
|
|
if (!metadata.isComplete()) {
|
2007-12-02 12:04:57 +00:00
|
|
|
kWarning(550) << "Metadata file is not complete";
|
2007-07-20 04:23:27 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-12-02 12:04:57 +00:00
|
|
|
// write metadata in a temporary file
|
2007-07-20 04:23:27 +00:00
|
|
|
KTemporaryFile metadataFile;
|
2007-12-02 12:04:57 +00:00
|
|
|
if (!metadataFile.open()) {
|
|
|
|
return false;
|
2007-07-20 04:23:27 +00:00
|
|
|
}
|
2007-12-02 12:04:57 +00:00
|
|
|
metadata.write(metadataFile.fileName(), icon);
|
2007-07-20 04:23:27 +00:00
|
|
|
|
2007-12-02 12:04:57 +00:00
|
|
|
// put everything into a zip archive
|
2007-07-20 04:23:27 +00:00
|
|
|
KZip creation(destination);
|
|
|
|
creation.setCompression(KZip::NoCompression);
|
|
|
|
if (!creation.open(QIODevice::WriteOnly)) {
|
|
|
|
return false;
|
|
|
|
}
|
2008-02-29 08:14:32 +00:00
|
|
|
|
2007-07-20 04:23:27 +00:00
|
|
|
creation.addLocalFile(metadataFile.fileName(), "metadata.desktop");
|
2007-10-19 01:03:21 +00:00
|
|
|
creation.addLocalDirectory(source, "contents");
|
2007-07-20 04:23:27 +00:00
|
|
|
creation.close();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-06-17 00:25:16 +00:00
|
|
|
} // Namespace
|