Как стать автором
Обновить
122.03
Wiren Board
Оборудование для автоматизации и мониторинга

Пишем скрипты wb-rules с ChatGPT: быстро, просто, эффективно

Уровень сложностиПростой
Время на прочтение7 мин
Количество просмотров3.1K

Ранее я написал несколько статей про подключение беспроводных Zigbee-устройств к контроллеру Wiren Board. Сначала я использовал для программирования сценариев оболочку Sprut.hub, затем перешел на Node-RED. Наконец, я разобрался в языке программирования wb-rules и написал сценарии на нем. Недавно я начал знакомиться с ChatGPT и удивился тому, что он умеет в wb-rules.

Действительно умеет
Действительно умеет

Что ж настало время проверить возможности ChatGPT. Сможет ли он запрограммировать на wb-rules сценарии, которые я использовал в своих статьях ранее? Проверим!

В статье «Подключаем Zigbee-устройства к контроллеру Wiren Board и пишем сценарии на wb-rules» я подробно описал, как установить модуль WBE2R-R-ZIGBEE v.2 в слот контроллера Wiren Board, затем инсталлировать бесплатный программный пакет zigbee2mqtt и подключить устройства Zigbee.

Я проделал все описанные операции и успешно подключил все четыре устройства Zigbee. Задал им понятные имена Lamp, Socket, Motion_Sensor, Temperature_Sensor, но добавил к каждому год 2024, чтобы было меньше путаницы. Пробелы я заменил подчеркиваниями.

Я задал устройствам Zigbee понятные имена
Я задал устройствам Zigbee понятные имена

Через web-интерфейс контроллера Wiren Board 7 управлять устройствами Zigbee нельзя, следует использовать скрипты wb-rules.

Сценарии автоматизации

Настало время написать скрипты wb-rules, но не самому, а с помощью ChatGPT. Я буду использовать модель GPT-4o, которая была доступна на дату публикации. С базовой информацией о скриптах wb-rules можно ознакомиться здесь. Там же приведены примеры термостата и темной комнаты.

Мне потребовался интерфейс Wiren Board, который доступен по IP-адресу. В нем есть полезное меню «Настройки» — «Каналы MQTT», где можно посмотреть все доступные устройства, контролы и темы MQTT. Там же приведены текущие значения. Все свои устройства Zigbee я здесь увидел.

Каналы MQTT
Каналы MQTT

Сами скрипты следует добавлять в пункте «Правила».

Правила
Правила

Сценарий термостата

Здесь я буду использовать умную розетку Aqara Smart Plug и датчик влажности и температуры Aqara Temperature and Humidity Sensor. Они подключены к контроллеру Wiren Board через Zigbee.

Я составил следующий запрос. По опыту работы лучше сразу дать ChatGPT информацию о контролах устройств, саму логику он программирует хорошо.

Напиши мне скрипт для сценария термостата. Я хочу использовать значение температуры датчика, по которому нужно сделать условное ветвление. Если температура меньше 19 градусов, то включаю розетку. Если  температура больше 23 градусов, тогда выключаю розетку. Если температура находится в диапазоне между 19 и 23 градусами, то ничего не делаю.

Название датчик температуры: Temperature_Sensor_2024

Название розетки: Socket_2024

Датчик температуры и розетка работают через модуль wb-zigbee2mqtt. Как с ним работать прочитай в файле (я приложил сохраненную страницу HTML).

Примечание: я специально вставил картинкой, чтобы плохой код не индексировался
Примечание: я специально вставил картинкой, чтобы плохой код не индексировался

Как видим, скрипт почти правильный. Разве что ChatGPT не владеет полностью всем синтаксисом контрола, в строке: 

whenChanged: "zigbee2mqtt/Temperature_Sensor_2024",

должен быть код:

whenChanged: "Temperature_Sensor_2024/temperature",

Попробую скормить страницу, где расписаны контролы датчика температуры (она здесь).

Теперь код почти верный, добавка zigbee2mqtt в топике температуры лишняя. Но эта особенность реализации zigbee2mqtt под Wiren Board, ChatGPT может ее и не знать. Попросил убрать и получил финальный готовый код термостата. Вот он: 

