Перейти к содержимому

Термостат на ESP32 с удаленным управлением. Часть 9. Термостат и управление нагрузкой

Добрый день, уважаемый читатель!

Продолжаю описание самодельного устройства на базе ESP32 и ESP-IDF с возможностью дистанционного контроля и управления через WiFi. Возможно, вы наткнулись на эту статью случайно, поэтому приведу список предыдущих статей.

В этой части я расскажу, как легко и просто можно добавить в прошивку управление двумя встроенными реле, а следовательно и нагрузкой. Например сделаем простейший термостат, как это изначально и планировалось. Ну а потом можно будет прикрутить расписания и несколько уровней и прочая, прочая, прочая… Ну а вы сами можете изменить код так, чтобы управлять, скажем, вентилятором в ванной комнате по датчику влажности.


Управление GPIO в ESP-IDF

На самом деле в управлении нагрузкой на ESP32 и ESP-IDF нет вообще ничего сложно и страшного. Не сложнее, чем в Arduino.

  • gpio_set_direction(pin, GPIO_MODE_OUTPUT) – настроить вывод pin в режим вывода
  • gpio_set_level(pin, level) – записать в этот самый вывод pin уровень level

В принципе, этого уже вполне достаточно, чтобы организовать управление реле, которые смонтированы на плате, по своему усмотрению.

Ну дальше, как всегда, появляются дополнительные идеи и хотелки:

  • Нужно включить нагрузку только на определенное время? Ничего сложного – только добавьте воды таймер.
  • Нужно включать и выключать нагрузку в импульсном режиме – добавьте ещё один таймер.
  • Хочется фиксировать время включения и выключения нагрузки, а также длительность работы – опять же предельно просто – несколько переменных и готово
  • Публиковать всё это на mqtt-брокере в виде JSON? И с этим можно легко справится.

Но всё это повторяется из проекта в проект, и чтобы не переносить один и тот же код копипастой, я написал для этого специальную библиотечку. Как всегда, она довольно тесно интегрирована с другими моими библиотеками для ESP-IDF, и идеально подходит для проекта термостата.

Встречайте – reLoadCtrl: GitHub – kotyara12/reLoadCtrl


 

reLoadCtrl – класс для управления нагрузкой

Данная библиотека, как и многие другие, реализована в виде класса, что позволяет легко использовать одновременно несколько отдельных экземпляров класса в рамках одного проекта. Кроме того, ООП позволяет изящно организовать разные реализации либо для работы со встроенными GPIO микроконтроллера, либо для различных расширителей GPIO.

Основные возможности:

  • Два варианта реализации: rLoadGpioController для встроенных GPIO и rLoadIoExpController для расширителей IO.
  • Управление нагрузкой по низкому или высокому уровню на выходе – может работать с любыми схемами и готовыми модулями
  • Инициализация для встроенных GPIO – вам не нужно заботится об этом
  • Хранение текущего состояния нагрузки
  • Фиксация времени включения и выключения нагрузки, вычисление длительности работы за последний цикл, за сутки, неделю, месяц, учетный период (например с 25 числа месяца по 25 числа следующего месяца), год.
  • Включение нагрузки на определенное время по таймеру. Всю работу с таймерами библиотека берет на себя
  • Управление нагрузкой в циклическом режиме в пределах общей длительности. Например: включаем насос на 30 секунд, затем 60 секунд ждем, затем цикл повторяется. И так 30 минут.
  • Формирование JSON для публикации на MQTT с данными о состоянии нагрузки и всеми счетчиками и интервалами.

Создание экземпляра класса

Для того, чтобы начать жизнь заново и использовать эту библиотеку, создайте глобальный или статический экземпляр класса с помощью функции:

rLoadController(
  uint8_t pin,                       // номер вывода GPIO ESP32 или вывода IO EXT
  uint8_t level_on,                  // уровень, которым включается нагрузка
  bool use_timer,                    // использовать постоянный таймер для управления нагрузкой в циклическом режиме
  const char* nvs_space,             // ключ для сохранения данных счетчиков в NVS разделе
  uint32_t* cycle_duration,          // указатель на переменную, в которой хранится длительность включения нагрузки в циклическом режиме
  uint32_t* cycle_interval,          // указатель на переменную, в которой хранится длительность паузы между включениями нагрузки в циклическом режиме
  timeintv_t cycle_type,             // размерность цикла
  cb_load_change_t cb_gpio_before,   // функция обратного вызова, которая будет вызвана перед любым изменением состояния выхода 
  cb_load_change_t cb_gpio_after,    // функция обратного вызова, которая будет вызвана после любого изменения состояния выхода
  cb_load_change_t cb_state_changed, // функция обратного вызова, которая будет вызвана сразу после успешного изменения состояния вывода
  cb_load_publish_t cb_mqtt_publish  // функция обратного вызова для публикации пакета JSON на mqtt брокере
);

