/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright (C) 2016 Simon Stürz * * Copyright (C) 2016 Michael Zanetti * * * * This file is part of nymea. * * * * This library is free software; you can redistribute it and/or * * modify it under the terms of the GNU Lesser General Public * * License as published by the Free Software Foundation; either * * version 2.1 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 * * Lesser General Public License for more details. * * * * You should have received a copy of the GNU Lesser General Public * * License along with this library; If not, see * * . * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #include "devicepluginnetworkdetector.h" #include "devices/device.h" #include "plugininfo.h" #include #include DevicePluginNetworkDetector::DevicePluginNetworkDetector() { m_broadcastPing = new BroadcastPing(this); connect(m_broadcastPing, &BroadcastPing::finished, this, &DevicePluginNetworkDetector::broadcastPingFinished); } DevicePluginNetworkDetector::~DevicePluginNetworkDetector() { } void DevicePluginNetworkDetector::init() { } void DevicePluginNetworkDetector::setupDevice(DeviceSetupInfo *info) { Device *device = info->device(); qCDebug(dcNetworkDetector()) << "Setup" << device->name() << device->params(); DeviceMonitor *monitor = new DeviceMonitor(device->name(), device->paramValue(networkDeviceDeviceMacAddressParamTypeId).toString(), device->paramValue(networkDeviceDeviceAddressParamTypeId).toString(), device->stateValue(networkDeviceIsPresentStateTypeId).toBool(), this); connect(monitor, &DeviceMonitor::reachableChanged, this, &DevicePluginNetworkDetector::deviceReachableChanged); connect(monitor, &DeviceMonitor::addressChanged, this, &DevicePluginNetworkDetector::deviceAddressChanged); connect(monitor, &DeviceMonitor::seen, this, &DevicePluginNetworkDetector::deviceSeen); monitor->setGracePeriod(device->setting(networkDeviceSettingsGracePeriodParamTypeId).toInt()); m_monitors.insert(monitor, device); connect(device, &Device::settingChanged, this, [this, device](const ParamTypeId ¶mTypeId, const QVariant &value){ if (paramTypeId != networkDeviceSettingsGracePeriodParamTypeId) { return; } DeviceMonitor *monitor = m_monitors.key(device); if (monitor) { monitor->setGracePeriod(value.toInt()); } }); if (!m_pluginTimer) { m_pluginTimer = hardwareManager()->pluginTimerManager()->registerTimer(30); connect(m_pluginTimer, &PluginTimer::timeout, m_broadcastPing, &BroadcastPing::run); m_broadcastPing->run(); } info->finish(Device::DeviceErrorNoError); } void DevicePluginNetworkDetector::discoverDevices(DeviceDiscoveryInfo *info) { Discovery *discovery = new Discovery(this); discovery->discoverHosts(25); // clean up discovery object when this discovery info is deleted connect(info, &DeviceDiscoveryInfo::destroyed, discovery, &Discovery::deleteLater); connect(discovery, &Discovery::finished, info, [this, info](const QList &hosts) { qCDebug(dcNetworkDetector()) << "Discovery finished. Found" << hosts.count() << "devices"; foreach (const Host &host, hosts) { DeviceDescriptor descriptor(networkDeviceDeviceClassId, host.hostName().isEmpty() ? host.address() : host.hostName(), host.address() + " (" + host.macAddress() + ")"); foreach (Device *existingDevice, myDevices()) { if (existingDevice->paramValue(networkDeviceDeviceMacAddressParamTypeId).toString() == host.macAddress()) { descriptor.setDeviceId(existingDevice->id()); break; } } ParamList params; params << Param(networkDeviceDeviceMacAddressParamTypeId, host.macAddress()); params << Param(networkDeviceDeviceAddressParamTypeId, host.address()); descriptor.setParams(params); info->addDeviceDescriptor(descriptor); } info->finish(Device::DeviceErrorNoError); }); } void DevicePluginNetworkDetector::deviceRemoved(Device *device) { DeviceMonitor *monitor = m_monitors.key(device); m_monitors.remove(monitor); delete monitor; if (m_monitors.isEmpty()) { hardwareManager()->pluginTimerManager()->unregisterTimer(m_pluginTimer); m_pluginTimer = nullptr; } } void DevicePluginNetworkDetector::broadcastPingFinished() { foreach (DeviceMonitor *monitor, m_monitors.keys()) { monitor->update(); } } void DevicePluginNetworkDetector::deviceReachableChanged(bool reachable) { DeviceMonitor *monitor = static_cast(sender()); Device *device = m_monitors.value(monitor); if (device->stateValue(networkDeviceIsPresentStateTypeId).toBool() != reachable) { qCDebug(dcNetworkDetector()) << "Device" << device->name() << device->paramValue(networkDeviceDeviceMacAddressParamTypeId).toString() << "reachable changed" << reachable; device->setStateValue(networkDeviceIsPresentStateTypeId, reachable); } } void DevicePluginNetworkDetector::deviceAddressChanged(const QString &address) { DeviceMonitor *monitor = static_cast(sender()); Device *device = m_monitors.value(monitor); if (device->paramValue(networkDeviceDeviceAddressParamTypeId).toString() != address) { qCDebug(dcNetworkDetector()) << "Device" << device->name() << device->paramValue(networkDeviceDeviceMacAddressParamTypeId).toString() << "changed IP address to" << address; device->setParamValue(networkDeviceDeviceAddressParamTypeId, address); } } void DevicePluginNetworkDetector::deviceSeen() { DeviceMonitor *monitor = static_cast(sender()); Device *device = m_monitors.value(monitor); QDateTime oldLastSeen = QDateTime::fromTime_t(device->stateValue(networkDeviceLastSeenTimeStateTypeId).toInt()); if (oldLastSeen.addSecs(60) < QDateTime::currentDateTime()) { device->setStateValue(networkDeviceLastSeenTimeStateTypeId, QDateTime::currentDateTime().toTime_t()); } }