Сценарии - новая версия API
-
Добрый день.
this.isChanged(device1) проверяет свойство Error устройства device1 ?
В сценарии есть команда
if(this.isChanged(TERMOREG_1)) {
//
}Сценарий запускается по изменению, но при этом результат проверки условия false.
02.12 16:10:27.687 Check(TERMOREG_1) => true
02.12 16:10:27.688 Started
02.12 16:10:27.689 isChanged(TERMOREG_1,aval)=false Changed: {"TERMOREG_1":{"err":"Device timeout error!"}}
02.12 16:10:27.689 isChanged(LAMP_32,dval)=false Changed: {"TERMOREG_1":{"err":"Device timeout error!"}}
02.12 16:10:27.689 isChanged(TERMOREG_3_1,aval)=false Changed: {"TERMOREG_1":{"err":"Device timeout error!"}}
02.12 16:10:27.689 StoppedТип устройства: аналоговый датчик.
-
@fanagor, добрый день.
Метод this.isChanged позволяет узнать, какое событие стало причиной запуска сценария:
this.isChanged(device, 'setpoint') - изменили уставку
this.isChanged(device, 'auto') - изменился флаг auto
this.isChanged(device, 'value') - изменилось основное значениеthis.isChanged(device) сокращенная форма для this.isChanged(device, 'value'), то есть =true, если сценарий запущен при изменении основного свойства устройства (aval для аналогового, dval для дискретного устройства)
isChanged(TERMOREG_1,aval)=false Changed: {"TERMOREG_1":{"err":"Device timeout error!"}} означает:
Сценарий запустился при изменении основного значения TERMOREG_1?
Нет, изменился флаг ошибки "err", так как сработал таймаут
Истинным в данном случае будет
this.isChanged(TERMOREG_1, 'error') или this.isChanged(TERMOREG_1, 'err')
-
Это сообщение удалено!
-
Добрый день!
При использовании функции boot() сценарий запускается при старте сервера и остается постоянно активным хотя в нем никаких таймеров и слушателей нет. Так и должно быть?
-
@Lost, добрый день!
Нет, так быть не должно.
Функция boot должна проверить условие, и если истина, запускается функция start(). Дальше работает как обычно. Этот же сценарий может запускаться дальше по startOnChange() или интерактивноscript({ boot() { return lamp.isOn(); // Запустит на старте функцию start, если выполнено условие // return true; // Будет всегда запускать на старте функцию start }, start() { ..... },
В самой boot никаких действий делать не надо!
-
Я и не делал. Сделано пока вот так для пробы:
script({
boot() {
return true;
},
start() {
global.set(‘Armed’, 1);
},
Скрипт после перезагрузки сервера запускается и висит активным в рабочих сценариях (есть зеленая галочка). Вручную останавливается. Если вручную запустить то он после отработки останавливается (увеличивается счетчик количества запусков). Что не так?
-
@Lost, Вы правы, это баг
Сценарий с boot, не имеющий асинхронных механизмов, остается в состоянии активный (зеленая галочка)
Исправим в версии 4.7.7, выйдет на следующей неделе
-
Здравствуйте, уважаемые разработчики и форумчане! Поздравляю всех с наступившим Новым 2021 годом!
Желаю всем творческих успехов.
А теперь о проблеме. Пишу сценарий для управления новогодней гирляндой. У нее есть http API. При управлении используются данные из предыдущих ответов на запрос. Проблема в том, что не получается распарсить ответ и присвоить его разным переменным. Вот, собственно, сценарий/** * @name twinkly_http * @desc * @version 4 */ script({ host: '192.168.144.68', port: 80, method: '', path: '', myObj: '', start() { // login this.path = '/xled/v1/login'; this.method = 'POST'; this.send_http(); // verify // Действие }, send_http() { const http = require('http'); const querystring = require('querystring'); const postData = JSON.stringify({ challenge: 'AAECAwQFBgcICQoLDA0ODxAREhMUFRYXGBkaGxwdHh8=' }); this.log('postData: '+postData); var options = { hostname: this.host, port: this.port, path: this.path, method: this.method, headers: { 'Host': '192.168.144.68', 'Content-Type': 'application/json', 'Content-Length': postData.length //Buffer.byteLength(postData) } }; var req = http.request(options, (res) => { this.log('STATUS: '+res.statusCode); this.log('HEADERS:' +JSON.stringify(res.headers)); res.setEncoding('utf8'); res.on('data', (chunk) => this.myObj += chunk); res.on('end', () => { this.log('Data:' +this.myObj); const data = JSON.parse(this.myObj); this.log('Code:' +data); this.log(typeof this.myObj === 'string'); this.log('No more data in response.'); }); }); req.on('error', (e) => { this.log('problem with request: ' +e.message); }); // write data to request body req.write(postData); req.end(); } });
А вот сообщение в отладчике
03.01 13:15:08.970 Started
03.01 13:15:08.971 log: postData: {"challenge":"AAECAwQFBgcICQoLDA0ODxAREhMUFRYXGBkaGxwdHh8="}
03.01 13:15:08.994 Stopped
03.01 13:15:09.013 log: STATUS: 200
03.01 13:15:09.017 log: HEADERS:{"connection":"close","content-length":"155","content-type":"application/json"}
03.01 13:15:09.021 log: Data:{"authentication_token":"4Be8Ix3sQBQ=","authentication_token_expires_in":14400,"challenge-response":"7ef99322220ff8a9c45aa1efa26ba01d2320aecc","code":1000}
03.01 13:15:09.024 log: Code:[object Object]
03.01 13:15:09.026 log: true
03.01 13:15:09.028 log: No more data in response.Данные из строки Data не могу получить по отдельности. Перечитал разные доки, форумы. Ничего не получается. Что я делаю не так?
-
@int144, добрый день! Удачи в Новом году!!!
Данные из строки Data не могу получить по отдельности.
Если только в этом проблема, то здесь у вас уже все есть, нужно брать свойства data через точку:
const data = JSON.parse(this.myObj); // из строки получился прекрасный объект, дальше присваиваете свойствам сценария или локально:
this.authentication_token = data.authentication_token;
const challenge-response = data.challenge-response;
-
@intrapro, спасибо за быстрый ответ.
Заработало, хотя вроде пробовал так, но может не совсем так. Но!
Значения выводятся в лог, только если я их присвоил в разделе res.on('end'..., а если я пытаюсь вывести в лог присвоенные значения из start после вызова this.send_http, то получаю пустые значения
-
@int144, все верно, start работает синхронно, он просто запускает запрос в this.send_http и идет дальше. Не ждет, когда придет ответ на запрос.
03.01 13:15:08.994 Stopped - здесь start уже работу свою закончил
Можно взвести таймер (на 1 сек или меньше, 0.1 даже достаточно). И в этой функции значения уже должны быть
-
@intrapro, а таймер взводить в start?
-
@int144, да
-
@intrapro, спасибо, буду пробовать
-
Добрый день!
Не работает команда операционной системы
Вот сценарийconst mon = Device("POWER_MIRROR"); // Запускаем сценарий при изменении состояния выключателя зеркала startOnChange(mon); script({ start() { if (mon.isOff()) // Если он выключен this.execOS(`/home/pi/mirror_off.sh`); // Даем команду на выключение монитора else this.execOS(`/home/pi/mirror_on.sh`); // Иначе - на включение } });
В отладчике пишет
04.01 15:56:49.463 Trigger POWER_MIRROR
04.01 15:56:49.464 Started
04.01 15:56:49.465 execOS: /home/pi/mirror_off.sh
04.01 15:56:49.492 Stopped
04.01 15:56:49.522 execOS error: Error: Command failed: /home/pi/mirror_off.sh
No protocol specified
xset: unable to open display ":0.0"04.01 15:56:49.523 stdout:
04.01 15:56:49.524 stderr: No protocol specified
xset: unable to open display ":0.0"Хотя из терминала все прекрасно срабатывает. Может кто-то подсказать куда копать?
-
Подскажите как можно вывести среднюю температуру с 3 датчиков температури в сценарии? Хочу чтобы рекуператор менял уставку в зависимости от температуры в помещении.
-
@amgstone, ну а в чем проблема? Подтягиваете в сценарий своих 3 датчика температуры, в переменную заносите среднюю температуру от них и присваиваете 4-му датчику усредненное значение, по которому будет работать рекуператор.
-
@Alex_Jet от непонимаю как ето сделать в ih.
-
let avr = (t1.value + t2.value + t3.value)/3; //Закон усреднения показаний от трех датчиков this.assign(t_avr, "value", avr); //Присвоение аналоговому датчику значения
-
@Alex_Jet спасибо, все получилось))))