где:

  • uint8_t pin – номер вывода GPIO ESP32 или вывода IO EXT, который используется для управления нагрузкой
  • bool level_on – уровень, которым включается нагрузка. 1 – если установить высокий уровень для включения нагрузки, 0 – если нужно установить низкий уровень для включения нагрузки
  • bool use_timer – использовать постоянный таймер для управления нагрузкой. Если true, то таймер будет создан при инициализации GPIO и он останется “жить” до удаления экземпляра класса. Если false, то таймер будет создан при вызове loadSetTimer() и после использования удален. Это чуть-чуть экономит кучу, если таймер используется нечасто.
  • const char* nvs_space – ключ для сохранения данных счетчиков в NVS разделе. Должен быть не длиннее 10 символов!
  • uint32_t* cycle_duration – указатель на переменную, в которой хранится длительность включения нагрузки в циклическом режиме. Это именно указатель на переменную, а не само значение – сделано это для того, чтобы можно было менять параметр во время работы программы
  • uint32_t* cycle_interval – указатель на переменную, в которой хранится длительность паузы между включениями нагрузки в циклическом режиме.
  • timeintv_t cycle_type – размерность цикла: миллисекунды, секунды, минуты, часы
  • cb_load_change_t cb_gpio_before – функция обратного вызова, которая будет вызвана перед любым изменением состояния выхода. Например можно отключить прерывания по входам перед коммутацией мощной нагрузки
  • cb_load_change_t cb_gpio_after – функция обратного вызова, которая будет вызвана после любого изменения состояния выхода.
  • cb_load_change_t cb_state_changed – функция обратного вызова, которая будет вызвана сразу после успешного изменения состояния вывода. Например можно отправить уведомление в telegram: “котёл включен
  • cb_load_publish_t cb_mqtt_publish – функция обратного вызова для публикации пакета JSON на mqtt брокере. В принципе, можно было бы обойтись и без этого, но тогда бы пришлось инклудить в библиотеку mqtt-клиент, а это не очень удобно.

Затем, в любом удобном месте программы, нужно инициализировать вывод с помощью функцию loadInit(init_value), в котором как раз и будет выполнена настройка вывода и таймеров.

 

Простое управление нагрузкой

Для переключения состояния реле используйте функцию loadSetState:

loadSetState(bool new_state, bool forced, bool publish)

где:

  • bool new_state – новое состояние нагрузки
  • bool forced – записать состояние в выход, даже если текущее состояние нагрузки соответствует новому. В противном случае изменение произойдет, только если текущее состояние отличается от нового.
  • bool publish – выполнить публикацию состояния на MQTT после изменения состояния

При этом если при создании экземпляра класса заданы cycle_duration и cycle_interval, и они больше 0, то нагрузка будет включаться в периодическом режиме.

 

Включение нагрузки на определенный период

Если вам нужно включить нагрузку только на определенное время, используйте функцию loadSetTimer():

bool loadSetTimer(uint32_t duration_ms);

Эта функция принимает единственный параметр – длительность интервала включения нагрузки.

Для принудительного отключения нагрузки до истечения заданного времени таймера используйте описанную выше функцию loadSetState(false, …) – работа таймера будет прервана досрочно и нагрузка выключена.


 

Термостат

Теперь давайте используем этот класс для управления котлом по температуре в комнате. То есть создадим самый обычный комнатный термостат, к созданию которого мы так долго шли.

Предупреждение! В прошлой статье я рассказывал, как изменить код прошивки для того, чтобы можно было подключить другие сенсоры. В этой версии я вернул все сенсоры в прошивке на предыдущий вариант – DHT22 + BME280.

 

1. Добавляем параметры