defineRule("thermostat_control", {
  whenChanged: "Temperature_Sensor_2024/temperature",
  then: function (newValue, devName, cellName) {
    var temperature = parseFloat(newValue); // Получаем значение температуры
    if (temperature < 19) {
      publish("zigbee2mqtt/Socket_2024/set", JSON.stringify({ state: "ON" }), 2, false); // Включаем розетку
    } else if (temperature > 23) {
      publish("zigbee2mqtt/Socket_2024/set", JSON.stringify({ state: "OFF" }), 2, false); // Выключаем розетку
    }
  }
});
Скриншот диалога

Сценарий темной комнаты

Перейдем к сценарию темной комнаты. Здесь я буду использовать лампу Aqara LED Light Bulb, E27, 9Вт и датчик движения Aqara Motion Sensor. Они подключены к контроллеру Wiren Board через Zigbee.

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

Если движение есть (датчик отправил true), то включаем лампу, сбрасываем таймер.

Если движения нет (датчик отправил false), то ставим задержку 2 минуты, а после ее истечения выключаем лампу. Если во время задержки будет определено движение, то таймер сбросится.

Название датчика движения: Motion_Sensor_2024

Название лампы: Lamp_2024

В файлах посмотри, как правильно работать с контролами датчика движения и розетки, а также с модулем wb-zigbee2mqtt, через который они подключены.

Я приложил страницы с контролами лампы и датчика движения. Датчик температуры и розетка работают через модуль wb-zigbee2mqtt. Как с ним работать можно прочитать в файле (тоже приложил).

Скриншот диалога

Продолжаем отладку. Здесь я столкнулся с той же ошибкой, что и в предыдущей статье: датчик высылает true или false как текст. Это «фича» данного датчика движения, ChatGPT может ее и не знать. Попрошу поправить код.

Скриншот диалога

Что ж, почти правильно, разве что ChatGPT переусердствовал с добавлением zigbee2mqtt во все топики. Поправим.

Вот правильный код:

var motionTimer = null;

defineRule("motion_control", {
  whenChanged: "Motion_Sensor_2024/occupancy",
  then: function (newValue, devName, cellName) {
    if (newValue === "true") {
      // Движение обнаружено: включаем свет и сбрасываем таймер
      publish("zigbee2mqtt/Lamp_2024/set", JSON.stringify({ state: "ON" }));
      if (motionTimer) {
        clearTimeout(motionTimer); // Сбрасываем существующий таймер
        motionTimer = null;
      }
    } else if (newValue === "false") {
      // Нет движения: устанавливаем таймер на 2 минуты для выключения света
      if (motionTimer) {
        clearTimeout(motionTimer); // Сбрасываем предыдущий таймер
      }
      motionTimer = setTimeout(function () {
        publish("zigbee2mqtt/Lamp_2024/set", JSON.stringify({ state: "OFF" }));
        motionTimer = null; // Сбрасываем таймер
      }, 120000); // 2 минуты = 120000 миллисекунд
    }
  }
});

Для темной комнаты приведенного выше скрипта достаточно. Но допустим, нам нужно включать лампу при наличии движения на веранде или в коридоре с окнами, причем только в темноте или сумерках. Сначала нужно выбрать пороговый уровень включения датчика. Он должен быть ниже освещенности от солнечного света, чтобы днем датчик не срабатывал. Но уровень должен быть выше освещенности от ламп, чтобы датчик продолжал срабатывать после их включения (допустим, в коридоре продолжается движение на протяжении десяти минут, свет должен гореть). Я взял за порог 100 люкс и добавил в скрипт еще одно условие. Датчик движения будет работать только при яркости меньше 100 люкс. Если яркость больше 100 люкс, лампа выключится. Вы можете добавить любое другое условие, исходя из конфигурации вашего умного дома.

Скрипт рабочий, давай его улучшим. Я хочу, чтобы лампа включалась только при яркости меньше 100 люкс. Если яркость больше 100 люкс, то лампа не включается.

Скриншот диалога

Скрипт получился полностью рабочий, вот его код.

var motionTimer = null;

