#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include WiFiManager wifiManager; Settings settings; MiLightClient* milightClient = NULL; MiLightRadioFactory* radioFactory = NULL; MiLightHttpServer *httpServer = NULL; MqttClient* mqttClient = NULL; MiLightDiscoveryServer* discoveryServer = NULL; uint8_t currentRadioType = 0; GroupStateCache* stateCache = new GroupStateCache(100); int numUdpServers = 0; MiLightUdpServer** udpServers; WiFiUDP udpSeder; void initMilightUdpServers() { if (udpServers) { for (int i = 0; i < numUdpServers; i++) { if (udpServers[i]) { delete udpServers[i]; } } delete udpServers; } udpServers = new MiLightUdpServer*[settings.numGatewayConfigs]; numUdpServers = settings.numGatewayConfigs; for (size_t i = 0; i < settings.numGatewayConfigs; i++) { GatewayConfig* config = settings.gatewayConfigs[i]; MiLightUdpServer* server = MiLightUdpServer::fromVersion( config->protocolVersion, milightClient, config->port, config->deviceId ); if (server == NULL) { Serial.print(F("Error creating UDP server with protocol version: ")); Serial.println(config->protocolVersion); } else { udpServers[i] = server; udpServers[i]->begin(); } } } void onPacketSentHandler(uint8_t* packet, const MiLightRemoteConfig& config) { StaticJsonBuffer<200> buffer; JsonObject& result = buffer.createObject(); config.packetFormatter->parsePacket(packet, result, stateCache); if (!result.containsKey("device_id") ||!result.containsKey("group_id") ||!result.containsKey("device_type")) { Serial.println(F("Skipping update because packet formatter didn't supply necessary information.")); return; } uint16_t deviceId = result["device_id"]; uint16_t groupId = result["group_id"]; const MiLightRemoteConfig* remoteConfig = MiLightRemoteConfig::fromType(result.get("device_type")); GroupId bulbId(deviceId, groupId, remoteConfig->type); GroupState* groupState = stateCache->get(bulbId); groupState->patch(result); groupState->applyState(result); char output[200]; result.printTo(output); if (mqttClient) { mqttClient->sendUpdate(config, deviceId, groupId, output); } httpServer->handlePacketSent(packet, config); } void handleListen() { if (! settings.listenRepeats) { return; } MiLightRadio* radio = milightClient->switchRadio(currentRadioType++ % milightClient->getNumRadios()); for (size_t i = 0; i < settings.listenRepeats; i++) { if (milightClient->available()) { uint8_t readPacket[MILIGHT_MAX_PACKET_LENGTH]; size_t packetLen = milightClient->read(readPacket); const MiLightRemoteConfig* remoteConfig = MiLightRemoteConfig::fromReceivedPacket( radio->config(), readPacket, packetLen ); if (remoteConfig == NULL) { Serial.println(F("ERROR: Couldn't find remote for received packet!")); return; } onPacketSentHandler(readPacket, *remoteConfig); } } } void applySettings() { if (milightClient) { delete milightClient; } if (radioFactory) { delete radioFactory; } if (mqttClient) { delete mqttClient; } radioFactory = MiLightRadioFactory::fromSettings(settings); if (radioFactory == NULL) { Serial.println(F("ERROR: unable to construct radio factory")); } milightClient = new MiLightClient(radioFactory); milightClient->begin(); milightClient->onPacketSent(onPacketSentHandler); if (settings.mqttServer().length() > 0) { mqttClient = new MqttClient(settings, milightClient); mqttClient->begin(); } initMilightUdpServers(); if (discoveryServer) { delete discoveryServer; discoveryServer = NULL; } if (settings.discoveryPort != 0) { discoveryServer = new MiLightDiscoveryServer(settings); discoveryServer->begin(); } } bool shouldRestart() { if (! settings.isAutoRestartEnabled()) { return false; } return settings.getAutoRestartPeriod()*60*1000 < millis(); } void setup() { Serial.begin(9600); wifiManager.autoConnect(); SPIFFS.begin(); Settings::load(settings); applySettings(); if (! MDNS.begin("milight-hub")) { Serial.println(F("Error setting up MDNS responder")); } MDNS.addService("http", "tcp", 80); SSDP.setSchemaURL("description.xml"); SSDP.setHTTPPort(80); SSDP.setName("ESP8266 MiLight Gateway"); SSDP.setSerialNumber(ESP.getChipId()); SSDP.setURL("/"); SSDP.setDeviceType("upnp:rootdevice"); SSDP.begin(); httpServer = new MiLightHttpServer(settings, milightClient); httpServer->onSettingsSaved(applySettings); httpServer->on("/description.xml", HTTP_GET, []() { SSDP.schema(httpServer->client()); }); httpServer->begin(); Serial.println(F("Setup complete")); } void loop() { httpServer->handleClient(); if (mqttClient) { mqttClient->handleClient(); } if (udpServers) { for (size_t i = 0; i < settings.numGatewayConfigs; i++) { udpServers[i]->handleClient(); } } if (discoveryServer) { discoveryServer->handleClient(); } handleListen(); if (shouldRestart()) { Serial.println(F("Auto-restart triggered. Restarting...")); ESP.restart(); } }