Сценарии - новая версия API



  • Несколько вопросов по js:

    1. Как в callback передать параметры? В отладчике такая ошибка:

    29.12 09:26:38.091 Done timer T2
    29.12 09:26:38.091 exec function TurnOffOLED(megad5,30)
    29.12 09:26:38.092 Error function TurnOffOLED(megad5,30):sceneSet[name][func] is not a function
    
    

    Задача - после передачи данных на дисплей хочу чтобы он через 10 секунд выключился:

        DataToOLED(text, value, nameunit, channel) {
          this.TurnOnOLED(nameunit, channel);
    
          //Форматирование цифровых значений
          value = value.toFixed(1);
          let str = value.toString();
          if(str.length < 4 && value > 0) str = 's+' +str;
          else if(str.length < 4) str = 'ss' +str;
          else if(str.length < 5) str = 's' +str;
    
          //Вывод команд на дисплей
          this.pluginCommand({unit: nameunit, command: '/sec/?pt=' +channel+ '&text=' +text+ '&col=0&row=0'});
          this.pluginCommand({unit: nameunit, command: '/sec/?pt=' +channel+ '&text=' +str+ ':'});
    
          this.startTimer("T2", 10, "TurnOffOLED(" +nameunit+ "," +channel+ ")");
        },
    
    
    

    Сама функция выключения:

        TurnOffOLED(nameunit, channel) {
          this.pluginCommand({unit: nameunit, command: '/sec/?cmd=' +channel+ ':0'});
        }
    
    
    

    2. Первый раз столкнулся с задачей конвертировать строку из utf-8 в cp866…в js это сделать оказалось не тривиальной задачей 😞 . Люди на форумах так и пишут - делайте это в php... можете помочь?

    3. Придумал алгоритм отображения различных показаний на OLED-экране. Подскажите как при срабатывании кнопки в сценарии можно каждый раз увеличивать значение переменно? Пока сделал с помощью глобальной переменной:

          //Перебор параметров при каждом нажатии кнопки
          let value = global.get('Button_Guest') + 1;
          if(isNaN(value) || value > 2) global.set('Button_Guest', 0);
          global.set('Button_Guest', value);
    
    
    


  • Добрый день!

    Может я просто не до конца разобрался в инструкциях. У меня освещением на кухне управляет мультисценарий включения света по датчику движения с учетом аналогового датчика освещенности. Очень хочется сделать так, что бы начиная с 23.00 и до утра основной свет не включался, а включалась подсветка. Т.е. один сценарий прекратил работу, а с 23.00 начал работу другой. Подскажите пожалуйста как это сделать. Сценарий скачан отсюда https://ih-systems.com/ru/scenes-examples/.



  • @Alex_Jet:

    Несколько вопросов по js:

    Начнем с конца

    @Alex_Jet:

    3. Придумал алгоритм отображения различных показаний на OLED-экране. Подскажите как при срабатывании кнопки в сценарии можно каждый раз увеличивать значение переменно? Пока сделал с помощью глобальной переменной:

    Это можно сделать, объявив переменную сценария. Сценарий у нас - не набор функций, а объект, поэтому может иметь свойство - переменную, которая сохраняется между запусками: Объявить ее нужно прямо в объекте, а не в теле функции. Инициализация начальным значением происходит при загрузке сценария (старте сервера). Далее переменную можно использовать через this в любом месте сценария

    
    script({
       myVar:0,
       start() {
        if (this.myVar <10) this.myVar += 1;
       },
    
      clear() {
        this.myVar = 0;
      } 
    })
    
    
    

    @Alex_Jet:

    2. Первый раз столкнулся с задачей конвертировать строку из utf-8 в cp866…в js это сделать оказалось не тривиальной задачей 😞 . Люди на форумах так и пишут - делайте это в php... можете помочь?

    Для конвертирования строк есть готовые модули для node.js, которые можно загрузить через npm

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

    Но если очень хочется, то можно попробовать 🙂

    Есть библиотека iconv-lite, нужно ее установить в /opt/intrahouse-c/backend/

    Можно установить вручную, можно просто дописать в /opt/intrahouse-c/backend/package.json в раздел dependencies

    "iconv-lite": "^0.4.24",

    И перезагрузить сервер

    В папке /opt/intrahouse-c/backend/node_modules должна появиться папка iconv-lite

    В сценарии для кодирования написать:

    let buf = require("iconv-lite").encode("привет, это строка для кодирования", "cp866")

    Получили буфер байт, его надо включить в url как строку (здесь результат не гарантирован, но должно сработать)

    let text = buf.toString()

    @Alex_Jet:

    1. Как в callback передать параметры?

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



  • Спасибо за ответы! С переменной сценария все получилось.

    Еще вопрос по переменным. Хочу сделать мультисценарий, как в нем описать плагин, который хочу выбирать в "Запуск для устройств" чтобы использовать для вывода информации на OLED? В конечном счете надо чтобы его имя подставлялось в

    this.pluginCommand({unit: plugin, command: '/sec/?pt=' +channel+ '&disp_cmd=1&row=0'});
    
    

    А еще как сделать чтобы в "Запуск для устройств" можно было указать канал плагина?

    Идея такая - в мультисценарии выбираем кнопку, нужные датчики (4 шт.), плагин и номер канала, к которому подключен OLED. В комнатах где будет OLED на нем будут отображаться "свои" показания датчиков.

    Кстати, нашел пару опечаток:

    1. В отладчике пишет - "plugincCommand"

    2. При создании нового сценарий - "Новый сценарийТ"



  • @sergeyygr:

    Добрый день!

    Может я просто не до конца разобрался в инструкциях. У меня освещением на кухне управляет мультисценарий включения света по датчику движения с учетом аналогового датчика освещенности. Очень хочется сделать так, что бы начиная с 23.00 и до утра основной свет не включался, а включалась подсветка. Т.е. один сценарий прекратил работу, а с 23.00 начал работу другой. Подскажите пожалуйста как это сделать. Сценарий скачан отсюда https://ih-systems.com/ru/scenes-examples/.

    Можно сделать так.

    Сценарии освещения по датчику движения оставить как есть. Только они должны учитывать режим Auto.

    Плюс добавить два сценария включения/выключения режима Auto для этих двух светильников.

    Первый сценарий будет запускаться по расписанию в 23.00 Устанавливать "auto" для одного светильника и снимать "auto" для другого

    Второй наоборот, и запускается в 7.00

    Так как эти сценарии запускаются по расписанию, триггер в них не нужен

    В виде блок-схемы сценарий выглядит так:
    scen201.jpg

    Если есть желание, можно немного усовершенствовать. Если в момент переключения режима авто первый светильник был включен, можно его выключить и включить второй светильник:
    scen301.jpg



  • @Alex_Jet:

    Еще вопрос по переменным. Хочу сделать мультисценарий, как в нем описать плагин, который хочу выбирать в "Запуск для устройств" чтобы использовать для вывода информации на OLED? В конечном счете надо чтобы его имя подставлялось в

    > this.pluginCommand({unit: plugin, command: '/sec/?pt=' +channel+ '&disp_cmd=1&row=0'});
    > 
    

    А еще как сделать чтобы в "Запуск для устройств" можно было указать канал плагина?

    Идея такая - в мультисценарии выбираем кнопку, нужные датчики (4 шт.), плагин и номер канала, к которому подключен OLED. В комнатах где будет OLED на нем будут отображаться "свои" показания датчиков.

    Через "Запуск для устройств" так сделать не получится. Ну, или очень криво.

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

    На данный момент вижу только не универсальное решение: прямо в сценарии задать какую-то связь устройств и плагин+канал 😞

     let plugin;
     let channel;
     switch (button.id) {
      case 'BUTTON1':  plugin = 'megad1';  channel = 32; break;
      ....
     } 
    
    
    


  • @homa:

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

    Ошибка исправлена в версии 4.4.9



  • @intrapro:

    …можно просто дописать в /opt/intrahouse-c/backend/package.json в раздел dependencies

    "iconv-lite": "^0.4.24",

    И перезагрузить сервер

    В папке /opt/intrahouse-c/backend/node_modules должна появиться папка iconv-lite

    После перезагрузки iH папка появилась, но сценарий заработал только когда указал полный путь: let buf = require("/opt/intrahouse-c/backend/node_modules/iconv-lite").encode(text, "cp866");

    По сути верно, поскольку сами скрипты лежат тут: /var/lib/intrahouse-c/projects/"PROJECT_NAME"/scenes/script



  • @Alex_Jet:

    @intrapro:

    …можно просто дописать в /opt/intrahouse-c/backend/package.json в раздел dependencies

    "iconv-lite": "^0.4.24",

    И перезагрузить сервер

    В папке /opt/intrahouse-c/backend/node_modules должна появиться папка iconv-lite

    После перезагрузки iH папка появилась, но сценарий заработал только когда указал полный путь: let buf = require("/opt/intrahouse-c/backend/node_modules/iconv-lite").encode(text, "cp866");

    По сути верно, поскольку сами скрипты лежат тут: /var/lib/intrahouse-c/projects/"PROJECT_NAME"/scenes/script

    Добрый день, здорово что у Вас получилось 😉



  • @intrahouse:

    @sergeyygr:

    Добрый день!

    Может я просто не до конца разобрался в инструкциях. У меня освещением на кухне управляет мультисценарий включения света по датчику движения с учетом аналогового датчика освещенности. Очень хочется сделать так, что бы начиная с 23.00 и до утра основной свет не включался, а включалась подсветка. Т.е. один сценарий прекратил работу, а с 23.00 начал работу другой. Подскажите пожалуйста как это сделать. Сценарий скачан отсюда https://ih-systems.com/ru/scenes-examples/.

    Можно сделать так.

    Сценарии освещения по датчику движения оставить как есть. Только они должны учитывать режим Auto.

    Плюс добавить два сценария включения/выключения режима Auto для этих двух светильников.

    Первый сценарий будет запускаться по расписанию в 23.00 Устанавливать "auto" для одного светильника и снимать "auto" для другого

    Второй наоборот, и запускается в 7.00

    Так как эти сценарии запускаются по расписанию, триггер в них не нужен

    В виде блок-схемы сценарий выглядит так:

    scen201.jpg

    Если есть желание, можно немного усовершенствовать. Если в момент переключения режима авто первый светильник был включен, можно его выключить и включить второй светильник:

    scen301.jpg

    Спасибо! Работает! В сценарии добавил автоматический режим (в шаблоне небыло).



  • @dev:

    Добрый день, здорово что у Вас получилось 😉

    Коллеги, с наступившим Новым Годом Вас!!!

    Я немного поторопился с выводами о работе iconv…сценарий при запуске не ругался и в отладчике что-то подобное cp866 выводилось. Однако на практике оказалось не все так радужно - на OLED ничего нет. Связываю это с большой скорострельностью node.js и его "параллельной работе". Да, кстати, при обновлении системы из dependencies исчезает дописанная строка с версией iconv-lite.

    Еще смысл такой, что функцию вывода данных на OLED пришлось наворотить таймерами...таймер для очистки первой строки дисплея, для вывода новой первой строки, для вывода значения параметра... иначе все не читабельно и с кучей артефактов:(

        DataToOLED(text, value, unit, plugin, channel) {
          let iconv = require("/opt/intrahouse-c/backend/node_modules/iconv-lite");
          let buf = iconv.encode(text, "cp866");
          this.text = buf.toString();
    
          //Центрирование параметра на OLED
          let length = this.text.length;
          let col = (128 - length*6)/2;
          this.col = col.toFixed(0);
    
          //Форматирование цифровых значений
          let str = ""; //Вспомогательная переменная
          if(value < 100) value = value.toFixed(1);
          str = value.toString();
          if(str.length < 4 && value > 0) this.str = 's+' +str;
          else if(str.length < 4) this.str = 'ss' +str;
          else if(str.length < 5) this.str = 's' +str;
    
          //Выбор единицы измерения
          if(unit == "grad") this.unit = ":";
          else if(unit == "percent") this.unit = "s"; //"%"
          else if(unit == "ppm") this.unit = "s"; //"p"
    
          //Вывод команд на дисплей
          this.ClearOLED(); //Очищаем первую строчку
          this.startTimer("T2", 0.3, "SendTextToOLED");
          this.startTimer("T3", 0.5, "SendValueToOLED");
          if(!this.oled_state) {  //Если дисплей выключен
            //то включаем его после обновления параметра
            this.startTimer("T4", 0.7, "TurnOnOLED");
          }
        },
    
    

    На php все гораздо легче и работает на порядок предсказуемо, никакие задержки не нужны:

    $my_temp = number_format($temp_bme280, 1); // Получаем значение из базы и форматируем его
    if ( strlen($my_temp) < 4 && $my_temp > 0 )
    $my_temp = "s+$my_temp";
    elseif (strlen($my_temp) < 4 )
    $my_temp = "ss$my_temp";
    elseif (strlen($my_temp) < 5 )
    $my_temp = "s$my_temp";
    
    $text = "Улица";
    $text = iconv("utf-8", "cp866", $text);
    $length = strlen($text);
    $col = (128 - $length*6)/2;
    $col = round($col);
    
    file_get_contents("http://192.168.11.25/sec/?pt=31&disp_cmd=1&row=0");
    file_get_contents("http://192.168.11.25/sec/?pt=31&text=" .$text. "&col=" .$col. "&row=0");
    file_get_contents("http://192.168.11.25/sec/?pt=31&text=$my_temp:");
    
    


  • Всех с Новым Годом!! Удачи и интересных задач в Новом Году!!
    @Alex_Jet:

    Я немного поторопился с выводами о работе iconv…сценарий при запуске не ругался и в отладчике что-то подобное cp866 выводилось. Однако на практике оказалось не все так радужно - на OLED ничего нет. Связываю это с большой скорострельностью node.js и его "параллельной работе".

    То есть русские буквы не выводятся, а другие символы выводятся? Тогда дело не в скорости. Нужно попробовать вывести просто текст. И разбираться с перекодировкой.

    @Alex_Jet:

    Да, кстати, при обновлении системы из dependencies исчезает дописанная строка с версией iconv-lite.

    Это нормально. Вы же уже установили модуль. Данные dependencies в package.json нужны только для установки новых пакетов.

    @Alex_Jet:

    Еще смысл такой, что функцию вывода данных на OLED пришлось наворотить таймерами…таймер для очистки первой строки дисплея, для вывода новой первой строки, для вывода значения параметра... иначе все не читабельно и с кучей артефактов:(

    Была мысль добавить в pluginCommand время - интервал для запуска. Здесь это как раз востребовано.

    Добавим в новом году 🙂



  • Подскажите (только начал осваивать систему), а можно ли в сценариях выполнить произвольный запрос? Имеется TFT wi-fi дисплей. Вывод текста на дисплей выполняется запросом [ip адрес]/lcdmsg?st=X&txt=text. Где X - номер строки, text - текст, который надо вывести. Необходимо сформировать строку текста (text) со значением температуры, полученной от устройства и вывести эту строку на дисплей запросом. Подскажите как это сделать?



  • @gis:

    Подскажите (только начал осваивать систему), а можно ли в сценариях выполнить произвольный запрос? Имеется TFT wi-fi дисплей. Вывод текста на дисплей выполняется запросом [ip адрес]/lcdmsg?st=X&txt=text. Где X - номер строки, text - текст, который надо вывести. Необходимо сформировать строку текста (text) со значением температуры, полученной от устройства и вывести эту строку на дисплей запросом. Подскажите как это сделать?

    Добрый день, попробуйте так:

    const request = require('/opt/intrahouse-c/backend/node_modules/request');
    const temp = DeviceT("DEVICE_ID");
    
    startOnChange([temp]);
    
    script({
      start() {
         request(`http://127.0.0.1/lcdmsg?st=1&txt=My temp ${temp.getValue()}℃`);
      }
    });
    
    
    


  • @intrapro:

    То есть русские буквы не выводятся, а другие символы выводятся? Тогда дело не в скорости. Нужно попробовать вывести просто текст. И разбираться с перекодировкой.

    Текст на латинице конечно выводится, а вот русские буквы - никак. Хотя на PHP с этим вообще никаких проблем нет (см.код выше).



  • @Alex_Jet:

    @intrapro:

    То есть русские буквы не выводятся, а другие символы выводятся? Тогда дело не в скорости. Нужно попробовать вывести просто текст. И разбираться с перекодировкой.

    Текст на латинице конечно выводится, а вот русские буквы - никак. Хотя на PHP с этим вообще никаких проблем нет (см.код выше).

    Добрый день, сделайте вывод текста который подставляете в url на php и nodejs. Если строки одинаковые то, дело в чем-то другом.



  • @dev:

    Добрый день, попробуйте так:

    > const request = require('/opt/intrahouse-c/backend/node_modules/request');
    > const temp = DeviceT("DEVICE_ID");
    > 
    > startOnChange([temp]);
    > 
    > script({
    >   start() {
    >      request(`http://127.0.0.1/lcdmsg?st=1&txt=My temp ${temp.getValue()}℃`);
    >   }
    > });
    > 
    > 
    

    К сожалению сценарий не сработал, при попытке запуска выдается сообщение внизу экрана:

    Error action: {command 'startscene TFT'} request is not defined

    TFT - название сценария

    DEVICE_ID - подставил идентификатор рабочего датчика температуры



  • @gis:

    К сожалению сценарий не сработал, при попытке запуска выдается сообщение внизу экрана:

    Error action: {command 'startscene TFT'} request is not defined

    TFT - название сценария

    DEVICE_ID - подставил идентификатор рабочего датчика температуры

    Попробуйте так:

    const temp = Device("DEVICE_ID"); 
    
    startOnChange(temp); 
    
    script({
        start() {
            require("http").get(`http://192.168.0.xx/lcdmsg?st=1&txt=${temp.value}` );
        } 
    });
    
    
    


  • @Alex_Jet:

    Текст на латинице конечно выводится, а вот русские буквы - никак. Хотя на PHP с этим вообще никаких проблем нет (см.код выше).

    iconv-lite преобразует из utf8 верно, проблема скорее всего в преобразовании буфера в строку для размещения в url

    let text = buf.toString("hex" или "latin1" или ??)

    Хотелось бы действительно увидеть строку запроса с русскими буквами в кодировке cp866, которую формирует php.

    Например, для символов АБВГ



  • @intrapro:

    @Alex_Jet:

    Текст на латинице конечно выводится, а вот русские буквы - никак. Хотя на PHP с этим вообще никаких проблем нет (см.код выше).

    iconv-lite преобразует из utf8 верно, проблема скорее всего в преобразовании буфера в строку для размещения в url

    let text = buf.toString("hex" или "latin1" или ??)

    Хотелось бы действительно увидеть строку запроса с русскими буквами в кодировке cp866, которую формирует php.

    Например, для символов АБВГ

    
    Nodejs проверить не могу, все онлайн инструменты глючат  :(
    

Авторизуйтесь, чтобы ответить