This repository provides IoT dongle platform over Wi-Fi.
IoT dongle enables the device to connect the Internet and get the Internet service.
In general, home appliances have long life-cycle, more than 10 years. Then, which air-conditioner do you want to buy? One is not connected to the Internet and you cannot get the innovated service for more than 10 years. The other is connected to the Internet and you need to pay more money. As of now, there is no killer app to force you to buy the Internet connected device.
IoT dongle is another approach. Now, you buy the IoT dongle ready device. Later, you may buy the IoT dongle to enable you to get the valuable service from the Internet, separately.
The below shows the architecture of IoT dongle platform. IoT dongle platform includes device and service broker which are bridging between the device and the Internet service. The common layer with IoT dongle protocol provides the base of the device and the dongle. Device agent running on the device works for the real device function. The device and the dongle communicate through UART (serial). The dongle and the Internet service communicate through Wi-Fi. The device can be on ARDUINO (w/o Internet) and the dongle can be on ESP8266 (Wi-Fi SoC).
The device is implemented using ARDUINO UNO. And the dongle is on ESP8266 as shown the below drawing. The device provides power (VCC/GND) to the dongle and they are communicating through UART (RX/TX). You can get more detail information from ESP8266.com/arduino.
You can easily download IoTDongle.zip and install it on your ARDUINO IDE, using the menu - Sketch / Include Library / Add .ZIP Library. You can get more detail guide how to install ARDUINO library from arduino.cc.
You can get examples of IoT dongle from the menu - File / Examples / IoTDongle. Open AirConditioner in device examples. The below shows the skeleton of AirConditioner device.
#include <IoTDongle.h>
using namespace IoTD;
AirConditioner device;
const byte POWER_OFF = 0;
const byte POWER_ON = 1;
byte power = POWER_OFF;
void CH_GET_POWER(Command cmd, Method api, Resource res)
{
Device::sendCommand(COMMAND_RETURN, api, res);
Device::sendData(power);
}
void setup() {
device.setCallback(COMMAND_CALL, METHOD_GET, AirConditioner::RESOURCE_POWER, CH_GET_POWER);
device.begin();
}
void loop() {
device.loop();
}
Here is the step to make the device to support IoT dongle.
-
Include "IoTDongle.h" and use namespace IoTD.
#include <IoTDongle.h> using namespace IoTD;
-
Ddeclare the device that you want to develop.
AirConditioner device;
-
Set callbacks and begin the device in setup().
void CH_GET_POWER(Command cmd, Method api, Resource res) { Device::sendCommand(COMMAND_RETURN, api, res); Device::sendData(power); } void setup() { device.setCallback(COMMAND_CALL, METHOD_GET, AirConditioner::RESOURCE_POWER, CH_GET_POWER); device.begin(); }
-
Call device loop in ARDUINO main loop.
void loop() { device.loop(); }
When the dongle wants to know whether the air conditioner turned on or off, the dongle sends message (METHOD_GET / AirConditioner::RESOURCE_POWER ). When the device receives it, the registered callback CH_GET_POWER will be invoked. It replies the request with the current POWER value.
In the full example code, you can easily come to know that the device simulates the real air conditioner. When power is on, the temperature will go down to the preferred temperature.
You can connect ARDUINO to your PC and debug it using Serial program. In this case, Serial program can be the dongle. You can ask whether the power is on or off, by sending 0x22. The returned 0xA200 from the device means that the power is off. You can request to turn on, by sending 0x4201. The returned 0xC200 means that there is no error to handle the request. You can see the message 0x4320 from the device (ARDUINO). It means the current temperature is 32(0x20) degrees. The temperature was going up to 40(0x28) degrees. After turning on the air conditioner (0x4201 sent), the temperature was going down.
In the very same manner, you can make the Internet connected dongle using ESP8266. This dongle is an example to make the AirConditioner (device) connected to Sugar Home which provides smart home framework over MQTT.
Find ESP8266 board library in ARDUINO board manager (as shown in the below snapshot) and install it. It enables you to develop ESP8266 software using ARDUINO IDE. esp8266.com/arduino gives more information.
Find MQTT(PubSub) library in ARDUINO library manager (as shown in the below snapshot) and install it.
Developing the dongle is almost same as developing the device.
-
Include "IoTDongle.h" and use namespace IoTD. And, include "ESP8266WiFi.h" and "PubSubClient.h" for MQTT on ESP8266.
#include <ESP8266WiFi.h> #include <PubSubClient.h> #include <IoTDongle.h> using namespace IoTD;
-
Declare the device that you want to develop.
AirConditioner device;
-
Set callbacks and begin the device in setup(). And make connected to MQTT broker(iot.eclipse.org:1883) through Wi-Fi. To do that, you should specify SSID and password for your own Wi-Fi AP. If you want to use another MQTT broker, you can change it.
const char* ssid = "<your own ssid for wi-fi "; const char* password = "<your own password for wi-fi>"; const char* host = "iot.eclipse.org"; const int port = 1883; const char* homeID = "sugar_home"; // it should be unique ID for home const char* deviceID = "air_conditioner"; // it should be unique ID for device WiFiClient wifi; PubSubClient mqtt(wifi); // mqtt client void CH_SET_POWER(Command cmd, Method api, Resource res) { READ_SERIAL_N_PUB_POWER; Device::sendCommand(COMMAND_RETURN, api, res); Device::sendData(ERROR_NONE); } void setup() { device.setCallback(COMMAND_CALL, METHOD_SET, AirConditioner::RESOURCE_POWER, CH_SET_POWER); device.begin(); WiFi.begin(ssid, password); while(WiFi.status() != WL_CONNECTED) delay(500); // wait for Wi-Fi connection mqtt.setServer(host, port); mqtt.setCallback(MQTT_CALLBACK); }
-
Connect MQTT server and call device loop in ARDUINO main loop.
void loop() { if (!mqtt.connected()){ if(mqtt.connect(deviceID)){ MQTT_SUB(AirConditioner::RESOURCE_POWER); MQTT_SUB(AirConditioner::RESOURCE_PREFERRED); } } if (mqtt.connected()) mqtt.loop(); device.loop(); }
-
Handle MQTT event - get (resource ID, value) from MQTT event and send it to device.
void MQTT_CALLBACK(char* topic, byte* payload, unsigned int length) { char* ptr = strtok(topic, "/"); if(strcmp(ptr, homeID)) return; ptr = strtok(NULL, "/"); if(strcmp(ptr, deviceID)) return; ptr = strtok(NULL, "/"); unsigned char res = (unsigned char)atoi(ptr); ptr = strtok(NULL, "/"); if(strcmp(ptr, "R")) return; Device::sendCommand(COMMAND_CALL, METHOD_SET, res); payload[length]=0; Device::sendData((unsigned char)atoi((char*)payload)); }
You can get the full code from the examples - examples / IoTDongle / SugarHome / AirConditioner.
Here is the sequence diagram from the mobile SugarHome app to the air conditioner (device).
When you press the button "power" and make it on in your Sugar Home app (or MQTT client), app publishes topic ("<Home ID>/<Device ID>/<Resource ID=2>/R") to MQTT broker. MQTT broker forwards the topic change to the dongle. The dongle gets the changed (RESOURCE ID, VALUE) and sends to the device.
In the other case, the device wants to notify the temperature change to the dongle. The device sends the message which sets the current temperature to 28 degrees. IoT dongle platform invokes CH_SET_TEMPERATURE() in the dongle. The dongle also publishes topic ("<Home ID>/<Device ID>/<Resource ID=3>") to MQTT broker. MQTT broker forwards the topic change to your Sugar Home app.
MQTT Dashboard is one good MQTT client at Mobile. You can use it also as shown in the below image.
This dongle is an example to make the AirConditioner (device) connected to Blynk which provides DIY mobile app to control the device.
Find Blynk library in ARDUINO library manager (as shown in the below snapshot) and install it.
-
Include "IoTDongle.h" and use namespace IoTD. And, include "ESP8266WiFi.h" and "BlynkSimpleEsp8266.h" for Blynk on ESP8266.
#include <ESP8266WiFi.h> #include <BlynkSimpleEsp8266.h> #include <IoTDongle.h> using namespace IoTD;
-
Declare the device that you want to develop.
AirConditioner device;
-
Set callbacks and begin the device in setup(). And connect Blynk server by Blynk.begin(). You should specify SSID and password for your own Wi-Fi AP. When you create Blynk project on your mobile (Blynk app), AUTH token is provided. You should use the same AUTH token to connect Blynk server.
const char ssid[] = "<your own ssid for wi-fi "; const char password[] = "<your own password for wi-fi>"; const char auth[] = "<your own auth for blynk>"; void CH_SET_POWER(Command cmd, Method api, Resource res) { READ_SERIAL_N_WRITE_BLYNK_POWER; Device::sendCommand(COMMAND_RETURN, api, res); Device::sendData(ERROR_NONE); } void setup() { device.setCallback(COMMAND_CALL, METHOD_SET, AirConditioner::RESOURCE_POWER, CH_SET_POWER); device.begin(); Blynk.begin(auth, ssid, password); }
-
Call device loop and Blynk loop(run()) in ARDUINO main loop.
void loop() { device.loop(); Blynk.run(); }
-
Handle Blynk event.
BLYNK_WRITE(BLYNK_PORT_POWER) { READ_BLYNK_N_WRITE_SERIAL_POWER; }
You can get the full code from the examples - examples / IoTDongle / Blynk / AirConditioner.
Here is the sequence diagram from the mobile Blynk app to the air conditioner (device).
When you press the button "power" and make it on, it invokes BLYNK_WRITE event. The dongle gets the event and sends "set power on" message - (CALL, METHOD_SET, AirConditioner::RESOURCE_POWER, ON) - through IoT dongle platform. IoT dongle platform invokes CH_SET_POWER() in the device which handles the request from the dongle. And it will answer with NO_ERROR to the dongle. Iot dongle platform invokes RH_SET_POWER() in the dongle.
In the other case, the device wants to notify the temperature change to the dongle. The device sends the message which sets the current temperature to 28 degrees. IoT dongle platform invokes CH_SET_TEMPERATURE() in the dongle. It forwards the temperature to Blynk and makes UI widget on your mobile update the current temperature (28 degrees). And the dongle replies with NO_ERROR. IoT dongle platform invokes RH_SET_TEMPERATURE() in the device.
Blynk uses virtual port to connect between GUI widget and the value. In this example, "POWER" button widget is connected to V1 port, "TEMPERATURE" text widget is connected to V2 port and "PREFERRED" slider widget is connected to V3 port. The below shows how to create your own UI and connect virtual ports on Blynk app.
Click the below image, you can watch DEMO video (youtube).