В первую очередь нам нужно добавить рабочие параметры: температуру и гистерезис. Параметр – это переменная, которую можно изменять “извне” во время работы прошивки, например через mqtt-сервер. Сделать это можно двумя способами:

  • в простом случае мы просто задаем две температуры – температуру включения и температуру выключения, например 22°С и 24°С. Это достаточно просто, но редко применяется на практике – в “заводских” термостатах этот метод не используется, и далее вы поймете почему
  • более популярный вариант – целевая температура плюс гистерезис. Например 23°С и 1°С. Это гораздо удобнее для восприятия простыми пользователями, так как для установки новой температуры в помещении нужно изменить только один единственный параметртемпературу, а гистерезис остается постоянным и вовсе обычно “спрятан” внутри системных настроек.

Итак, с параметрами определились. Их можно определить как в sensors.h, так и в sensors.cpp – в данном случае без разницы. Я предпочитаю первый вариант – отделяю “мух от котлет” (хотя может быть идеологически это и не совсем правильно).

// Параметры регулирования температуры в доме
static float thermostatInternalTemp = 22.0;
static float thermostatInternalHyst = 1.0;

Ок, но ведь эти параметры желательно бы как-то изменять в процессе работы устройства. А это значит, их нужно где-то хранить в flash-памяти устройства и иметь возможность изменить через mqtt в любой момент.

За эти возможности в моей прошивке отвечают библиотеки reNvs и reParams. Подробней про них я еще расскажу отдельно, а пока просто воспользуемся их возможностями “как есть”.

Найдите в файле sensors.cpp функцию void sensorsInitParameters(). В ней мы вначале создадим группу параметров – pgThermostat, а “внутри” неё уже будем добавлять нужные параметры. Создание группы – не обязательное действие, но создает некий порядок в топиках.

Для создания группы и параметров вначале потребуется добавить в код константы:

#define CONTROL_THERMOSTAT_GROUP_KEY              "ths"
#define CONTROL_THERMOSTAT_GROUP_TOPIC            "thermostat"
#define CONTROL_THERMOSTAT_GROUP_FRIENDLY         "Термостат"

#define CONTROL_THERMOSTAT_PARAM_TEMP_KEY         "temperature"
#define CONTROL_THERMOSTAT_PARAM_TEMP_FRIENDLY    "Температура"
#define CONTROL_THERMOSTAT_PARAM_HYST_KEY         "hysteresis"
#define CONTROL_THERMOSTAT_PARAM_HYST_FRIENDLY    "Гистерезис"

#define CONTROL_THERMOSTAT_LOCAL                  false
#define CONTROL_THERMOSTAT_QOS                    1
#define CONTROL_THERMOSTAT_RETAINED               1

Регистрируем группу параметров, а после этого и сами параметры:

paramsGroupHandle_t pgTempMonitor;

static void sensorsInitParameters()
{
  // Группы параметров
  ....
  paramsGroupHandle_t pgThermostat = paramsRegisterGroup(nullptr, 
    CONTROL_THERMOSTAT_GROUP_KEY, CONTROL_THERMOSTAT_GROUP_TOPIC, CONTROL_THERMOSTAT_GROUP_FRIENDLY);
  ....
  ....
  // Параметры термостата
  if (pgThermostat) {
    paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_FLOAT, nullptr, pgThermostat,
      CONTROL_THERMOSTAT_PARAM_TEMP_KEY, CONTROL_THERMOSTAT_PARAM_TEMP_FRIENDLY,
      CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatInternalTemp);
    paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_FLOAT, nullptr, pgThermostat,
      CONTROL_THERMOSTAT_PARAM_HYST_KEY, CONTROL_THERMOSTAT_PARAM_HYST_FRIENDLY,
      CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatInternalHyst);
  };
};

 

Хотя, вы можете вполне обойтись и без всех этих #define. Но я так не люблю – потом замучаешься лазить по коду, чтобы что-то исправить. А так все настройки в одном месте, и их очень легко найти и изменить.

Всё готово! Если теперь скомпилировать и запустить проект, то мы увидим два новых топика в секции location/device/confirm. Но отправлять новые данные нужно в location/device/config! Об этом я писал в статье “Топики”…

 

2. Добавляем контроллер управления реле / котлом

Теперь можно добавить сам объект-контроллер. Для начала не забываем добавить библиотеку в список:

#include “reLoadCtrl.h”

Затем объявляем собственно сам экземпляр. В библиотеке объявлены три класса – один виртуальный и два обычных. Поскольку мы используем реле, подключенное ко встроенному GPIO, мы должны использовать класс rLoadGpioController.

