File indexing completed on 2024-09-08 06:42:26
0001 /* 0002 SPDX-FileCopyrightText: 2007-2009 Aaron Seigo <aseigo@kde.org> 0003 SPDX-FileCopyrightText: 2012 Sebastian Kügler <sebas@kde.org> 0004 0005 SPDX-License-Identifier: LGPL-2.0-or-later 0006 */ 0007 0008 #include "private/packagejobthread_p.h" 0009 #include "private/utils.h" 0010 0011 #include "config-package.h" 0012 #include "package.h" 0013 0014 #include <KArchive> 0015 #include <KLocalizedString> 0016 #include <KTar> 0017 #include <kzip.h> 0018 0019 #include "kpackage_debug.h" 0020 #include <QDir> 0021 #include <QFile> 0022 #include <QIODevice> 0023 #include <QJsonDocument> 0024 #include <QMimeDatabase> 0025 #include <QMimeType> 0026 #include <QProcess> 0027 #include <QRegularExpression> 0028 #include <QStandardPaths> 0029 #include <QUrl> 0030 #include <qtemporarydir.h> 0031 0032 namespace KPackage 0033 { 0034 bool copyFolder(QString sourcePath, QString targetPath) 0035 { 0036 QDir source(sourcePath); 0037 if (!source.exists()) { 0038 return false; 0039 } 0040 0041 QDir target(targetPath); 0042 if (!target.exists()) { 0043 QString targetName = target.dirName(); 0044 target.cdUp(); 0045 target.mkdir(targetName); 0046 target = QDir(targetPath); 0047 } 0048 0049 const auto lstEntries = source.entryList(QDir::Files); 0050 for (const QString &fileName : lstEntries) { 0051 QString sourceFilePath = sourcePath + QDir::separator() + fileName; 0052 QString targetFilePath = targetPath + QDir::separator() + fileName; 0053 0054 if (!QFile::copy(sourceFilePath, targetFilePath)) { 0055 return false; 0056 } 0057 } 0058 const auto lstEntries2 = source.entryList(QDir::AllDirs | QDir::NoDotAndDotDot); 0059 for (const QString &subFolderName : lstEntries2) { 0060 QString sourceSubFolderPath = sourcePath + QDir::separator() + subFolderName; 0061 QString targetSubFolderPath = targetPath + QDir::separator() + subFolderName; 0062 0063 if (!copyFolder(sourceSubFolderPath, targetSubFolderPath)) { 0064 return false; 0065 } 0066 } 0067 0068 return true; 0069 } 0070 0071 bool removeFolder(QString folderPath) 0072 { 0073 QDir folder(folderPath); 0074 return folder.removeRecursively(); 0075 } 0076 0077 class PackageJobThreadPrivate 0078 { 0079 public: 0080 QString installPath; 0081 QString errorMessage; 0082 std::function<void()> run; 0083 int errorCode; 0084 }; 0085 0086 PackageJobThread::PackageJobThread(PackageJob::OperationType type, const QString &src, const QString &dest, const KPackage::Package &package) 0087 : QObject() 0088 , QRunnable() 0089 { 0090 d = new PackageJobThreadPrivate; 0091 d->errorCode = KJob::NoError; 0092 if (type == PackageJob::Install) { 0093 d->run = [this, src, dest, package]() { 0094 install(src, dest, package); 0095 }; 0096 } else if (type == PackageJob::Update) { 0097 d->run = [this, src, dest, package]() { 0098 update(src, dest, package); 0099 }; 0100 } else if (type == PackageJob::Uninstall) { 0101 const QString packagePath = package.path(); 0102 d->run = [this, packagePath]() { 0103 uninstall(packagePath); 0104 }; 0105 0106 } else { 0107 Q_UNREACHABLE(); 0108 } 0109 } 0110 0111 PackageJobThread::~PackageJobThread() 0112 { 0113 delete d; 0114 } 0115 0116 void PackageJobThread::run() 0117 { 0118 Q_ASSERT(d->run); 0119 d->run(); 0120 } 0121 bool PackageJobThread::install(const QString &src, const QString &dest, const Package &package) 0122 { 0123 bool ok = installPackage(src, dest, package, PackageJob::Install); 0124 Q_EMIT installPathChanged(d->installPath); 0125 Q_EMIT jobThreadFinished(ok, errorCode(), d->errorMessage); 0126 return ok; 0127 } 0128 0129 static QString resolveHandler(const QString &scheme) 0130 { 0131 QString envOverride = qEnvironmentVariable("KPACKAGE_DEP_RESOLVERS_PATH"); 0132 QStringList searchDirs; 0133 if (!envOverride.isEmpty()) { 0134 searchDirs.push_back(envOverride); 0135 } 0136 searchDirs.append(QStringLiteral(KDE_INSTALL_FULL_LIBEXECDIR_KF "/kpackagehandlers")); 0137 // We have to use QStandardPaths::findExecutable here to handle the .exe suffix on Windows. 0138 return QStandardPaths::findExecutable(scheme + QLatin1String("handler"), searchDirs); 0139 } 0140 0141 bool PackageJobThread::installDependency(const QUrl &destUrl) 0142 { 0143 auto handler = resolveHandler(destUrl.scheme()); 0144 if (handler.isEmpty()) { 0145 return false; 0146 } 0147 0148 QProcess process; 0149 process.setProgram(handler); 0150 process.setArguments({destUrl.toString()}); 0151 process.setProcessChannelMode(QProcess::ForwardedChannels); 0152 process.start(); 0153 process.waitForFinished(); 0154 0155 return process.exitCode() == 0; 0156 } 0157 0158 bool PackageJobThread::installPackage(const QString &src, const QString &dest, const Package &package, PackageJob::OperationType operation) 0159 { 0160 QDir root(dest); 0161 if (!root.exists()) { 0162 QDir().mkpath(dest); 0163 if (!root.exists()) { 0164 d->errorMessage = i18n("Could not create package root directory: %1", dest); 0165 d->errorCode = PackageJob::JobError::RootCreationError; 0166 // qCWarning(KPACKAGE_LOG) << "Could not create package root directory: " << dest; 0167 return false; 0168 } 0169 } 0170 0171 QFileInfo fileInfo(src); 0172 if (!fileInfo.exists()) { 0173 d->errorMessage = i18n("No such file: %1", src); 0174 d->errorCode = PackageJob::JobError::PackageFileNotFoundError; 0175 return false; 0176 } 0177 0178 QString path; 0179 QTemporaryDir tempdir; 0180 bool archivedPackage = false; 0181 0182 if (fileInfo.isDir()) { 0183 // we have a directory, so let's just install what is in there 0184 path = src; 0185 // make sure we end in a slash! 0186 if (!path.endsWith(QLatin1Char('/'))) { 0187 path.append(QLatin1Char('/')); 0188 } 0189 } else { 0190 KArchive *archive = nullptr; 0191 QMimeDatabase db; 0192 QMimeType mimetype = db.mimeTypeForFile(src); 0193 if (mimetype.inherits(QStringLiteral("application/zip"))) { 0194 archive = new KZip(src); 0195 } else if (mimetype.inherits(QStringLiteral("application/x-compressed-tar")) || // 0196 mimetype.inherits(QStringLiteral("application/x-tar")) || // 0197 mimetype.inherits(QStringLiteral("application/x-bzip-compressed-tar")) || // 0198 mimetype.inherits(QStringLiteral("application/x-xz")) || // 0199 mimetype.inherits(QStringLiteral("application/x-lzma"))) { 0200 archive = new KTar(src); 0201 } else { 0202 // qCWarning(KPACKAGE_LOG) << "Could not open package file, unsupported archive format:" << src << mimetype.name(); 0203 d->errorMessage = i18n("Could not open package file, unsupported archive format: %1 %2", src, mimetype.name()); 0204 d->errorCode = PackageJob::JobError::UnsupportedArchiveFormatError; 0205 return false; 0206 } 0207 0208 if (!archive->open(QIODevice::ReadOnly)) { 0209 // qCWarning(KPACKAGE_LOG) << "Could not open package file:" << src; 0210 delete archive; 0211 d->errorMessage = i18n("Could not open package file: %1", src); 0212 d->errorCode = PackageJob::JobError::PackageOpenError; 0213 return false; 0214 } 0215 0216 archivedPackage = true; 0217 path = tempdir.path() + QLatin1Char('/'); 0218 0219 d->installPath = path; 0220 0221 const KArchiveDirectory *source = archive->directory(); 0222 source->copyTo(path); 0223 0224 QStringList entries = source->entries(); 0225 if (entries.count() == 1) { 0226 const KArchiveEntry *entry = source->entry(entries[0]); 0227 if (entry->isDirectory()) { 0228 path = path + entry->name() + QLatin1Char('/'); 0229 } 0230 } 0231 0232 delete archive; 0233 } 0234 0235 Package copyPackage = package; 0236 copyPackage.setPath(path); 0237 if (!copyPackage.isValid()) { 0238 d->errorMessage = i18n("Package is not considered valid"); 0239 d->errorCode = PackageJob::JobError::InvalidPackageStructure; 0240 return false; 0241 } 0242 0243 KPluginMetaData meta = copyPackage.metadata(); // The packagestructure might have set the metadata, so use that 0244 QString pluginName = meta.pluginId().isEmpty() ? QFileInfo(src).baseName() : meta.pluginId(); 0245 qCDebug(KPACKAGE_LOG) << "pluginname: " << meta.pluginId(); 0246 if (pluginName == QLatin1String("metadata")) { 0247 // qCWarning(KPACKAGE_LOG) << "Package plugin id not specified"; 0248 d->errorMessage = i18n("Package plugin id not specified: %1", src); 0249 d->errorCode = PackageJob::JobError::PluginIdInvalidError; 0250 return false; 0251 } 0252 0253 // Ensure that package names are safe so package uninstall can't inject 0254 // bad characters into the paths used for removal. 0255 const QRegularExpression validatePluginName(QStringLiteral("^[\\w\\-\\.]+$")); // Only allow letters, numbers, underscore and period. 0256 if (!validatePluginName.match(pluginName).hasMatch()) { 0257 // qCDebug(KPACKAGE_LOG) << "Package plugin id " << pluginName << "contains invalid characters"; 0258 d->errorMessage = i18n("Package plugin id %1 contains invalid characters", pluginName); 0259 d->errorCode = PackageJob::JobError::PluginIdInvalidError; 0260 return false; 0261 } 0262 0263 QString targetName = dest; 0264 if (targetName[targetName.size() - 1] != QLatin1Char('/')) { 0265 targetName.append(QLatin1Char('/')); 0266 } 0267 targetName.append(pluginName); 0268 0269 if (QFile::exists(targetName)) { 0270 if (operation == PackageJob::Update) { 0271 KPluginMetaData oldMeta; 0272 if (QString jsonPath = targetName + QLatin1String("/metadata.json"); QFileInfo::exists(jsonPath)) { 0273 oldMeta = KPluginMetaData::fromJsonFile(jsonPath); 0274 } 0275 0276 if (readKPackageType(oldMeta) != readKPackageType(meta)) { 0277 d->errorMessage = i18n("The new package has a different type from the old version already installed."); 0278 d->errorCode = PackageJob::JobError::UpdatePackageTypeMismatchError; 0279 } else if (isVersionNewer(oldMeta.version(), meta.version())) { 0280 const bool ok = uninstallPackage(targetName); 0281 if (!ok) { 0282 d->errorMessage = i18n("Impossible to remove the old installation of %1 located at %2. error: %3", pluginName, targetName, d->errorMessage); 0283 d->errorCode = PackageJob::JobError::OldVersionRemovalError; 0284 } 0285 } else { 0286 d->errorMessage = i18n("Not installing version %1 of %2. Version %3 already installed.", meta.version(), meta.pluginId(), oldMeta.version()); 0287 d->errorCode = PackageJob::JobError::NewerVersionAlreadyInstalledError; 0288 } 0289 } else { 0290 d->errorMessage = i18n("%1 already exists", targetName); 0291 d->errorCode = PackageJob::JobError::PackageAlreadyInstalledError; 0292 } 0293 0294 if (d->errorCode != KJob::NoError) { 0295 d->installPath = targetName; 0296 return false; 0297 } 0298 } 0299 0300 // install dependencies 0301 const QStringList optionalDependencies{QStringLiteral("sddmtheme.knsrc")}; 0302 const QStringList dependencies = meta.value(QStringLiteral("X-KPackage-Dependencies"), QStringList()); 0303 for (const QString &dep : dependencies) { 0304 QUrl depUrl(dep); 0305 const QString knsrcFilePath = QStandardPaths::locate(QStandardPaths::GenericDataLocation, QLatin1String("knsrcfiles/") + depUrl.host()); 0306 if (knsrcFilePath.isEmpty() && optionalDependencies.contains(depUrl.host())) { 0307 qWarning() << "Skipping depdendency due to knsrc files being missing" << depUrl; 0308 continue; 0309 } 0310 if (!installDependency(depUrl)) { 0311 d->errorMessage = i18n("Could not install dependency: '%1'", dep); 0312 d->errorCode = PackageJob::JobError::PackageCopyError; 0313 return false; 0314 } 0315 } 0316 0317 if (archivedPackage) { 0318 // it's in a temp dir, so just move it over. 0319 const bool ok = copyFolder(path, targetName); 0320 removeFolder(path); 0321 if (!ok) { 0322 // qCWarning(KPACKAGE_LOG) << "Could not move package to destination:" << targetName; 0323 d->errorMessage = i18n("Could not move package to destination: %1", targetName); 0324 d->errorCode = PackageJob::JobError::PackageMoveError; 0325 return false; 0326 } 0327 } else { 0328 // it's a directory containing the stuff, so copy the contents rather 0329 // than move them 0330 const bool ok = copyFolder(path, targetName); 0331 if (!ok) { 0332 // qCWarning(KPACKAGE_LOG) << "Could not copy package to destination:" << targetName; 0333 d->errorMessage = i18n("Could not copy package to destination: %1", targetName); 0334 d->errorCode = PackageJob::JobError::PackageCopyError; 0335 return false; 0336 } 0337 } 0338 0339 if (archivedPackage) { 0340 // no need to remove the temp dir (which has been successfully moved if it's an archive) 0341 tempdir.setAutoRemove(false); 0342 } 0343 0344 d->installPath = targetName; 0345 return true; 0346 } 0347 0348 bool PackageJobThread::update(const QString &src, const QString &dest, const Package &package) 0349 { 0350 bool ok = installPackage(src, dest, package, PackageJob::Update); 0351 Q_EMIT installPathChanged(d->installPath); 0352 Q_EMIT jobThreadFinished(ok, errorCode(), d->errorMessage); 0353 return ok; 0354 } 0355 0356 bool PackageJobThread::uninstall(const QString &packagePath) 0357 { 0358 bool ok = uninstallPackage(packagePath); 0359 // Do not emit the install path changed, information about the removed package might be useful for consumers 0360 // qCDebug(KPACKAGE_LOG) << "Thread: installFinished" << ok; 0361 Q_EMIT jobThreadFinished(ok, errorCode(), d->errorMessage); 0362 return ok; 0363 } 0364 0365 bool PackageJobThread::uninstallPackage(const QString &packagePath) 0366 { 0367 if (!QFile::exists(packagePath)) { 0368 d->errorMessage = packagePath.isEmpty() ? i18n("package path was deleted manually") : i18n("%1 does not exist", packagePath); 0369 d->errorCode = PackageJob::JobError::PackageFileNotFoundError; 0370 return false; 0371 } 0372 QString pkg; 0373 QString root; 0374 { 0375 // TODO KF6 remove, pass in packageroot, type and pluginName separately? 0376 QStringList ps = packagePath.split(QLatin1Char('/')); 0377 int ix = ps.count() - 1; 0378 if (packagePath.endsWith(QLatin1Char('/'))) { 0379 ix = ps.count() - 2; 0380 } 0381 pkg = ps[ix]; 0382 ps.pop_back(); 0383 root = ps.join(QLatin1Char('/')); 0384 } 0385 0386 bool ok = removeFolder(packagePath); 0387 if (!ok) { 0388 d->errorMessage = i18n("Could not delete package from: %1", packagePath); 0389 d->errorCode = PackageJob::JobError::PackageUninstallError; 0390 return false; 0391 } 0392 0393 return true; 0394 } 0395 0396 PackageJob::JobError PackageJobThread::errorCode() const 0397 { 0398 return static_cast<PackageJob::JobError>(d->errorCode); 0399 } 0400 0401 } // namespace KPackage 0402 0403 #include "moc_packagejobthread_p.cpp"