Cache device classes and keep devices with missing plugins in the system

pull/265/head
Michael Zanetti 2020-02-09 19:05:53 +01:00
parent 6e4a0bc84b
commit 68e9fe1ba1
6 changed files with 166 additions and 36 deletions

View File

@ -0,0 +1,52 @@
#include "deviceclasscache.h"
#include <QSettings>
#include <QStandardPaths>
#include <QDir>
#include <QJsonDocument>
#include <QJsonObject>
#include "loggingcategories.h"
PluginInfoCache::PluginInfoCache()
{
}
void PluginInfoCache::cachePluginInfo(const QJsonObject &metaData)
{
QString fileName = metaData.value("id").toString().remove(QRegExp("[{}]")) + ".cache";
QDir path = QStandardPaths::writableLocation(QStandardPaths::CacheLocation) + "/plugininfo/";
if (!path.exists()) {
if (!path.mkpath(path.absolutePath())) {
qCWarning(dcDeviceManager()) << "Error creating device class cache dir at" << path.absolutePath();
}
}
QFile file(path.absoluteFilePath(fileName));
if (!file.open(QFile::WriteOnly | QFile::Truncate)) {
qCWarning(dcDeviceManager()) << "Error opening device class cache for writing at" << path.absoluteFilePath(fileName);
return;
}
file.write(QJsonDocument::fromVariant(metaData.toVariantMap()).toJson(QJsonDocument::Compact));
file.close();
}
QJsonObject PluginInfoCache::loadPluginInfo(const PluginId &pluginId)
{
QString fileName = pluginId.toString().remove(QRegExp("[{}]")) + ".cache";
QDir path = QStandardPaths::writableLocation(QStandardPaths::CacheLocation) + "/plugininfo/";
QFile file(path.absoluteFilePath(fileName));
if (!file.open(QFile::ReadOnly)) {
return QJsonObject();
}
QByteArray data = file.readAll();
QJsonParseError error;
QJsonDocument jsonDoc = QJsonDocument::fromJson(data, &error);
if (error.error != QJsonParseError::NoError) {
qCWarning(dcDeviceManager()) << "Error parsing plugin info cache entry:" << path.absoluteFilePath(fileName);
return QJsonObject();
}
return QJsonObject::fromVariantMap(jsonDoc.toVariant().toMap());
}

View File

@ -0,0 +1,16 @@
#ifndef PLUGININFOCACHE_H
#define PLUGININFOCACHE_H
#include "types/deviceclass.h"
#include "devices/deviceplugin.h"
class PluginInfoCache
{
public:
PluginInfoCache();
static void cachePluginInfo(const QJsonObject &metaData);
static QJsonObject loadPluginInfo(const PluginId &pluginId);
};
#endif // PLUGININFOCACHE_H

View File