static rLoadGpioController lcBoiler(
    CONFIG_GPIO_RELAY_BOILER,           // GPIO, к которому подключено реле
    0x01,                               // Реле включается высоким уровнем на выходе
    false,                              // Таймер не нужен
    CONTROL_THERMOSTAT_BOILER_KEY,      // Ключ, по которому будет хранится статистика в NVS space
    nullptr, nullptr, TI_MILLISECONDS,  // Указатели на параметры периодического управления нагрузкой, не нужны
    nullptr, nullptr,                   // Указатели на callbacks, которые будут вызываны перед и после изменения состояния выхода
    boilerStateChange,                  // Функция обратного вызова, которая будет вызвана при изменении состояния нагрузки
    boilerMqttPublish                   // Функция обратного вызова, которая будет вызвана при необходимости отправить данные на MQTT брокер
  );

Я думаю, здесь все всем будет понятно, пока поподробнее остановимся на последних функциях обратного вызова. Выглядят они пока следующим образом:

void boilerStateChange(rLoadController *ctrl, bool state, time_t duration)
{
  // Не убивай меня Иван-царевич, я тебе еще пригожусь...
}

bool boilerMqttPublish(rLoadController *ctrl, char* topic, char* payload, bool free_topic, bool free_payload)
{
  return mqttPublish(topic, payload, CONTROL_THERMOSTAT_QOS, CONTROL_THERMOSTAT_RETAINED, free_topic, free_payload);
}

Следующим этапом нужно настроить GPIO выход и сам объект, делается это достаточно просто:

void sensorsInitRelays()
{
  #if defined(CONFIG_ELTARIFFS_ENABLED) && CONFIG_ELTARIFFS_ENABLED
  lcBoiler.setPeriodStartDay(elTariffsGetReportDayAddress());
  #endif // CONFIG_ELTARIFFS_ENABLED
  lcBoiler.countersNvsRestore();
  lcBoiler.loadInit(false);
}
  • первая строка (не обязательная) передает в объект указатель на день месяца, с которого начинается очередной учетный период, например 25 число.
  • второй командой считываем ранее сохраненную статистику из NVS памяти
  • ну и последняя команда настраивает выход GPIO и все внутренние структуры и отключает нагрузку (false)

Далее, чтобы статистика включений и выключений нагрузки сохранялась в NVS памяти, нужно добавить соответствующую строчку в конце функции sensorsStoreData():

static void sensorsStoreData()
{
  rlog_i(logTAG, "Store sensors data");

  sensorOutdoor.nvsStoreExtremums(SENSOR_OUTDOOR_KEY);
  sensorIndoor.nvsStoreExtremums(SENSOR_INDOOR_KEY);
  sensorBoiler.nvsStoreExtremums(SENSOR_BOILER_KEY);

  tempMonitorIndoor.nvsStore(CONTROL_TEMP_INDOOR_KEY);
  tempMonitorBoiler.nvsStore(CONTROL_TEMP_BOILER_KEY);

  lcBoiler.countersNvsStore();
}

Все вместе это обеспечит сохранение статистики включений и выключений нагрузки в течение всего срока службы устройства. Если вам не нужно сохранение статистики, вы можете исключить эти команды. Но согласитесь – мелочь, а удобно.

Но сами по себе интервалы этот класс считать не умеет. Вместо этого встроенный в прошивку шедулер рассылает через систему событий временные метки – начало (00,000 секунд) каждой минуты, часа, дня, и т д. Нам требуется только подписаться на нужно событие и ждать. Добавляем строку в обработчик события sensorsTimeEventHandler():

static void sensorsTimeEventHandler(void* arg, esp_event_base_t event_base, int32_t event_id, void* event_data)
{
  if (event_id == RE_TIME_START_OF_DAY) {
    _sensorsNeedStore = true;
  };
  lcBoiler.countersTimeEventHandler(event_id, event_data);
}

И последнее приготовление… Прежде чем мы сможем что-либо публиковать на MQTT-брокере, мы должны сгенерировать топик. Топики в данной прошивке не есть что-то постоянное, они создаются в момент подключения к брокеру, хранятся в оперативной памяти и удаляются в момент отключения от него. Сделано это так потому, что MQTT-брокера может быть два (основной и резервный) и они могут иметь совершенно разные настройки. В соответствии с этим топики создаются по разным правилам.

