File indexing completed on 2024-09-08 09:41:24

0001 /*
0002     This file is part of the KDE project
0003     SPDX-FileCopyrightText: 2003, 2007 Matthias Kretz <kretz@kde.org>
0004     SPDX-FileCopyrightText: 2013 Sebastian Kügler <sebas@kde.org>
0005 
0006     SPDX-License-Identifier: LGPL-2.0-only
0007 */
0008 
0009 #include "kplugininfo.h"
0010 
0011 QT_WARNING_PUSH
0012 QT_WARNING_DISABLE_DEPRECATED
0013 
0014 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 90)
0015 #include "servicesdebug.h"
0016 #include <QDirIterator>
0017 #include <QJsonArray>
0018 #include <QMimeDatabase>
0019 #include <QStandardPaths>
0020 
0021 #include "ksycoca.h"
0022 #include "ksycoca_p.h"
0023 #include <KAboutData>
0024 #include <KDesktopFile>
0025 #include <KPluginMetaData>
0026 #include <kservicetypefactory_p.h>
0027 #include <kservicetypetrader.h>
0028 
0029 // clang-format off
0030 //#ifndef NDEBUG
0031 #define KPLUGININFO_ISVALID_ASSERTION \
0032     do { \
0033         if (!d) { \
0034             qFatal("Accessed invalid KPluginInfo object"); \
0035         } \
0036     } while (false)
0037 //#else
0038 //#define KPLUGININFO_ISVALID_ASSERTION
0039 //#endif
0040 
0041 #define GlobalQStringLiteral(name, str) inline QString name() { return QStringLiteral(str); }
0042 
0043 namespace {
0044 
0045 GlobalQStringLiteral(s_hiddenKey, "Hidden")
0046 GlobalQStringLiteral(s_nameKey, "Name")
0047 GlobalQStringLiteral(s_commentKey, "Comment")
0048 GlobalQStringLiteral(s_iconKey, "Icon")
0049 GlobalQStringLiteral(s_libraryKey, "X-KDE-Library")
0050 GlobalQStringLiteral(s_authorKey, "X-KDE-PluginInfo-Author")
0051 GlobalQStringLiteral(s_emailKey, "X-KDE-PluginInfo-Email")
0052 GlobalQStringLiteral(s_pluginNameKey, "X-KDE-PluginInfo-Name")
0053 GlobalQStringLiteral(s_versionKey, "X-KDE-PluginInfo-Version")
0054 GlobalQStringLiteral(s_websiteKey, "X-KDE-PluginInfo-Website")
0055 GlobalQStringLiteral(s_categoryKey, "X-KDE-PluginInfo-Category")
0056 GlobalQStringLiteral(s_licenseKey, "X-KDE-PluginInfo-License")
0057 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 79) && KCOREADDONS_BUILD_DEPRECATED_SINCE(5, 79)
0058 GlobalQStringLiteral(s_dependenciesKey, "X-KDE-PluginInfo-Depends")
0059 #endif
0060 GlobalQStringLiteral(s_serviceTypesKey, "ServiceTypes")
0061 GlobalQStringLiteral(s_xKDEServiceTypes, "X-KDE-ServiceTypes")
0062 GlobalQStringLiteral(s_mimeTypeKey, "MimeType")
0063 GlobalQStringLiteral(s_formFactorsKey, "X-KDE-FormFactors")
0064 GlobalQStringLiteral(s_enabledbyDefaultKey, "X-KDE-PluginInfo-EnabledByDefault")
0065 GlobalQStringLiteral(s_enabledKey, "Enabled")
0066 
0067 // these keys are used in the json metadata
0068 GlobalQStringLiteral(s_jsonDescriptionKey, "Description")
0069 GlobalQStringLiteral(s_jsonAuthorsKey, "Authors")
0070 GlobalQStringLiteral(s_jsonEmailKey, "Email")
0071 GlobalQStringLiteral(s_jsonCategoryKey, "Category")
0072 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 79) && KCOREADDONS_BUILD_DEPRECATED_SINCE(5, 79)
0073 GlobalQStringLiteral(s_jsonDependenciesKey, "Dependencies")
0074 #endif
0075 GlobalQStringLiteral(s_jsonEnabledByDefaultKey, "EnabledByDefault")
0076 GlobalQStringLiteral(s_jsonFormFactorsKey, "FormFactors")
0077 GlobalQStringLiteral(s_jsonLicenseKey, "License")
0078 GlobalQStringLiteral(s_jsonIdKey, "Id")
0079 GlobalQStringLiteral(s_jsonVersionKey, "Version")
0080 GlobalQStringLiteral(s_jsonWebsiteKey, "Website")
0081 GlobalQStringLiteral(s_jsonMimeTypesKey, "MimeTypes")
0082 GlobalQStringLiteral(s_jsonKPluginKey, "KPlugin")
0083 
0084 }
0085 // clang-format on
0086 
0087 class KPluginInfoPrivate : public QSharedData
0088 {
0089 public:
0090     KPluginInfoPrivate()
0091         : hidden(false)
0092         , pluginenabled(false)
0093         , kcmservicesCached(false)
0094     {
0095     }
0096 
0097     static QStringList deserializeList(const QString &data);
0098 
0099     bool hidden : 1;
0100     bool pluginenabled : 1;
0101     mutable bool kcmservicesCached : 1;
0102 
0103     KPluginMetaData metaData;
0104     KConfigGroup config;
0105     KService::Ptr service;
0106     mutable QList<KService::Ptr> kcmservices;
0107 
0108     /** assigns the @p md to @c metaData, but also ensures that compatibility values are handled */
0109     void setMetaData(const KPluginMetaData &md, bool warnOnOldStyle);
0110 };
0111 
0112 // This comes from KConfigGroupPrivate::deserializeList()
0113 QStringList KPluginInfoPrivate::deserializeList(const QString &data)
0114 {
0115     if (data.isEmpty()) {
0116         return QStringList();
0117     }
0118     if (data == QLatin1String("\\0")) {
0119         return QStringList(QString());
0120     }
0121     QStringList value;
0122     QString val;
0123     val.reserve(data.size());
0124     bool quoted = false;
0125     for (int p = 0; p < data.length(); p++) {
0126         if (quoted) {
0127             val += data[p];
0128             quoted = false;
0129         } else if (data[p].unicode() == '\\') {
0130             quoted = true;
0131         } else if (data[p].unicode() == ',' || data[p].unicode() == ';') {
0132             val.squeeze(); // release any unused memory
0133             value.append(val);
0134             val.clear();
0135             val.reserve(data.size() - p);
0136         } else {
0137             val += data[p];
0138         }
0139     }
0140     value.append(val);
0141     return value;
0142 }
0143 
0144 // maps the KService, QVariant and KDesktopFile keys to the new KPluginMetaData keys
0145 template<typename T, typename Func>
0146 static QJsonObject mapToJsonKPluginKey(const QString &name,
0147                                        const QString &description,
0148 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 79) && KCOREADDONS_BUILD_DEPRECATED_SINCE(5, 79)
0149                                        const QStringList &dependencies,
0150 #endif
0151                                        const QStringList &serviceTypes,
0152                                        const QStringList &formFactors,
0153                                        const T &data,
0154                                        Func accessor)
0155 {
0156     /* Metadata structure is as follows:
0157      "KPlugin": {
0158         "Name": "Date and Time",
0159         "Description": "Date and time by timezone",
0160         "Icon": "preferences-system-time",
0161         "Authors": { "Name": "Aaron Seigo", "Email": "aseigo@kde.org" },
0162         "Category": "Date and Time",
0163 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 79) && KCOREADDONS_BUILD_DEPRECATED_SINCE(5, 79)
0164         "Dependencies": [],
0165 #endif
0166         "EnabledByDefault": "true",
0167         "License": "LGPL",
0168         "Id": "time",
0169         "Version": "1.0",
0170         "Website": "http://plasma.kde.org/",
0171         "ServiceTypes": ["Plasma/DataEngine"]
0172         "FormFactors": ["tablet", "handset"]
0173     }
0174      */
0175     QJsonObject kplugin;
0176     kplugin[s_nameKey()] = name;
0177     kplugin[s_jsonDescriptionKey()] = description;
0178     kplugin[s_iconKey()] = accessor(data, s_iconKey());
0179     QJsonObject authors;
0180     authors[s_nameKey()] = accessor(data, s_authorKey());
0181     authors[s_jsonEmailKey()] = accessor(data, s_emailKey());
0182     kplugin[s_jsonAuthorsKey()] = authors;
0183     kplugin[s_jsonCategoryKey()] = accessor(data, s_categoryKey());
0184     QJsonValue enabledByDefault = accessor(data, s_enabledbyDefaultKey());
0185     // make sure that enabledByDefault is bool and not string
0186     if (!enabledByDefault.isBool()) {
0187         enabledByDefault = enabledByDefault.toString().compare(QLatin1String("true"), Qt::CaseInsensitive) == 0;
0188     }
0189     kplugin[s_jsonEnabledByDefaultKey()] = enabledByDefault;
0190     kplugin[s_jsonLicenseKey()] = accessor(data, s_licenseKey());
0191     kplugin[s_jsonIdKey()] = accessor(data, s_pluginNameKey());
0192     kplugin[s_jsonVersionKey()] = accessor(data, s_versionKey());
0193     kplugin[s_jsonWebsiteKey()] = accessor(data, s_websiteKey());
0194     kplugin[s_jsonFormFactorsKey()] = QJsonArray::fromStringList(formFactors);
0195     kplugin[s_serviceTypesKey()] = QJsonArray::fromStringList(serviceTypes);
0196 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 79) && KCOREADDONS_BUILD_DEPRECATED_SINCE(5, 79)
0197     kplugin[s_jsonDependenciesKey()] = QJsonArray::fromStringList(dependencies);
0198 #endif
0199     QJsonValue mimeTypes = accessor(data, s_mimeTypeKey());
0200     if (mimeTypes.isString()) {
0201         QStringList mimeList = KPluginInfoPrivate::deserializeList(mimeTypes.toString());
0202         if (!mimeList.isEmpty()) {
0203             mimeTypes = QJsonArray::fromStringList(mimeList);
0204         } else {
0205             mimeTypes = QJsonValue();
0206         }
0207     }
0208     kplugin[s_jsonMimeTypesKey()] = mimeTypes;
0209     return kplugin;
0210 }
0211 
0212 // TODO: KF6 remove
0213 static KPluginMetaData fromCompatibilityJson(const QJsonObject &json, const QString &lib, const QString &metaDataFile, bool warnOnOldStyle)
0214 {
0215     // This is not added to KPluginMetaData(QJsonObject, QString) to ensure that all the compatility code
0216     // remains in kservice and does not increase the size of kcoreaddons
0217     QStringList serviceTypes = KPluginMetaData::readStringList(json, s_xKDEServiceTypes());
0218     if (serviceTypes.isEmpty()) {
0219         serviceTypes = KPluginMetaData::readStringList(json, s_serviceTypesKey());
0220     }
0221     QJsonObject obj = json;
0222     QString name = KPluginMetaData::readTranslatedString(json, s_nameKey());
0223     if (warnOnOldStyle) {
0224         qWarning(
0225             "Constructing a KPluginInfo object from old style JSON. Please use"
0226             " kcoreaddons_desktop_to_json() for \"%s\" instead of kservice_desktop_to_json()"
0227             " in your CMake code.",
0228             qPrintable(lib));
0229     }
0230     QString description = KPluginMetaData::readTranslatedString(json, s_commentKey());
0231     QStringList formfactors = KPluginMetaData::readStringList(json, s_jsonFormFactorsKey());
0232     QJsonObject kplugin = mapToJsonKPluginKey(name,
0233                                               description,
0234 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 79) && KCOREADDONS_BUILD_DEPRECATED_SINCE(5, 79)
0235                                               KPluginMetaData::readStringList(json, s_dependenciesKey()),
0236 #endif
0237                                               serviceTypes,
0238                                               formfactors,
0239                                               json,
0240                                               [](const QJsonObject &o, const QString &key) {
0241                                                   return o.value(key);
0242                                               });
0243     obj.insert(s_jsonKPluginKey(), kplugin);
0244     return KPluginMetaData(obj, lib, metaDataFile);
0245 }
0246 
0247 void KPluginInfoPrivate::setMetaData(const KPluginMetaData &md, bool warnOnOldStyle)
0248 {
0249     const QJsonObject json = md.rawData();
0250     if (!json.contains(s_jsonKPluginKey())) {
0251         // "KPlugin" key does not exists -> convert from compatibility mode
0252         metaData = fromCompatibilityJson(json, md.fileName(), md.metaDataFileName(), warnOnOldStyle);
0253     } else {
0254         metaData = md;
0255     }
0256 }
0257 
0258 KPluginInfo::KPluginInfo(const KPluginMetaData &md)
0259     : d(new KPluginInfoPrivate)
0260 {
0261     d->setMetaData(md, true);
0262     if (!d->metaData.isValid()) {
0263         d.reset();
0264     }
0265 }
0266 
0267 KPluginInfo::KPluginInfo(const QString &filename /*, QStandardPaths::StandardLocation resource*/)
0268     : d(new KPluginInfoPrivate)
0269 {
0270     KDesktopFile file(/*resource,*/ filename);
0271 
0272     KConfigGroup cg = file.desktopGroup();
0273     if (!cg.exists()) {
0274         qCWarning(SERVICES) << filename << "has no desktop group, cannot construct a KPluginInfo object from it.";
0275         d.reset();
0276         return;
0277     }
0278     d->hidden = cg.readEntry(s_hiddenKey(), false);
0279     if (d->hidden) {
0280         return;
0281     }
0282 
0283     if (file.fileName().endsWith(QLatin1String(".desktop"))) {
0284         d->setMetaData(KPluginMetaData::fromDesktopFile(file.fileName()), true);
0285     } else {
0286         d->setMetaData(KPluginMetaData(file.fileName()), true);
0287     }
0288     if (!d->metaData.isValid()) {
0289         qCWarning(SERVICES) << "Failed to read metadata from .desktop file" << file.fileName();
0290         d.reset();
0291     }
0292 }
0293 
0294 KPluginInfo::KPluginInfo(const QVariantList &args, const QString &libraryPath)
0295     : d(new KPluginInfoPrivate)
0296 {
0297     const QString metaData = QStringLiteral("MetaData");
0298     for (const QVariant &v : args) {
0299         if (v.canConvert<QVariantMap>()) {
0300             const QVariantMap &m = v.toMap();
0301             const QVariant &_metadata = m.value(metaData);
0302             if (_metadata.canConvert<QVariantMap>()) {
0303                 const QVariantMap &map = _metadata.toMap();
0304                 if (map.value(s_hiddenKey()).toBool()) {
0305                     d->hidden = true;
0306                     break;
0307                 }
0308                 d->setMetaData(KPluginMetaData(QJsonObject::fromVariantMap(map), libraryPath), true);
0309                 break;
0310             }
0311         }
0312     }
0313     if (!d->metaData.isValid()) {
0314         d.reset();
0315     }
0316 }
0317 
0318 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 88)
0319 KPluginInfo::KPluginInfo(const KService::Ptr service)
0320     : d(new KPluginInfoPrivate)
0321 {
0322     if (!service) {
0323         d = nullptr; // isValid() == false
0324         return;
0325     }
0326     d->service = service;
0327     if (service->isDeleted()) {
0328         d->hidden = true;
0329         return;
0330     }
0331 
0332     KSycoca::self()->ensureCacheValid();
0333 
0334     QJsonObject json;
0335     const auto propertyList = service->propertyNames();
0336     for (const QString &key : propertyList) {
0337         QMetaType::Type t = KSycocaPrivate::self()->serviceTypeFactory()->findPropertyTypeByName(key);
0338         if (t == QMetaType::UnknownType) {
0339             t = QMetaType::QString; // default to string if the type is not known
0340         }
0341         QVariant v = service->property(key, t);
0342         if (v.isValid()) {
0343             json[key] = QJsonValue::fromVariant(v);
0344         }
0345     }
0346     // reintroduce the separation between MimeType= and X-KDE-ServiceTypes=
0347     // we could do this by modifying KService and KSyCoCa, but as this is only compatibility
0348     // code we just query QMimeDatabase whether a ServiceType is a valid MIME type.
0349     const QStringList services = service->serviceTypes();
0350     if (!services.isEmpty()) {
0351         QMimeDatabase db;
0352         QStringList mimeTypes;
0353         mimeTypes.reserve(services.size());
0354         QStringList newServiceTypes;
0355         newServiceTypes.reserve(services.size());
0356         for (const QString &s : services) {
0357             // If we have a valid mime type of a wildcard, we know it is a mime type and not a service type
0358             // this is for example required in the thumbnailers.
0359             // Also the conversion code in kcoreaddons does not check the validity.
0360             if (db.mimeTypeForName(s).isValid() || s.contains(QLatin1Char('*'))) {
0361                 mimeTypes << s;
0362             } else {
0363                 newServiceTypes << s;
0364             }
0365         }
0366         json[s_mimeTypeKey()] = QJsonArray::fromStringList(mimeTypes);
0367         json[s_xKDEServiceTypes()] = QJsonArray::fromStringList(newServiceTypes);
0368         json[s_serviceTypesKey()] = QJsonValue();
0369     }
0370 
0371     d->setMetaData(KPluginMetaData(json, service->library(), service->entryPath()), false);
0372     if (!d->metaData.isValid()) {
0373         d.reset();
0374     }
0375 }
0376 #endif
0377 
0378 KPluginInfo::KPluginInfo()
0379     : d(nullptr) // isValid() == false
0380 {
0381 }
0382 
0383 bool KPluginInfo::isValid() const
0384 {
0385     return d.data() != nullptr;
0386 }
0387 
0388 KPluginInfo::KPluginInfo(const KPluginInfo &rhs)
0389     : d(rhs.d)
0390 {
0391 }
0392 
0393 KPluginInfo &KPluginInfo::operator=(const KPluginInfo &rhs)
0394 {
0395     d = rhs.d;
0396     return *this;
0397 }
0398 
0399 bool KPluginInfo::operator==(const KPluginInfo &rhs) const
0400 {
0401     return d == rhs.d;
0402 }
0403 
0404 bool KPluginInfo::operator!=(const KPluginInfo &rhs) const
0405 {
0406     return d != rhs.d;
0407 }
0408 
0409 bool KPluginInfo::operator<(const KPluginInfo &rhs) const
0410 {
0411     if (category() < rhs.category()) {
0412         return true;
0413     }
0414     if (category() == rhs.category()) {
0415         return name() < rhs.name();
0416     }
0417     return false;
0418 }
0419 
0420 bool KPluginInfo::operator>(const KPluginInfo &rhs) const
0421 {
0422     if (category() > rhs.category()) {
0423         return true;
0424     }
0425     if (category() == rhs.category()) {
0426         return name() > rhs.name();
0427     }
0428     return false;
0429 }
0430 
0431 KPluginInfo::~KPluginInfo()
0432 {
0433 }
0434 
0435 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 88)
0436 QList<KPluginInfo> KPluginInfo::fromServices(const KService::List &services, const KConfigGroup &config)
0437 {
0438     QList<KPluginInfo> infolist;
0439     for (KService::List::ConstIterator it = services.begin(); it != services.end(); ++it) {
0440         KPluginInfo info(*it);
0441         if (info.isValid()) {
0442             info.setConfig(config);
0443             infolist += info;
0444         }
0445     }
0446     return infolist;
0447 }
0448 #endif
0449 
0450 QList<KPluginInfo> KPluginInfo::fromFiles(const QStringList &files, const KConfigGroup &config)
0451 {
0452     QList<KPluginInfo> infolist;
0453     infolist.reserve(files.size());
0454 
0455     std::transform(files.cbegin(), files.cend(), std::back_inserter(infolist), [&config](const QString &file) {
0456         KPluginInfo info(file);
0457         info.setConfig(config);
0458         return info;
0459     });
0460 
0461     return infolist;
0462 }
0463 
0464 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 81)
0465 QList<KPluginInfo> KPluginInfo::fromKPartsInstanceName(const QString &name, const KConfigGroup &config)
0466 {
0467     QStringList files;
0468     const QStringList dirs =
0469         QStandardPaths::locateAll(QStandardPaths::GenericDataLocation, name + QLatin1String("/kpartplugins"), QStandardPaths::LocateDirectory);
0470     for (const QString &dir : dirs) {
0471         QDirIterator it(dir, QStringList() << QStringLiteral("*.desktop"));
0472         while (it.hasNext()) {
0473             files.append(it.next());
0474         }
0475     }
0476     return fromFiles(files, config);
0477 }
0478 #endif
0479 
0480 bool KPluginInfo::isHidden() const
0481 {
0482     KPLUGININFO_ISVALID_ASSERTION;
0483     return d->hidden;
0484 }
0485 
0486 void KPluginInfo::setPluginEnabled(bool enabled)
0487 {
0488     KPLUGININFO_ISVALID_ASSERTION;
0489     // qDebug() << Q_FUNC_INFO;
0490     d->pluginenabled = enabled;
0491 }
0492 
0493 bool KPluginInfo::isPluginEnabled() const
0494 {
0495     KPLUGININFO_ISVALID_ASSERTION;
0496     // qDebug() << Q_FUNC_INFO;
0497     return d->pluginenabled;
0498 }
0499 
0500 bool KPluginInfo::isPluginEnabledByDefault() const
0501 {
0502     KPLUGININFO_ISVALID_ASSERTION;
0503     // qDebug() << Q_FUNC_INFO;
0504     return d->metaData.isEnabledByDefault();
0505 }
0506 
0507 QString KPluginInfo::name() const
0508 {
0509     KPLUGININFO_ISVALID_ASSERTION;
0510     return d->metaData.name();
0511 }
0512 
0513 QString KPluginInfo::comment() const
0514 {
0515     KPLUGININFO_ISVALID_ASSERTION;
0516     return d->metaData.description();
0517 }
0518 
0519 QString KPluginInfo::icon() const
0520 {
0521     KPLUGININFO_ISVALID_ASSERTION;
0522     return d->metaData.iconName();
0523 }
0524 
0525 QString KPluginInfo::entryPath() const
0526 {
0527     KPLUGININFO_ISVALID_ASSERTION;
0528     return d->metaData.metaDataFileName();
0529 }
0530 
0531 QString KPluginInfo::author() const
0532 {
0533     KPLUGININFO_ISVALID_ASSERTION;
0534     const QList<KAboutPerson> &authors = d->metaData.authors();
0535     return authors.isEmpty() ? QString() : authors[0].name();
0536 }
0537 
0538 QString KPluginInfo::email() const
0539 {
0540     KPLUGININFO_ISVALID_ASSERTION;
0541     const QList<KAboutPerson> &authors = d->metaData.authors();
0542     return authors.isEmpty() ? QString() : authors[0].emailAddress();
0543 }
0544 
0545 QString KPluginInfo::category() const
0546 {
0547     KPLUGININFO_ISVALID_ASSERTION;
0548     return d->metaData.category();
0549 }
0550 
0551 QStringList KPluginInfo::formFactors() const
0552 {
0553     KPLUGININFO_ISVALID_ASSERTION;
0554     return d->metaData.formFactors();
0555 }
0556 
0557 QString KPluginInfo::pluginName() const
0558 {
0559     KPLUGININFO_ISVALID_ASSERTION;
0560     return d->metaData.pluginId();
0561 }
0562 
0563 QString KPluginInfo::libraryPath() const
0564 {
0565     KPLUGININFO_ISVALID_ASSERTION;
0566     return d->metaData.fileName();
0567 }
0568 
0569 QString KPluginInfo::version() const
0570 {
0571     KPLUGININFO_ISVALID_ASSERTION;
0572     return d->metaData.version();
0573 }
0574 
0575 QString KPluginInfo::website() const
0576 {
0577     KPLUGININFO_ISVALID_ASSERTION;
0578     return d->metaData.website();
0579 }
0580 
0581 QString KPluginInfo::license() const
0582 {
0583     KPLUGININFO_ISVALID_ASSERTION;
0584     return d->metaData.license();
0585 }
0586 
0587 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 79) && KCOREADDONS_BUILD_DEPRECATED_SINCE(5, 79)
0588 QStringList KPluginInfo::dependencies() const
0589 {
0590     KPLUGININFO_ISVALID_ASSERTION;
0591     return d->metaData.dependencies();
0592 }
0593 #endif
0594 
0595 QStringList KPluginInfo::serviceTypes() const
0596 {
0597     KPLUGININFO_ISVALID_ASSERTION;
0598     // KService/KPluginInfo include the MIME types in serviceTypes()
0599     return d->metaData.serviceTypes() + d->metaData.mimeTypes();
0600 }
0601 
0602 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 70)
0603 KService::Ptr KPluginInfo::service() const
0604 {
0605     KPLUGININFO_ISVALID_ASSERTION;
0606     return d->service;
0607 }
0608 #endif
0609 
0610 QList<KService::Ptr> KPluginInfo::kcmServices() const
0611 {
0612     KPLUGININFO_ISVALID_ASSERTION;
0613     if (!d->kcmservicesCached) {
0614         d->kcmservices =
0615             KServiceTypeTrader::self()->query(QStringLiteral("KCModule"), QLatin1Char('\'') + pluginName() + QLatin1String("' in [X-KDE-ParentComponents]"));
0616         // qDebug() << "found" << d->kcmservices.count() << "offers for" << d->pluginName;
0617 
0618         d->kcmservicesCached = true;
0619     }
0620 
0621     return d->kcmservices;
0622 }
0623 
0624 void KPluginInfo::setConfig(const KConfigGroup &config)
0625 {
0626     KPLUGININFO_ISVALID_ASSERTION;
0627     d->config = config;
0628 }
0629 
0630 KConfigGroup KPluginInfo::config() const
0631 {
0632     KPLUGININFO_ISVALID_ASSERTION;
0633     return d->config;
0634 }
0635 
0636 QVariant KPluginInfo::property(const QString &key) const
0637 {
0638     KPLUGININFO_ISVALID_ASSERTION;
0639     if (d->service) {
0640         return d->service->property(key);
0641     }
0642     QVariant result = d->metaData.rawData().value(key).toVariant();
0643     if (result.isValid()) {
0644         KSycoca::self()->ensureCacheValid();
0645         const QMetaType::Type t = KSycocaPrivate::self()->serviceTypeFactory()->findPropertyTypeByName(key);
0646 
0647         // special case if we want a stringlist: split values by ',' or ';' and construct the list
0648         if (t == QMetaType::QStringList) {
0649             if (result.canConvert<QString>()) {
0650                 result = KPluginInfoPrivate::deserializeList(result.toString());
0651             } else if (result.canConvert<QVariantList>()) {
0652                 const QVariantList list = result.toList();
0653                 QStringList newResult;
0654                 for (const QVariant &value : list) {
0655                     newResult += value.toString();
0656                 }
0657                 result = newResult;
0658             } else {
0659                 qCWarning(SERVICES) << "Cannot interpret" << result << "into a string list";
0660             }
0661         }
0662         return result;
0663     }
0664 
0665     // If the key was not found check compatibility for old key names and print a warning
0666     // These warnings should only happen if JSON was generated with kcoreaddons_desktop_to_json
0667     // but the application uses KPluginTrader::query() instead of KPluginLoader::findPlugins()
0668     // TODO: KF6 remove
0669 #define RETURN_WITH_DEPRECATED_WARNING(ret)                                                                                                                    \
0670     qWarning("Calling KPluginInfo::property(\"%s\") is deprecated, use KPluginInfo::" #ret " in \"%s\" instead.",                                              \
0671              qPrintable(key),                                                                                                                                  \
0672              qPrintable(d->metaData.fileName()));                                                                                                              \
0673     return ret;
0674     if (key == s_authorKey()) {
0675         RETURN_WITH_DEPRECATED_WARNING(author());
0676     } else if (key == s_categoryKey()) {
0677         RETURN_WITH_DEPRECATED_WARNING(category());
0678     } else if (key == s_commentKey()) {
0679         RETURN_WITH_DEPRECATED_WARNING(comment());
0680 #if KSERVICE_BUILD_DEPRECATED_SINCE(5, 79) && KCOREADDONS_BUILD_DEPRECATED_SINCE(5, 79)
0681     } else if (key == s_dependenciesKey()) {
0682         RETURN_WITH_DEPRECATED_WARNING(dependencies());
0683 #endif
0684     } else if (key == s_emailKey()) {
0685         RETURN_WITH_DEPRECATED_WARNING(email());
0686     } else if (key == s_enabledbyDefaultKey()) {
0687         RETURN_WITH_DEPRECATED_WARNING(isPluginEnabledByDefault());
0688     } else if (key == s_libraryKey()) {
0689         RETURN_WITH_DEPRECATED_WARNING(libraryPath());
0690     } else if (key == s_licenseKey()) {
0691         RETURN_WITH_DEPRECATED_WARNING(license());
0692     } else if (key == s_nameKey()) {
0693         RETURN_WITH_DEPRECATED_WARNING(name());
0694     } else if (key == s_pluginNameKey()) {
0695         RETURN_WITH_DEPRECATED_WARNING(pluginName());
0696     } else if (key == s_serviceTypesKey()) {
0697         RETURN_WITH_DEPRECATED_WARNING(serviceTypes());
0698     } else if (key == s_versionKey()) {
0699         RETURN_WITH_DEPRECATED_WARNING(version());
0700     } else if (key == s_websiteKey()) {
0701         RETURN_WITH_DEPRECATED_WARNING(website());
0702     } else if (key == s_xKDEServiceTypes()) {
0703         RETURN_WITH_DEPRECATED_WARNING(serviceTypes());
0704     } else if (key == s_formFactorsKey()) {
0705         RETURN_WITH_DEPRECATED_WARNING(formFactors());
0706     }
0707 #undef RETURN_WITH_DEPRECATED_WARNING
0708     // not a compatibility key -> return invalid QVariant
0709     return result;
0710 }
0711 
0712 QVariantMap KPluginInfo::properties() const
0713 {
0714     return d->metaData.rawData().toVariantMap();
0715 }
0716 
0717 void KPluginInfo::save(KConfigGroup config)
0718 {
0719     KPLUGININFO_ISVALID_ASSERTION;
0720     // qDebug() << Q_FUNC_INFO;
0721     if (config.isValid()) {
0722         config.writeEntry(pluginName() + s_enabledKey(), isPluginEnabled());
0723     } else {
0724         if (!d->config.isValid()) {
0725             qCWarning(SERVICES) << "no KConfigGroup, cannot save";
0726             return;
0727         }
0728         d->config.writeEntry(pluginName() + s_enabledKey(), isPluginEnabled());
0729     }
0730 }
0731 
0732 void KPluginInfo::load(const KConfigGroup &config)
0733 {
0734     KPLUGININFO_ISVALID_ASSERTION;
0735     // qDebug() << Q_FUNC_INFO;
0736     if (config.isValid()) {
0737         setPluginEnabled(config.readEntry(pluginName() + s_enabledKey(), isPluginEnabledByDefault()));
0738     } else {
0739         if (!d->config.isValid()) {
0740             qCWarning(SERVICES) << "no KConfigGroup, cannot load";
0741             return;
0742         }
0743         setPluginEnabled(d->config.readEntry(pluginName() + s_enabledKey(), isPluginEnabledByDefault()));
0744     }
0745 }
0746 
0747 void KPluginInfo::defaults()
0748 {
0749     // qDebug() << Q_FUNC_INFO;
0750     setPluginEnabled(isPluginEnabledByDefault());
0751 }
0752 
0753 uint qHash(const KPluginInfo &p)
0754 {
0755     return qHash(reinterpret_cast<quint64>(p.d.data()));
0756 }
0757 
0758 KPluginInfo KPluginInfo::fromMetaData(const KPluginMetaData &md)
0759 {
0760     return KPluginInfo(md);
0761 }
0762 
0763 KPluginMetaData KPluginInfo::toMetaData() const
0764 {
0765     KPLUGININFO_ISVALID_ASSERTION;
0766     return d->metaData;
0767 }
0768 
0769 KPluginMetaData KPluginInfo::toMetaData(const KPluginInfo &info)
0770 {
0771     return info.toMetaData();
0772 }
0773 
0774 KPluginInfo::List KPluginInfo::fromMetaData(const QVector<KPluginMetaData> &list)
0775 {
0776     KPluginInfo::List ret;
0777     ret.reserve(list.size());
0778     for (const KPluginMetaData &md : list) {
0779         ret.append(KPluginInfo::fromMetaData(md));
0780     }
0781     return ret;
0782 }
0783 
0784 QVector<KPluginMetaData> KPluginInfo::toMetaData(const KPluginInfo::List &list)
0785 {
0786     QVector<KPluginMetaData> ret;
0787     ret.reserve(list.size());
0788     for (const KPluginInfo &info : list) {
0789         ret.append(info.toMetaData());
0790     }
0791     return ret;
0792 }
0793 
0794 #undef KPLUGININFO_ISVALID_ASSERTION
0795 #endif