diff --git a/README.md b/README.md index 2002268..14906a1 100644 --- a/README.md +++ b/README.md @@ -148,6 +148,72 @@ pio device monitor -b 115200 ``` +### Firmware configuration commands + +HomeGenie Mini device support also commands via serial terminal. You can enter any API +command using the `/api/` prefix, or enter system commands prefixed by `#` character. + +#### System core commands + +``` +#CONFIG:device-name +#CONFIG:wifi-ssid +#CONFIG:wifi-password +#CONFIG:system-time ::. +#CONFIG:system-zone-id +#CONFIG:system-zone-offset +#VERSION +#RESET +``` + +#### Getting/Setting parameters + +``` +#GET: +#SET: +``` + +##### System configuration parameters list + +| Key | Description | Default | +|------------|----------------------------|------------------| +| `sys-rb-n` | Factory reset button GPIO# | -1 (-1=not used) | +| `sys-sl-n` | System status LED GPIO# | -1 (-1=not used) | +| `io-typ01` | I/O Ch.1 type | | +| `io-pin01` | I/O Ch.1 GPIO# | -1 (-1=not used) | +| `io-typ02` | I/O Ch.2 type | | +| `io-pin02` | I/O Ch.2 GPIO# | -1 (-1=not used) | +| `io-typ03` | I/O Ch.3 type | | +| `io-pin03` | I/O Ch.3 GPIO# | -1 (-1=not used) | +| `io-typ04` | I/O Ch.4 type | | +| `io-pin04` | I/O Ch.4 GPIO# | -1 (-1=not used) | +| `io-typ05` | I/O Ch.5 type | | +| `io-pin05` | I/O Ch.5 GPIO# | -1 (-1=not used) | +| `io-typ06` | I/O Ch.6 type | | +| `io-pin06` | I/O Ch.6 GPIO# | -1 (-1=not used) | +| `io-typ07` | I/O Ch.7 type | | +| `io-pin07` | I/O Ch.7 GPIO# | -1 (-1=not used) | +| `io-typ08` | I/O Ch.8 type | | +| `io-pin08` | I/O Ch.8 GPIO# | -1 (-1=not used) | + +*Example **setting** configuration from a terminal connected to the serial port of the device:* + +``` +#SET:sys-sl-n=10 +#SET:sys-rb-n=0 +#RESET +``` + +*Example **getting** configuration value:* + +``` +#GET:sys-sl-n +``` +response: +``` +#GET:sys-sl-n=10 +``` + ## Firmwares examples with source code diff --git a/examples/color-light/README.md b/examples/color-light/README.md index 99ce642..c3b2c61 100644 --- a/examples/color-light/README.md +++ b/examples/color-light/README.md @@ -5,7 +5,7 @@ A smart light device with addressable RGB LEDs. - [Documentation and firmware install page](https://homegenie.it/mini/1.2/examples/smart-led-lights/) -## Firmware configuration +## Firmware configuration (in addition to default system options) | Key | Description | Default | |------------|----------------------------|-----------------------------------------| diff --git a/platformio.ini b/platformio.ini index fda565d..d21e21f 100644 --- a/platformio.ini +++ b/platformio.ini @@ -43,8 +43,13 @@ lib_deps_8266 = platform = espressif32@6.7.0 build_flags = ${env.build_flags} +[env:default-c3] +platform = espressif32@6.7.0 +board = esp32-c3-devkitc-02 +#board = esp32-c3-devkitm-1 +build_flags = ${env.build_flags} -I examples -I src -D ESP32_C3 -D ARDUINO_USB_MODE=1 -D ARDUINO_USB_CDC_ON_BOOT=1 -[env:d1-mini] +[env:default-d1mini] platform = espressif8266@2.6.3 board = d1_mini build_flags = ${env.build_flags} -D DISABLE_UI -D DISABLE_AUTOMATION @@ -55,19 +60,11 @@ lib_ignore = ESP32_BleSerial LovyanGFX -[env:d1-mini-esp32] +[env:sonoff-dualr3] platform = espressif32@6.7.0 -build_flags = ${env.build_flags} -D MINI_ESP32 -D CONFIG_ServiceButtonPin=16 -D CONFIG_StatusLedPin=26 -D CONFIG_GPIO_OUT={18,19,23,5} - +build_flags = ${env.build_flags} -D DISABLE_UI -D CONFIG_AUTOMATION_SPAWN_FREERTOS_TASK + -D DEFAULT_CONFIG='{"sys_bt_pin":"0","sys_sl_pin":"13","io-typ01":"Switch","io-pin01":"14","io-typ02":"Switch","io-pin02":"27"}' -[env:switch-relay-x2-esp32] -platform = espressif32@6.7.0 -build_flags = ${env.build_flags} -D DISABLE_UI -D CONFIG_ServiceButtonPin=0 -D CONFIG_StatusLedPin=23 -D CONFIG_GPIO_OUT="{16,17}" - - -[env:sonoff] -platform = espressif32@6.7.0 -build_flags = ${env.build_flags} -D DISABLE_UI -D CONFIG_AUTOMATION_SPAWN_FREERTOS_TASK -D CONFIG_ServiceButtonPin=0 -D CONFIG_StatusLedPin=13 -D CONFIG_GPIO_OUT={14,27} -D CONFIG_GPIO_IN={32,33} #------------------[ Examples ]------------------ @@ -81,7 +78,8 @@ lib_deps = ${env.lib_deps} [env:smart-sensor-d1-mini-esp32] platform = espressif32@6.7.0 -build_flags = ${env.build_flags} -I examples -I src -D MINI_ESP32 -D DISABLE_UI -D CONFIG_ServiceButtonPin=16 -D CONFIG_StatusLedPin=26 -D CONFIG_GPIO_OUT={18,19,23,5} +build_flags = ${env.build_flags} -I examples -I src -D MINI_ESP32 -D DISABLE_UI + -D DEFAULT_CONFIG='{"sys_bt_pin":"16","sys_sl_pin":"26","io-typ01":"Dimmer","io-pin01":"18","io-typ02":"Dimmer","io-pin02":"19","io-typ03":"Dimmer","io-pin03":"23","io-typ04":"Dimmer","io-pin04":"5"}' build_src_filter = + - + lib_deps = ${env.lib_deps} OneWire@2.3.8 @@ -110,7 +108,8 @@ lib_deps = ${env.lib_deps} [env:smart-sensor-display-s3] platform = espressif32@6.7.0 board = esp32-s3-devkitc-1 -build_flags = ${env.build_flags} -I examples -I src -D ESP32_S3 -D CONFIG_AUTOMATION_SPAWN_FREERTOS_TASK -D CONFIG_ENABLE_POWER_MANAGER -DESP32S3_DEV -DBOARD_HAS_PSRAM -D CONFIG_GPIO_OUT={15,17,18} +build_flags = ${env.build_flags} -I examples -I src -D ESP32_S3 -D CONFIG_AUTOMATION_SPAWN_FREERTOS_TASK -D CONFIG_ENABLE_POWER_MANAGER -DESP32S3_DEV -DBOARD_HAS_PSRAM + -D DEFAULT_CONFIG='{"sys_bt_pin":"0","sys_sl_pin":"-1","io-typ01":"Dimmer","io-pin01":"15","io-typ02":"Dimmer","io-pin02":"17","io-typ03":"Dimmer","io-pin03":"18"}' build_src_filter = + - + board_build.mcu = esp32s3 board_build.f_cpu = 240000000L @@ -226,7 +225,7 @@ lib_deps = ${env.lib_deps} platform = espressif32@6.7.0 board = esp32-c3-devkitc-02 #board = esp32-c3-devkitm-1 -build_flags = ${env.build_flags} -I examples -I src -D ESP32_C3 -D DISABLE_UI -D CONFIG_StatusLedPin=-1 -D ARDUINO_USB_MODE=1 -D ARDUINO_USB_CDC_ON_BOOT=1 +build_flags = ${env.build_flags} -I examples -I src -D ESP32_C3 -D DISABLE_UI -D ARDUINO_USB_MODE=1 -D ARDUINO_USB_CDC_ON_BOOT=1 build_src_filter = + - + + lib_deps = ${env.lib_deps} https://github.com/adafruit/Adafruit_NeoPixel#1.12.0 @@ -244,7 +243,7 @@ lib_deps = ${env.lib_deps} platform = espressif32@6.7.0 board = esp32-c3-devkitc-02 #board = esp32-c3-devkitm-1 -build_flags = ${env.build_flags} -I examples -I src -D ESP32_C3 -D DISABLE_UI -D CONFIG_StatusLedPin=-1 -D ARDUINO_USB_MODE=1 -D ARDUINO_USB_CDC_ON_BOOT=1 +build_flags = ${env.build_flags} -I examples -I src -D ESP32_C3 -D DISABLE_UI -D ARDUINO_USB_MODE=1 -D ARDUINO_USB_CDC_ON_BOOT=1 build_src_filter = + - + lib_deps = ${env.lib_deps} arduino-libraries/Stepper@^1.1.3 diff --git a/src/Config.cpp b/src/Config.cpp index 62a003b..fb5811d 100644 --- a/src/Config.cpp +++ b/src/Config.cpp @@ -33,6 +33,9 @@ ZoneConfig Config::zone; SystemConfig Config::system; +short Config::ServiceButtonPin = -1; +short Config::StatusLedPin = -1; + bool Config::isStatusLedOn = false; void Config::statusLedOn() { @@ -104,10 +107,6 @@ void Config::handleConfigCommand(String &message) { } else if (message.equals("#RESET")) { delay(50); -#ifndef ESP8266 - esp_restart(); -#else - // TODO: not supported by ESP8266 -#endif + ESP.restart(); } } diff --git a/src/Config.h b/src/Config.h index c22546c..46e9dd0 100644 --- a/src/Config.h +++ b/src/Config.h @@ -33,6 +33,8 @@ #include "defs.h" #include +#include + #ifdef CONFIG_AUTOMATION_SPAWN_FREERTOS_TASK #include #endif @@ -94,9 +96,9 @@ class SystemConfig { class Config { public: - const static short ServiceButtonPin = CONFIG_ServiceButtonPin; - const static short StatusLedPin = CONFIG_StatusLedPin; - const static uint16_t ConfigureButtonPushInterval = 2500; + static short ServiceButtonPin; + static short StatusLedPin; + const static uint16_t ConfigureButtonPushInterval = 5000; static ZoneConfig zone; static SystemConfig system; #ifdef ESP32 @@ -135,7 +137,7 @@ class Config { preferences.begin(CONFIG_SYSTEM_NAME, false); preferences.putString(k.c_str(), value); preferences.end(); - return false; + return false; // TODO: ?!? } static String getSetting(const char* key, const char* defaultValue = "") { @@ -146,6 +148,24 @@ class Config { value = preferences.getString(k.c_str(), defaultValue); } preferences.end(); + // TODO: should use "pref.isKey(...)" for proper checking + if (value == "") { + // Lookup config factory defaults only if the value does not exists + String config = STRING_VALUE(DEFAULT_CONFIG); + if (!config.isEmpty()) { + JsonDocument doc; + DeserializationError error = deserializeJson(doc, config); + if (error.code() == 0) { + auto params = doc.as(); + for (auto p: params) { + if (p.key() == key) { + value = p.value().as(); + break; + } + } + } + } + } return value; } @@ -157,7 +177,10 @@ class Config { static void statusLedCallback(std::function new_func); static void init() { - // Setup status led + // Setup status LED and factory reset buttons + ServiceButtonPin = getSetting("sys-rb-n", "-1").toInt(); + StatusLedPin = getSetting("sys-sl-n", "-1").toInt(); + // Setup status LED if (StatusLedPin >= 0) pinMode(StatusLedPin, OUTPUT); Preferences preferences; diff --git a/src/HomeGenie.cpp b/src/HomeGenie.cpp index da5058d..d7c6488 100644 --- a/src/HomeGenie.cpp +++ b/src/HomeGenie.cpp @@ -39,8 +39,10 @@ namespace Service { Config::init(); // Setup button - pinMode(Config::ServiceButtonPin, INPUT_PULLUP); -// attachInterrupt(digitalPinToInterrupt(Config::ServiceButtonPin), buttonChange, CHANGE); + if (Config::ServiceButtonPin != -1) { + pinMode(Config::ServiceButtonPin, INPUT_PULLUP); +// attachInterrupt(digitalPinToInterrupt(Config::ServiceButtonPin), buttonChange, CHANGE); + } // Logger initialization Logger::begin(LOG_LEVEL_TRACE); diff --git a/src/HomeGenie.h b/src/HomeGenie.h index 1de2ec1..fbc6cc4 100644 --- a/src/HomeGenie.h +++ b/src/HomeGenie.h @@ -158,6 +158,9 @@ namespace Service { } } static void checkServiceButton() { + if (Config::ServiceButtonPin < 0) { + return; + } buttonChange(); int64_t elapsed = 0; if (getInstance()->buttonPressed) { diff --git a/src/automation/Scheduler.h b/src/automation/Scheduler.h index 1a85cfb..55f6ba2 100644 --- a/src/automation/Scheduler.h +++ b/src/automation/Scheduler.h @@ -30,7 +30,6 @@ #ifndef HOMEGENIE_MINI_SCHEDULER_H #define HOMEGENIE_MINI_SCHEDULER_H -#include #include #include diff --git a/src/defs.h b/src/defs.h index 2e8ec89..1569dca 100644 --- a/src/defs.h +++ b/src/defs.h @@ -68,31 +68,14 @@ #define DISABLE_BLUETOOTH_CLASSIC #define CONFIGURE_WITH_WPS #define WebServer ESP8266WebServer - #ifndef CONFIG_GPIO_OUT - #define CONFIG_GPIO_OUT {14,12,13,15} - #endif #endif #ifdef ESP32_C3 #undef DISABLE_BLUETOOTH_LE #define DISABLE_BLUETOOTH_CLASSIC - #define CONFIG_ServiceButtonPin 4 - #if CONFIG_StatusLedPin != -1 - #define CONFIG_StatusLedPin 0 - #endif - #ifndef CONFIG_GPIO_OUT - #define CONFIG_GPIO_OUT {6} - #endif #elif ESP32_S3 #undef DISABLE_BLUETOOTH_LE #define DISABLE_BLUETOOTH_CLASSIC - #define CONFIG_ServiceButtonPin 21 - #if CONFIG_StatusLedPin != -1 - #define CONFIG_StatusLedPin 33 - #endif - #ifndef CONFIG_GPIO_OUT - #define CONFIG_GPIO_OUT {15,16,17,18} - #endif #else #define DISABLE_BLUETOOTH_LE // #define DISABLE_BLUETOOTH_CLASSIC @@ -101,19 +84,6 @@ // #undef DISABLE_UI #endif -#ifndef CONFIG_ServiceButtonPin - #define CONFIG_ServiceButtonPin 2 -#endif -#ifndef CONFIG_StatusLedPin - #define CONFIG_StatusLedPin 16 -#endif -#ifndef CONFIG_GPIO_OUT - #define CONFIG_GPIO_OUT {14,12,13,17} -#endif -#ifndef CONFIG_GPIO_IN - #define CONFIG_GPIO_IN { /* not implemented */ } -#endif - // -------------------------------- diff --git a/src/net/MQTTServer.h b/src/net/MQTTServer.h index b51f1e8..34ce875 100644 --- a/src/net/MQTTServer.h +++ b/src/net/MQTTServer.h @@ -30,8 +30,6 @@ #ifndef HOMEGENIE_MINI_MQTTSERVER_H #define HOMEGENIE_MINI_MQTTSERVER_H -#include - #include "Task.h" #include "net/mqtt/MQTTBrokerMini.h" diff --git a/src/service/api/HomeGenieHandler.cpp b/src/service/api/HomeGenieHandler.cpp index 4aebbd3..03f561a 100644 --- a/src/service/api/HomeGenieHandler.cpp +++ b/src/service/api/HomeGenieHandler.cpp @@ -45,30 +45,34 @@ namespace Service { namespace API { } void HomeGenieHandler::init() { - // Add GPIO modules - // Output GPIO pins - uint8_t gpio[] = CONFIG_GPIO_OUT; - uint8_t gpio_count = *(&gpio + 1) - gpio; - for (int m = 0; m < gpio_count; m++) { - auto address = String(gpio[m]); - auto module = new Module(); - module->domain = IO::IOEventDomains::HomeAutomation_HomeGenie; - module->address = address; - module->type = "Dimmer"; - module->name = "GPIO " + address; - module->description = ""; - - // init pins with current level - float level = GPIOPort::loadLevel(gpio[m]); - level > 0 ? gpioPort->on(gpio[m]) : gpioPort->off(gpio[m]); - auto propLevel = new ModuleParameter(IOEventPaths::Status_Level, String(level)); - module->properties.add(propLevel); - - moduleList.add(module); + // Add builtin GPIO modules + for (int m = 0; m < 8; m++) { + auto in = String("io-typ0"); + in.concat(m + 1); + auto moduleType = Config::getSetting(in.c_str(), "Dimmer"); + in.replace("-typ0", "-pin0"); + int gpioNum = Config::getSetting(in.c_str(), "-1").toInt(); + if (gpioNum >= 0 && moduleType != "Sensor") { // TODO: enable INPUT channels (Sensor) + auto address = String(gpioNum); + auto module = new Module(); + module->domain = IO::IOEventDomains::HomeAutomation_HomeGenie; + module->address = address; + module->type = moduleType; + module->name = "GPIO " + address; + module->description = ""; + auto propLevel = new ModuleParameter(IOEventPaths::Status_Level, ""); + module->properties.add(propLevel); + if (moduleType == "Sensor") { +// TODO: ............ + } else { + // Recall last stored level (switchable modules only) + float level = GPIOPort::loadLevel(gpioNum); + level > 0 ? gpioPort->on(gpioNum) : gpioPort->off(gpioNum); + propLevel->value = String(level); + } + moduleList.add(module); + } } - - // TODO: implement Input GPIO pins as well (CONFIG_GPIO_IN) - } bool HomeGenieHandler::canHandleDomain(String* domain) { @@ -417,41 +421,35 @@ namespace Service { namespace API { return false; } } else { - - uint8_t gpio[] = CONFIG_GPIO_OUT; uint8_t pinNumber = request->Address.toInt(); - bool validPin = std::find(std::begin(gpio), std::end(gpio), pinNumber) != std::end(gpio); - if (validPin) { - - Module* module = getModule(request->Domain.c_str(), request->Address.c_str()); - if (module != nullptr) { + Module* module = getModule(request->Domain.c_str(), request->Address.c_str()); + if (module != nullptr) { - auto levelProperty = module->getProperty(IOEventPaths::Status_Level); - if (levelProperty->value.toFloat() > 0) { - GPIOPort::saveLastOnLevel(pinNumber, levelProperty->value.toFloat()); - } + auto levelProperty = module->getProperty(IOEventPaths::Status_Level); + if (levelProperty->value.toFloat() > 0) { + GPIOPort::saveLastOnLevel(pinNumber, levelProperty->value.toFloat()); + } - float level = 0; - if (request->Command == ControlApi::Control_On) { + float level = 0; + if (request->Command == ControlApi::Control_On) { + level = gpioPort->on(pinNumber); + } else if (request->Command == ControlApi::Control_Off) { + level = gpioPort->off(pinNumber); + } else if (request->Command == ControlApi::Control_Level) { + level = gpioPort->level(pinNumber, (uint8_t)request->getOption(0).toFloat()); + } else if (request->Command == ControlApi::Control_Toggle) { + if (levelProperty->value.toFloat() == 0) { level = gpioPort->on(pinNumber); - } else if (request->Command == ControlApi::Control_Off) { + } else { level = gpioPort->off(pinNumber); - } else if (request->Command == ControlApi::Control_Level) { - level = gpioPort->level(pinNumber, (uint8_t)request->getOption(0).toFloat()); - } else if (request->Command == ControlApi::Control_Toggle) { - if (levelProperty->value.toFloat() == 0) { - level = gpioPort->on(pinNumber); - } else { - level = gpioPort->off(pinNumber); - } } + } - levelProperty->setValue(String(level).c_str()); - GPIOPort::saveLevel(pinNumber, levelProperty->value.toFloat()); + levelProperty->setValue(String(level).c_str()); + GPIOPort::saveLevel(pinNumber, levelProperty->value.toFloat()); - responseCallback->writeAll(ApiHandlerResponseText::OK); - return true; - } + responseCallback->writeAll(ApiHandlerResponseText::OK); + return true; } } return false; diff --git a/src/version.h b/src/version.h index bf80364..a64c897 100644 --- a/src/version.h +++ b/src/version.h @@ -29,7 +29,7 @@ #define VERSION_MAJOR 1 #define VERSION_MINOR 2 -#define VERSION_PATCH 22 +#define VERSION_PATCH 23 #define STRING_VALUE(...) STRING_VALUE__(__VA_ARGS__) #define STRING_VALUE__(...) #__VA_ARGS__