static void sensorsMqttTopicsCreate(bool primary)
{
  sensorOutdoor.topicsCreate(primary);
  sensorIndoor.topicsCreate(primary);
  if (tempMonitorIndoor.mqttTopicCreate(primary, CONTROL_TEMP_LOCAL, CONTROL_TEMP_GROUP_TOPIC, CONTROL_TEMP_INDOOR_TOPIC, nullptr)) {
    rlog_i(logTAG, "Generated topic for indoor temperture control: [ %s ]", tempMonitorIndoor.mqttTopicGet());
  };
  sensorBoiler.topicsCreate(primary);
  if (tempMonitorBoiler.mqttTopicCreate(primary, CONTROL_TEMP_LOCAL, CONTROL_TEMP_GROUP_TOPIC, CONTROL_TEMP_BOILER_TOPIC, nullptr)) {
    rlog_i(logTAG, "Generated topic for boiler temperture control: [ %s ]", tempMonitorBoiler.mqttTopicGet());
  };
  lcBoiler.mqttTopicCreate(primary, CONTROL_THERMOSTAT_LOCAL, CONTROL_THERMOSTAT_BOILER_TOPIC, nullptr, nullptr);
}

static void sensorsMqttTopicsFree()
{
  sensorOutdoor.topicsFree();
  sensorIndoor.topicsFree();
  tempMonitorIndoor.mqttTopicFree();
  sensorBoiler.topicsFree();
  tempMonitorBoiler.mqttTopicFree();
  lcBoiler.mqttTopicFree();
  rlog_d(logTAG, "Topics for temperture control has been scrapped");
}

Почти все готово. Осталось не забыть добавить в главный цикл публикацию состояния и статистики. Впрочем – это не строго обязательно. Во время переключения нагрузки объект сам опубликует свое состояние. Но неплохо периодически “обновлять” данные на брокере вместе с обновление данных с сенсоров.

// MQTT брокер
if (esp_heap_free_check() && statesMqttIsConnected() && timerTimeout(&mqttPubTimer)) {
  timerSet(&mqttPubTimer, iMqttPubInterval*1000);
  sensorOutdoor.publishData(false);
  sensorIndoor.publishData(false);
  tempMonitorIndoor.mqttPublish();
  sensorBoiler.publishData(false);
  tempMonitorBoiler.mqttPublish();
  lcBoiler.mqttPublish();
};

Вот теперь все готово. Конечно, процедура подготовки к работе reLoadController выглядит немного сложновато, но взамен предоставляется достаточно высокий уровень сервиса.

 

3. Управление бойлером в зависимости от температуры

Теперь вы можете написать код для управления нагрузкой. Для включения котла достаточно написать команду:

Можно прикрутить “входящий” топик, с помощью которого вы будете дистанционно включать и выключать реле, а соответственно и котёл. Но этом слишком не интересно и не практично – а вдруг вы заболели и не можете управлять температурой в доме??? Автоматика должна работать незаметно и автономно, что бы не случилось.

У нас же есть датчики температуры и мы добавили параметры для управления температурой! Напишем несложную функцию для автоматического управления.

void sensorsBoilerControl()
{
  bool oldState, newState = false;
  
  // Получаем текущее состояние нагрузки
  oldState = 1cBoiler.getState();

  // Получаем текущую температуру с сенсора (она хранится во втором "хранилище") без физического чтения данных с шины
  float tempIndoor = NAN;
  if (sensorIndoor.getStatus() == SENSOR_STATUS_OK) {
    tempIndoor = sensorIndoor.getValue2(false).filteredvalue;
  };

  // Eсли удалось считать температуру, проверяем её в зависимости от текущего состояния нагрузки
  if (!isnan(tempIndoor)) {
    if (oldState) {
      // Cейчас котел включен. Выключить мы его должны, когда температура достигнет порогового + 1/2 гистерезиса
      newState = tempIndoor < (thermostatInternalTemp + 0.5 * thermostatInternalHyst);
    } else {
      // Cейчас котел выключен. Включить мы его должны, когда температура снизится до порогового - 1/2 гистерезиса
      newState = tempIndoor < (thermostatInternalTemp - 0.5 * thermostatInternalHyst) ;
    };
  };

  // Применяем новое состояние
  1сBoiler.loadSetState(newState, false, true);
}

Не забываем добавить вызов sensorsBoilerControl(); в основной цикл задачи после получения данных с сенсоров!

В результате наших манипуляций устройство будет само, без нашего вмешательства, поддерживать температуру около заданной, отклоняясь не более чем на поллитраградуса.

Цель достигнута. Но не будем почивать на лаврах, ведь нет предела совершенству…

 

4. Добавляем расписание

