/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * This file is part of guh. * * * * Guh is free software: you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation, version 2 of the License. * * * * Guh 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 General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with guh. If not, see . * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #include "devicepluginphilipshue.h" #include "plugin/device.h" #include "devicemanager.h" #include "types/param.h" #include "huebridgeconnection.h" #include #include #include #include #include #include #include VendorId hueVendorId = VendorId("0ae1e001-2aa6-47ed-b8c0-334c3728a68f"); PluginId huePluginUuid = PluginId("5f2e634b-b7f3-48ee-976a-b5ae22aa5c55"); DeviceClassId hueDeviceClassId = DeviceClassId("d8f4c397-e05e-47c1-8917-8e72d4d0d47c"); StateTypeId hueColorStateTypeId = StateTypeId("d25423e7-b924-4b20-80b6-77eecc65d089"); ActionTypeId setHueColorActionTypeId = ActionTypeId("29cc299a-818b-47b2-817f-c5a6361545e4"); DevicePluginPhilipsHue::DevicePluginPhilipsHue(): m_discovery(new Discovery(this)) { connect(m_discovery, &Discovery::discoveryDone, this, &DevicePluginPhilipsHue::discoveryDone); m_nam = new QNetworkAccessManager(this); } QList DevicePluginPhilipsHue::supportedVendors() const { QList ret; Vendor philips(hueVendorId, "Philips"); ret.append(philips); return ret; } QList DevicePluginPhilipsHue::supportedDevices() const { QList ret; DeviceClass deviceClassHue(pluginId(), hueVendorId, hueDeviceClassId); deviceClassHue.setName("Hue"); deviceClassHue.setCreateMethod(DeviceClass::CreateMethodDiscovery); deviceClassHue.setSetupMethod(DeviceClass::SetupMethodPushButton); deviceClassHue.setPairingInfo("Please press the button on the Hue bridge and then press OK"); QList paramTypes; ParamType ipParam("ip", QVariant::String); paramTypes.append(ipParam); deviceClassHue.setParamTypes(paramTypes); QList hueStates; StateType colorState(hueColorStateTypeId); colorState.setName("color"); colorState.setType(QVariant::Color); colorState.setDefaultValue(QColor(Qt::black)); hueStates.append(colorState); deviceClassHue.setStateTypes(hueStates); QList hueActons; ActionType setColorAction(setHueColorActionTypeId); setColorAction.setName("Set color"); QList actionParamsSetColor; ParamType actionParamSetColor("color", QVariant::Color); actionParamsSetColor.append(actionParamSetColor); setColorAction.setParameters(actionParamsSetColor); hueActons.append(setColorAction); deviceClassHue.setActions(hueActons); ret.append(deviceClassHue); return ret; } DeviceManager::HardwareResources DevicePluginPhilipsHue::requiredHardware() const { return DeviceManager::HardwareResourceNone; } bool DevicePluginPhilipsHue::configureAutoDevice(QList loadedDevices, Device *device) const { // if (!m_bobClient->connected()) { // return false; // } // if (loadedDevices.count() < m_bobClient->lightsCount()) { // int index = loadedDevices.count(); // device->setName("Boblight Channel " + QString::number(index)); // QList params; // Param param("channel"); // param.setValue(index); // params.append(param); // device->setParams(params); // device->setStateValue(colorStateTypeId, m_bobClient->currentColor(index)); // return true; // } return false; } QString DevicePluginPhilipsHue::pluginName() const { return "Philips Hue"; } PluginId DevicePluginPhilipsHue::pluginId() const { return huePluginUuid; } QList DevicePluginPhilipsHue::configurationDescription() const { QList params; return params; } DeviceManager::DeviceError DevicePluginPhilipsHue::discoverDevices(const DeviceClassId &deviceClassId, const QVariantMap ¶ms) const { m_discovery->findBridges(4000); return DeviceManager::DeviceErrorAsync; } QPair DevicePluginPhilipsHue::confirmPairing(const QUuid &pairingTransactionId, const DeviceClassId &deviceClassId, const QList ¶ms) { Param ipParam; foreach (const Param ¶m, params) { if (param.name() == "ip") { ipParam = param; } } if (!ipParam.isValid()) { return reportDeviceSetup(DeviceManager::DeviceSetupStatusFailure, "Missing parameter: ip"); } QString username = "guh-" + QUuid::createUuid().toString().remove(QRegExp("[\\{\\}]*")).remove(QRegExp("\\-[0-9a-f\\-]*")); QVariantMap createUserParams; createUserParams.insert("devicetype", "guh"); createUserParams.insert("username", username); QJsonDocument jsonDoc = QJsonDocument::fromVariant(createUserParams); QByteArray data = jsonDoc.toJson(); QNetworkRequest request(QUrl("http://" + ipParam.value().toString() + "/api")); QNetworkReply *reply = m_nam->post(request, data); connect(reply, &QNetworkReply::finished, this, &DevicePluginPhilipsHue::createUserFinished); HueBridgeConnection *bridge = new HueBridgeConnection(QHostAddress(ipParam.value().toString()), username); m_pairings.insert(reply, qMakePair(pairingTransactionId, bridge)); return reportDeviceSetup(DeviceManager::DeviceSetupStatusAsync); } QPair DevicePluginPhilipsHue::executeAction(Device *device, const Action &action) { // if (!m_bobClient->connected()) { return report(DeviceManager::DeviceErrorSetupFailed, device->id().toString()); // } // QColor newColor = action.param("color").value().value(); // if (!newColor.isValid()) { // return report(DeviceManager::DeviceErrorActionParameterError, "color"); // } // qDebug() << "executing boblight action" << newColor; // m_bobClient->setColor(device->paramValue("channel").toInt(), newColor); // m_bobClient->sync(); // device->setStateValue(colorStateTypeId, newColor); // return report(); } void DevicePluginPhilipsHue::discoveryDone(const QList &bridges) { qDebug() << "discovered bridges" << bridges.count(); QList deviceDescriptors; foreach (const QHostAddress &bridge, bridges) { DeviceDescriptor descriptor(hueDeviceClassId, "Philips Hue bridge", bridge.toString()); QList params; Param param("ip", bridge.toString()); params.append(param); descriptor.setParams(params); deviceDescriptors.append(descriptor); } emit devicesDiscovered(hueDeviceClassId, deviceDescriptors); } void DevicePluginPhilipsHue::createUserFinished() { QNetworkReply *reply = static_cast(sender()); QByteArray data = reply->readAll(); QPair pair = m_pairings.take(reply); QUuid pairingTransactionId = pair.first; HueBridgeConnection *bridge = pair.second; QJsonParseError error; QJsonDocument jsonDoc = QJsonDocument::fromJson(data, &error); if (error.error != QJsonParseError::NoError) { emit pairingFinished(pairingTransactionId, DeviceManager::DeviceSetupStatusFailure, "Pairing failed. Failed to parse response from Hue Bridge."); delete bridge; return; } QVariantMap response = jsonDoc.toVariant().toList().first().toMap(); if (response.contains("error")) { qDebug() << "Failed to pair Hue bridge:" << response.value("error").toMap().value("description"); emit pairingFinished(pairingTransactionId, DeviceManager::DeviceSetupStatusFailure, "Pairing failed:" + response.value("error").toMap().value("description").toString()); delete bridge; return; } emit pairingFinished(pairingTransactionId, DeviceManager::DeviceSetupStatusSuccess, QString()); m_bridges.append(bridge); qDebug() << "response" << response << data; }