defineRule("motion_control", {
  whenChanged: "Motion_Sensor_2024/occupancy",
  then: function (newValue, devName, cellName) {
    if (newValue === "true") {
      // Получаем текущую яркость
      var illuminance = parseFloat(dev["Motion_Sensor_2024"]["illuminance"]);
      
      // Проверяем уровень освещенности
      if (illuminance < 100) {
        // Яркость ниже 100 люкс: включаем свет и сбрасываем таймер
        publish("zigbee2mqtt/Lamp_2024/set", JSON.stringify({ state: "ON" }));
        if (motionTimer) {
          clearTimeout(motionTimer); // Сбрасываем существующий таймер
          motionTimer = null;
        }
      }
    } else if (newValue === "false") {
      // Нет движения: устанавливаем таймер на 2 минуты для выключения света
      if (motionTimer) {
        clearTimeout(motionTimer); // Сбрасываем предыдущий таймер
      }
      motionTimer = setTimeout(function () {
        publish("zigbee2mqtt/Lamp_2024/set", JSON.stringify({ state: "OFF" }));
        motionTimer = null; // Сбрасываем таймер
      }, 120000); // 2 минуты = 120000 миллисекунд
    }
  }
});

Сценарий реле

Два сценария выше описывают работу беспроводных устройств Zigbee с контроллером Wiren Board, но почему бы не добавить проводные модули? В моем тестовом чемодане установлен модуль реле WB-MR6C v.2, к трем каналам которого (K1, K2, K3) штатно подключены индикаторы трех кнопок.

Схема тестового чемодана
Схема тестового чемодана
Кольцевые индикаторы кнопок
Кольцевые индикаторы кнопок
Модули в тестовом чемодане, в том числе релейный WB-MR6C v.2
Модули в тестовом чемодане, в том числе релейный WB-MR6C v.2

На странице с примерами можно посмотреть, как правильно работать с модулями реле. Я «скормил» эту страницу ChatGPT при постановке задачи. 

Давай дополним сценарий и будем включать по условию не только лампу, но и каналы K1, K2 и K3 проводного модуля реле с названием wb-mr6c_46. 

Примеры кода работы с реле ты можешь посмотреть в файле, который я приложил.

ChatGPT сгенерировал рабочий код с первого раза.

Скриншот диалога

Ниже я приведу код.

var motionTimer = null;

defineRule("motion_control", {
  whenChanged: "Motion_Sensor_2024/occupancy",
  then: function (newValue, devName, cellName) {
    if (newValue === "true") {
      // Получаем текущую яркость
      var illuminance = parseFloat(dev["Motion_Sensor_2024"]["illuminance"]);

      // Проверяем уровень освещенности
      if (illuminance < 100) {
        // Яркость ниже 100 люкс: включаем свет и реле, сбрасываем таймер
        publish("zigbee2mqtt/Lamp_2024/set", JSON.stringify({ state: "ON" }));
        dev["wb-mr6c_46/K1"] = true;
        dev["wb-mr6c_46/K2"] = true;
        dev["wb-mr6c_46/K3"] = true;

        if (motionTimer) {
          clearTimeout(motionTimer); // Сбрасываем существующий таймер
          motionTimer = null;
        }
      }
    } else if (newValue === "false") {
      // Нет движения: устанавливаем таймер на 2 минуты для выключения света и реле
      if (motionTimer) {
        clearTimeout(motionTimer); // Сбрасываем предыдущий таймер
      }
      motionTimer = setTimeout(function () {
        publish("zigbee2mqtt/Lamp_2024/set", JSON.stringify({ state: "OFF" }));
        dev["wb-mr6c_46/K1"] = false;
        dev["wb-mr6c_46/K2"] = false;
        dev["wb-mr6c_46/K3"] = false;
        motionTimer = null; // Сбрасываем таймер
      }, 120000); // 2 минуты = 120000 миллисекунд
    }
  }
});

Заключение

Да, ChatGPT действительно может программировать на wb-rules. Он устраняет головную боль тех электриков и инженеров, кто панически боится кода и программирования. Конечно, ошибки есть, но с ними можно разбираться. Все ошибки, с которыми я столкнулся, связаны с неправильным синтаксисом контролов, а логика кода везде была правильная. Кроме того, ChatGPT учится на вашем коде, и, если указывать ему на ошибки, он будет их исправлять.

Чтобы код был корректный, не забывайте «скармливать» документацию с примерами wb-rules и подробное описание правил wb-rules, правильными контролами устройств Zigbee и т.д. Можно дать ChatGPT и эту статью с примерами кода.

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

А вы уже использовали ChatGPT, чтобы писать код на wb-rules? Есть ли у вас положительный или отрицательный опыт? Пишите в комментариях.

Теги:
Хабы:
Всего голосов 8: ↑7 и ↓1+8
Комментарии7

Публикации

Информация

Сайт
wirenboard.com
Дата регистрации
Дата основания
Численность
101–200 человек
Местоположение
Россия
Представитель
Саша Дегтярев

Истории