Большинство современных электронных термостатов умеют поддерживать заданную температуру в пределах заданного времени суток. Например днем, когда все члены семьи на работе, можно котел и не включать, а включить только к возвращению жителей домой. Давайте добавим простейшее расписание – в пределах времени суток. Например котел может работать с 15:00 часов одного дня до 08:00 часов следующего, в остальное время работа термостата должна быть заблокирована.

Добавляем параметры.

static timespan_t thermostatTimespan = 15000800U;

#define CONTROL_THERMOSTAT_PARAM_TIME_KEY         "timespan"
#define CONTROL_THERMOSTAT_PARAM_TIME_FRIENDLY    "Суточное расписание"

// Параметры термостата
if (pgThermostat) {
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_FLOAT, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_TEMP_KEY, CONTROL_THERMOSTAT_PARAM_TEMP_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatInternalTemp);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_FLOAT, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_HYST_KEY, CONTROL_THERMOSTAT_PARAM_HYST_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatInternalHyst);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_TIMESPAN, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_TIME_KEY, CONTROL_THERMOSTAT_PARAM_TIME_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatTimespan);

Теперь можно добавить проверку времени работы. Это не просто, а очень просто!

void sensorsBoilerControl()
{
  bool oldState, newState = false;

  // Проверяем расписание
  if (checkTimespanNowEx(thermostatTimespan, true)) {
    // Получаем текущее состояние нагрузки
    oldState = 1cBoiler.getState();
  
    // Получаем текущую температуру с сенсора (она хранится во втором "хранилище") без физического чтения данных с шины
    float tempIndoor = NAN;
    if (sensorIndoor.getStatus() == SENSOR_STATUS_OK) {
      tempIndoor = sensorIndoor.getValue2(false).filteredvalue;
    };
  
    // Eсли удалось считать температуру, проверяем её в зависимости от текущего состояния нагрузки
    if (!isnan(tempIndoor)) {
      if (oldState) {
        // Cейчас котел включен. Выключить мы его должны, когда температура достигнет порогового + 1/2 гистерезиса
        newState = tempIndoor < (thermostatInternalTemp + 0.5 * thermostatInternalHyst);
      } else {
        // Cейчас котел выключен. Включить мы его должны, когда температура снизится до порогового - 1/2 гистерезиса
        newState = tempIndoor < (thermostatInternalTemp - 0.5 * thermostatInternalHyst) ;
      };
    };
  } else {
    newState = false;
  };
  
  // Применяем новое состояние
  1сBoiler.loadSetState(newState, false, true);
}

Вот, собственно, и все суточное расписание в простейшем варианте. А теперь подумайте, как сделать два уровня температуры и недельное расписание, например будни + выходные.

 

5. Принудительная блокировка или включение котла

А теперь давайте подумаем, как отключить термостат на лето. Ведь при любом незначительном понижении температуры в комнате наш термостат включит котел. Конечно, можно выключить сам котёл, но все-таки лучше предусмотреть летний режим на самом термостате. Второй вопрос – иногда нужно, чтобы котёл работал по заданному выше суточному расписанию вне зависимости от температуры в комнате. То есть нужно предусмотреть принудительное включение или блокировку работы термостата.

Объявляем новый тип данных и новую переменную – параметр “режим работы“:

// Режимы работы термостата
typedef enum {
  THERMOSTAT_OFF = 0,       // Котел выключен всегда
  THERMOSTAT_ON,            // Котел включен всегда (без учета расписания и температуры)
  THERMOSTAT_TIME,          // Только управление по расписанию (без учета температуры)
  THERMOSTAT_TEMP,          // Только управление по температуре (без учета расписания)
  THERMOSTAT_TIME_AND_TEMP  // Управление по расписанию и температуре одновременно
} thermostat_mode_t;

// Параметры регулирования температуры в доме
static float thermostatInternalTemp = 22.0;
static float thermostatInternalHyst = 1.0;
static timespan_t thermostatTimespan = 15000800U;
static thermostat_mode_t thermostatMode = THERMOSTAT_TIME_AND_TEMP;

Регистрируем параметр:

// Параметры термостата
if (pgThermostat) {
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_U8, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_MODE_KEY, CONTROL_THERMOSTAT_PARAM_MODE_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatMode);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_FLOAT, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_TEMP_KEY, CONTROL_THERMOSTAT_PARAM_TEMP_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatInternalTemp);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_FLOAT, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_HYST_KEY, CONTROL_THERMOSTAT_PARAM_HYST_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatInternalHyst);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_TIMESPAN, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_TIME_KEY, CONTROL_THERMOSTAT_PARAM_TIME_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatTimespan);