@ -38,6 +38,7 @@
#include "typeutils.h"
#include "nymeasettings.h"
#include "version.h"
#include "deviceclasscache.h"
#include "devices/devicediscoveryinfo.h"
#include "devices/devicepairinginfo.h"
@ -258,7 +259,7 @@ DeviceDiscoveryInfo* DeviceManagerImplementation::discoverDevices(const DeviceCl
if (!plugin) {
qCWarning(dcDeviceManager) << "Device discovery failed. Plugin not found for device class" << deviceClass.name();
DeviceDiscoveryInfo *discoveryInfo = new DeviceDiscoveryInfo(deviceClassId, params, this);
discoveryInfo->finish(Device::DeviceErrorPluginNotFound);
discoveryInfo->finish(Device::DeviceErrorPluginNotFound, tr("The plugin for this device or service is not loaded."));
return discoveryInfo;
}
@ -594,9 +595,10 @@ DevicePairingInfo *DeviceManagerImplementation::confirmPairing(const PairingTran
PairingContext context = m_pendingPairings.take(pairingTransactionId);
DeviceClassId deviceClassId = context.deviceClassId;
DevicePlugin *plugin = m_devicePlugins.value(m_supportedDevices.value(deviceClassId).pluginId());
DeviceClass deviceClass = m_supportedDevices.value(deviceClassId);
DevicePlugin *plugin = m_devicePlugins.value(deviceClass.pluginId());
if (!plugin) {
qCWarning(dcDeviceManager) << "Can't find a plugin for this device class";
qCWarning(dcDeviceManager) << "Can't find a plugin for this device class:" << deviceClass;
DevicePairingInfo *info = new DevicePairingInfo(pairingTransactionId, deviceClassId, context.deviceId, context.deviceName, context.params, context.parentDeviceId, this);
info->finish(Device::DeviceErrorPluginNotFound);
return info;
@ -633,7 +635,7 @@ DevicePairingInfo *DeviceManagerImplementation::confirmPairing(const PairingTran
Device *device = nullptr;
if (addNewDevice) {
device = new Device(plugin, deviceClass, internalInfo->deviceId(), this);
device = new Device(plugin->pluginId(), deviceClass, internalInfo->deviceId(), this);
if (internalInfo->deviceName().isEmpty()) {
device->setName(deviceClass.displayName());
} else {
@ -735,7 +737,7 @@ DeviceSetupInfo* DeviceManagerImplementation::addConfiguredDeviceInternal(const
return info;
}
Device *device = new Device(plugin, deviceClass, deviceId, this);
Device *device = new Device(plugin->pluginId(), deviceClass, deviceId, this);
device->setParentId(parentDeviceId);
if (name.isEmpty()) {
device->setName(deviceClass.name());
@ -809,13 +811,24 @@ BrowseResult *DeviceManagerImplementation::browseDevice(const DeviceId &deviceId
return result;
}
DevicePlugin *plugin = m_devicePlugins.value(device->pluginId());
if (!plugin) {
qCWarning(dcDeviceManager()) << "Cannot browse device. Plugin not found for device" << device;
return result;
}
if (!device->setupComplete()) {
qCWarning(dcDeviceManager()) << "Cannot browse device. Device did not finish setup" << device;
return result;
}
if (!device->deviceClass().browsable()) {
qCWarning(dcDeviceManager()) << "Cannot browse device. DeviceClass" << device->deviceClass().name() << "is not browsable.";
result->finish(Device::DeviceErrorUnsupportedFeature);
return result;
}
device->plugin()->browseDevice(result);
plugin->browseDevice(result);
connect(result, &BrowseResult::finished, this, [result](){
if (result->status() != Device::DeviceErrorNoError) {
qCWarning(dcDeviceManager()) << "Browse device failed:" << result->status();
@ -836,13 +849,24 @@ BrowserItemResult *DeviceManagerImplementation::browserItemDetails(const DeviceI
return result;
}
DevicePlugin *plugin = m_devicePlugins.value(device->pluginId());
if (!plugin) {
qCWarning(dcDeviceManager()) << "Cannot browse device. Plugin not found for device" << device;
return result;
}
if (device->setupStatus() != Device::DeviceSetupStatusComplete) {
qCWarning(dcDeviceManager()) << "Cannot browse device. Device did not finish setup" << device;
return result;
}
if (!device->deviceClass().browsable()) {
qCWarning(dcDeviceManager()) << "Cannot browse device. DeviceClass" << device->deviceClass().name() << "is not browsable.";
result->finish(Device::DeviceErrorUnsupportedFeature);
return result;
}
device->plugin()->browserItem(result);
plugin->browserItem(result);
connect(result, &BrowserItemResult::finished, this, [result](){
if (result->status() != Device::DeviceErrorNoError) {
qCWarning(dcDeviceManager()) << "Browse device failed:" << result->status();
@ -862,11 +886,24 @@ BrowserActionInfo* DeviceManagerImplementation::executeBrowserItem(const Browser
return info;
}
DevicePlugin *plugin = m_devicePlugins.value(device->pluginId());
if (!plugin) {
qCWarning(dcDeviceManager()) << "Cannot browse device. Plugin not found for device" << device;
info->finish(Device::DeviceErrorPluginNotFound);
return info;
}
if (device->setupStatus() != Device::DeviceSetupStatusComplete) {
qCWarning(dcDeviceManager()) << "Cannot browse device. Device did not finish setup" << device;
info->finish(Device::DeviceErrorSetupFailed);
return info;
}
if (!device->deviceClass().browsable()) {
info->finish(Device::DeviceErrorUnsupportedFeature);
return info;
}
device->plugin()->executeBrowserItem(info);
plugin->executeBrowserItem(info);
return info;
}
@ -881,13 +918,26 @@ BrowserItemActionInfo* DeviceManagerImplementation::executeBrowserItemAction(con
return info;
}
DevicePlugin *plugin = m_devicePlugins.value(device->pluginId());
if (!plugin) {
qCWarning(dcDeviceManager()) << "Cannot execute browser item action. Plugin not found for device" << device;
info->finish(Device::DeviceErrorPluginNotFound);
return info;
}
if (device->setupStatus() != Device::DeviceSetupStatusComplete) {
qCWarning(dcDeviceManager()) << "Cannot execute browser item action. Device did not finish setup" << device;
info->finish(Device::DeviceErrorSetupFailed);
return info;
}
if (!device->deviceClass().browsable()) {
info->finish(Device::DeviceErrorUnsupportedFeature);
return info;
}
// TODO: check browserItemAction.params with deviceClass
device->plugin()->executeBrowserItemAction(info);
plugin->executeBrowserItemAction(info);
return info;
}
@ -1099,7 +1149,8 @@ void DeviceManagerImplementation::loadPlugins()
continue;
}
PluginMetadata metaData(loader.metaData().value("MetaData").toObject());
QJsonObject pluginInfo = loader.metaData().value("MetaData").toObject();
PluginMetadata metaData(pluginInfo);
if (!metaData.isValid()) {
foreach (const QString &error, metaData.validationErrors()) {
qCWarning(dcDeviceManager()) << error;
@ -1119,6 +1170,7 @@ void DeviceManagerImplementation::loadPlugins()
continue;
}
loadPlugin(pluginIface, metaData);
PluginInfoCache::cachePluginInfo(pluginInfo);
}
}
@ -1249,13 +1301,28 @@ void DeviceManagerImplementation::loadConfiguredDevices()
foreach (const QString &idString, settings.childGroups()) {
settings.beginGroup(idString);
QString deviceName = settings.value("devicename").toString();
DevicePlugin *plugin = m_devicePlugins.value(PluginId(settings.value("pluginid").toString()));
PluginId pluginId = PluginId(settings.value("pluginid").toString());
DevicePlugin *plugin = m_devicePlugins.value(pluginId);
if (!plugin) {
qCWarning(dcDeviceManager()) << "Not loading device" << deviceName << idString << "because the plugin for this device could not be found.";
settings.endGroup(); // DeviceId
continue;
qCWarning(dcDeviceManager()) << "Plugin for device" << deviceName << idString << "not found. This device will not be functional until the plugin can be loaded.";
}
DeviceClassId deviceClassId = DeviceClassId(settings.value("deviceClassId").toString());
DeviceClass deviceClass = findDeviceClass(deviceClassId);
if (!deviceClass.isValid()) {
// Try to load the device class from the cache
QJsonObject pluginInfo = PluginInfoCache::loadPluginInfo(pluginId);
if (!pluginInfo.empty()) {
PluginMetadata pluginMetadata(pluginInfo);
deviceClass = pluginMetadata.deviceClasses().findById(deviceClassId);
if (deviceClass.isValid()) {
m_supportedDevices.insert(deviceClassId, deviceClass);
if (!m_supportedVendors.contains(deviceClass.vendorId())) {
Vendor vendor = pluginMetadata.vendors().findById(deviceClass.vendorId());
m_supportedVendors.insert(vendor.id(), vendor);
}
}
}
}
DeviceClass deviceClass = findDeviceClass(DeviceClassId(settings.value("deviceClassId").toString()));
if (!deviceClass.isValid()) {
qCWarning(dcDeviceManager()) << "Not loading device" << deviceName << idString << "because the device class for this device could not be found.";
settings.endGroup(); // DeviceId
@ -1263,12 +1330,12 @@ void DeviceManagerImplementation::loadConfiguredDevices()
}
// Cross-check if this plugin still implements this device class
if (!plugin->supportedDevices().contains(deviceClass)) {
if (plugin && !plugin->supportedDevices().contains(deviceClass)) {
qCWarning(dcDeviceManager()) << "Not loading device" << deviceName << idString << "because plugin" << plugin->pluginName() << "has removed support for it.";
settings.endGroup(); // DeviceId
continue;
}
Device *device = new Device(plugin, deviceClass, DeviceId(idString), this);
Device *device = new Device(pluginId, deviceClass, DeviceId(idString), this);
device->m_autoCreated = settings.value("autoCreated").toBool();
device->setName(deviceName);
device->setParentId(DeviceId(settings.value("parentid", QUuid()).toString()));
@ -1463,7 +1530,7 @@ void DeviceManagerImplementation::onAutoDevicesAppeared(const DeviceDescriptors
continue;
}
device = new Device(plugin, deviceClass, this);
device = new Device(plugin->pluginId(), deviceClass, this);
device->m_autoCreated = true;
device->setName(deviceDescriptor.title());
device->setParams(deviceDescriptor.params());
@ -1655,9 +1722,9 @@ DeviceSetupInfo* DeviceManagerImplementation::setupDevice(Device *device)
DevicePlugin *plugin = m_devicePlugins.value(deviceClass.pluginId());
if (!plugin) {
qCWarning(dcDeviceManager) << "Can't find a plugin for this device" << device->id();
DeviceSetupInfo *info = new DeviceSetupInfo(nullptr, this);
info->finish(Device::DeviceErrorPluginNotFound);
qCWarning(dcDeviceManager) << "Can't find a plugin for this device" << device;
DeviceSetupInfo *info = new DeviceSetupInfo(device, this);
info->finish(Device::DeviceErrorPluginNotFound, tr("The plugin for this device or service is not loaded."));
return info;
}

View File

@ -16,6 +16,7 @@ RESOURCES += $$top_srcdir/icons.qrc \
HEADERS += nymeacore.h \
devices/deviceclasscache.h \
devices/devicemanagerimplementation.h \
devices/translator.h \
experiences/experiencemanager.h \
@ -100,6 +101,7 @@ HEADERS += nymeacore.h \
SOURCES += nymeacore.cpp \
devices/deviceclasscache.cpp \
devices/devicemanagerimplementation.cpp \
devices/translator.cpp \
experiences/experiencemanager.cpp \

View File

@ -119,20 +119,20 @@
#include <QDebug>
/*! Construct an Device with the given \a pluginId, \a id, \a deviceClassId and \a parent. */
Device::Device(DevicePlugin *plugin, const DeviceClass &deviceClass, const DeviceId &id, QObject *parent):
Device::Device(const PluginId &pluginId, const DeviceClass &deviceClass, const DeviceId &id, QObject *parent):
QObject(parent),
m_deviceClass(deviceClass),
m_plugin(plugin),
m_pluginId(pluginId),
m_id(id)
{
}
/*! Construct an Device with the given \a pluginId, \a deviceClassId and \a parent. A new DeviceId will be created for this Device. */
Device::Device(DevicePlugin *plugin, const DeviceClass &deviceClass, QObject *parent):
Device::Device(const PluginId &pluginId, const DeviceClass &deviceClass, QObject *parent):
QObject(parent),
m_deviceClass(deviceClass),
m_plugin(plugin),
m_pluginId(pluginId),
m_id(DeviceId::createDeviceId())
{
@ -153,7 +153,7 @@ DeviceClassId Device::deviceClassId() const
/*! Returns the id of the \l{DevicePlugin} this Device is managed by. */
PluginId Device::pluginId() const
{
return m_plugin->pluginId();
return m_pluginId;
}
/*! Returns the \l{DeviceClass} of this device. */
@ -162,12 +162,6 @@ DeviceClass Device::deviceClass() const
return m_deviceClass;
}
/*! Returns the the \l{DevicePlugin} this Device is managed by. */
DevicePlugin *Device::plugin() const
{
return m_plugin;
}
/*! Returns the name of this Device. This is visible to the user. */
QString Device::name() const
{

View File

@ -105,7 +105,6 @@ public:
PluginId pluginId() const;
DeviceClass deviceClass() const;
DevicePlugin* plugin() const;
QString name() const;
void setName(const QString &name);
@ -153,14 +152,14 @@ signals:
private:
friend class DeviceManager;
friend class DeviceManagerImplementation;
Device(DevicePlugin *plugin, const DeviceClass &deviceClass, const DeviceId &id, QObject *parent = nullptr);
Device(DevicePlugin *plugin, const DeviceClass &deviceClass, QObject *parent = nullptr);
Device(const PluginId &pluginId, const DeviceClass &deviceClass, const DeviceId &id, QObject *parent = nullptr);
Device(const PluginId &pluginId, const DeviceClass &deviceClass, QObject *parent = nullptr);
void setSetupStatus(Device::DeviceSetupStatus status, Device::DeviceError setupError, const QString &displayMessage = QString());
private:
DeviceClass m_deviceClass;
DevicePlugin* m_plugin = nullptr;
PluginId m_pluginId;
DeviceId m_id;
DeviceId m_parentId;