Ранее написанную функцию проверки температуры в помещении я вынес в отдельную новую функцию, ниже будет понятно зачем:

bool sensorsBoilerTempCheck()
{
  // Получаем текущее состояние нагрузки
  bool oldState = lcBoiler.getState();

  // Получаем текущую температуру с сенсора (она хранится во втором "хранилище") без физического чтения данных с шины
  float tempIndoor = NAN;
  if (sensorIndoor.getStatus() == SENSOR_STATUS_OK) {
    tempIndoor = sensorIndoor.getValue2(false).filteredValue;
  };

  // Если удалось считать температуру, проверяем её в зависимости от текущего состояния нагрузки
  if (!isnan(tempIndoor)) {
    if (oldState) {
      // Сейчас котел включен. Выключить мы его должны, когда температура достигнет порогового + 1/2 гистерезиса
      return tempIndoor < (thermostatInternalTemp + 0.5 * thermostatInternalHyst);
    } else {
      // Сейчас котел выключен. Включить мы его должны, когда температура снизится до порогового - 1/2 гистерезиса
      return tempIndoor < (thermostatInternalTemp - 0.5 * thermostatInternalHyst);
    };
  };
  return false;
}

И затем полностью переделываем старую функцию управления:

void sensorsBoilerControl()
{
  bool newState;

  // Котел выключен всегда
  if (thermostatMode == THERMOSTAT_OFF) {
    newState = false;
  } 
  // Котел включен всегда (без учета расписания и температуры)
  else if (thermostatMode == THERMOSTAT_ON) {
    newState = true;
  } 
  // Только управление по расписанию (без учета температуры)
  else if (thermostatMode == THERMOSTAT_TIME) {
    newState = checkTimespanNowEx(thermostatTimespan, true);
  } 
  // Только управление по температуре (без учета расписания)
  else if (thermostatMode == THERMOSTAT_TEMP) {
    newState = sensorsBoilerTempCheck();
  } 
  // Управление по расписанию и температуре одновременно
  else if (thermostatMode == THERMOSTAT_TIME_AND_TEMP) {
    newState = checkTimespanNowEx(thermostatTimespan, true) && sensorsBoilerTempCheck();
  } 
  // Защита от ошибки программиста (а вдруг вы добавили еще режим и забыли написать обработчик?)
  else {
    newState = false;
  };

  // Применяем новое состояние
  lcBoiler.loadSetState(newState, false, true);
}

Вот теперь более-менее всё хорошо. В любой момент можно включить ручной режим со смартфона и вернуть в автоматический. Можно прошивать, упаковывать в коробку и устанавливать на место.

 

6. Добавляем уведомления о включении и выключении нагрузки

А что если вам захочется получать уведомления о включении и выключении котла, примерно так же как это происходит при выходе температуры за пределы? Что ж, это тоже достаточно легко организовать….

Чтобы уведомления можно было отключить в любой момент (если они надоели), добавим ещё одну переменную:

static bool thermostatNotify = true;

#define CONTROL_THERMOSTAT_PARAM_NOTIFY_KEY       "notifications"
#define CONTROL_THERMOSTAT_PARAM_NOTIFY_FRIENDLY  "Уведомления"

Данный параметр потребуется также зарегистрировать:

// Параметры термостата
if (pgThermostat) {
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_U8, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_MODE_KEY, CONTROL_THERMOSTAT_PARAM_MODE_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatMode);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_FLOAT, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_TEMP_KEY, CONTROL_THERMOSTAT_PARAM_TEMP_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatInternalTemp);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_FLOAT, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_HYST_KEY, CONTROL_THERMOSTAT_PARAM_HYST_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatInternalHyst);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_TIMESPAN, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_TIME_KEY, CONTROL_THERMOSTAT_PARAM_TIME_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatTimespan);
  paramsRegisterValue(OPT_KIND_PARAMETER, OPT_TYPE_U8, nullptr, pgThermostat,
    CONTROL_THERMOSTAT_PARAM_NOTIFY_KEY, CONTROL_THERMOSTAT_PARAM_NOTIFY_FRIENDLY,
    CONFIG_MQTT_PARAMS_QOS, (void*)&thermostatNotify);
};

Помните функцию boilerStateChange(), которую мы создали выше? Вот теперь пришла пора её написать:

void boilerStateChange(rLoadController *ctrl, bool state, time_t duration)
{
  if (thermostatNotify) {
    if (state) {
      tgSend(CONTROL_THERMOSTAT_NOTIFY_KIND, CONTROL_THERMOSTAT_NOTIFY_PRIORITY, CONTROL_THERMOSTAT_NOTIFY_ALARM, CONFIG_TELEGRAM_DEVICE, 
        CONTROL_THERMOSTAT_NOTIFY_ON);
    } else {
      tgSend(CONTROL_THERMOSTAT_NOTIFY_KIND, CONTROL_THERMOSTAT_NOTIFY_PRIORITY, CONTROL_THERMOSTAT_NOTIFY_ALARM, CONFIG_TELEGRAM_DEVICE, 
        CONTROL_THERMOSTAT_NOTIFY_OFF);
    };
  };  
}

где:

#define CONTROL_THERMOSTAT_NOTIFY_KIND            MK_MAIN
#define CONTROL_THERMOSTAT_NOTIFY_PRIORITY        MP_ORDINARY
#define CONTROL_THERMOSTAT_NOTIFY_ALARM           1
#define CONTROL_THERMOSTAT_NOTIFY_ON              "🟠 Работа котла <b>разрешена</b>"
#define CONTROL_THERMOSTAT_NOTIFY_OFF             "🟤 Работа котла <b>заблокирована</b>"

Как видите – ничего сложного… По аналогии вы теперь можете сделать управление вентилятором в ванной комнате или погребе или тепловентилятором в гараже.

 

В качестве домашнего задания

Теперь, при должной фантазии, вы сможете сделать сколь угодно сложный и удобный термостат своими руками. Конечно, данный вариант потребуется немного “допилить” – например добавить пониженный уровень температуры вне основного расписания. Или, например сделать его с недельным расписанием. А кроме того я вам предлагаю подумать вот над чем….

Бич всех термостатов – как ни странно – проветривание. Вы открываете окно, температура в комнате с термостатом резко падает, он включает котел, но при этом в остальных помещениях очень скоро становится слишком жарко.

Какие будут ваши идеи насчёт преодоления этого недостатка?


Ссылки

Как всегда, новая прошивка доступна в репозитории: GitHub – kotyara12/telemeter_dzen: Термостат + охранно-пожарная сигнализация

 

Все статьи цикла “Термостат и ОПС”:

Прошивка K12 для ESP32 и ESP-IDF:

Дополнительные статьи, которые применимы к любым устройствам, запрограммированным с помощью тех же самых библиотек.

 

💠 Полный архив статей вы найдете здесь

 


Пожалуйста, оцените статью:
[ 0 из 5, всего 0 оценок ]

9 комментариев для “Термостат на ESP32 с удаленным управлением. Часть 9. Термостат и управление нагрузкой”

  1. Александр

    Здравствуйте, очень интересный цикл статей, я новичок в этой теме поэтому подскажите а есть возможность вывод данных на какой-либо дисплей непосредственно с контролера ??

    1. Здравствуйте. Конечно есть, как и на всяком микроконтроллере. Подключайте нужную библиотеку и работайте. Но в данном устройстве он не предусмотрен. Этот термостат больше для удаленной дачи, чем для дома. Хотя, если есть желание – две шины IIC на разъемы выведены, подключить к любой из них техническая возможность есть.

  2. “Избавиться” от проветривания можно установкой датчиков температуры во всех комнатах и по температуре от всех датчиков “принимать решение ” включать котёл или нет.

  3. а как можно новичку который не знает С++ получить готовый код, может купить или бесплатно, спасибо

    1. Всё есть на гитхабе, в открытом доступе. Но подправить под ваши данные все равно потребуется – как минимум имя сети, имя сервера, пароли и т.д.

  4. у меня самодельный электрокотел неделю как работает, управление через Aqara датчик температуры и реле, если реле включится а батарея в датчике температуры сдохнет то котел будет греть до кипятка, хотелось бы чтото без управления через воздух, тоесть автономная работа с управлением через энкодер или телефон, есть варианты на ютюбе но они не работают почемуто

  5. Понятно, это я подправить могу конечно, я делал всякие термометры и дальномеры на esp, но это так, копировать вставить

  6. Можете тыкать в правильное направление, может ссылка есть на такой проект, очень мне поможете, спасибо

Добавить комментарий

Ваш адрес email не будет опубликован. Обязательные поля помечены *