Jump to content
iT4iT.CLUB

Kitsum

Members
  • Content Count

    389
  • Joined

  • Last visited

  • Days Won

    204

Everything posted by Kitsum

  1. Доброе время суток. Заранее спасибо, что помогаете друг другу! Да, все это возможно. Например, для расчета точки росы Вам понадобятся показания температуры и влажности, а также функция которая и будет производить расчет. Вот её упрощенный вариант. /* Функция расчета точки росы */ float dewPoint(float t, float h) { float a = 17.271; float b = 237.7; float temp = (a * t) / (b + t) + log(h * 0.01); return (b * temp) / (a - temp); } Теперь можно объявить новый сенсор, но для начала опишем настройки для визуализации. knob_t *DP = new knob_t(-40, 125, ".1", "Точка росы", "°C"); Ну и сам сенсор, но помните, что он берет данные от других датчиков, которые должны быть объявлены. У меня они имеют идентификаторы out_temperature и out_humidity для температуры и влажности соответственно. sensors.add(DP, device::out, "out_dewPoint", [&](){ return dewPoint( sensors.get("out_temperature"), sensors.get("out_humidity") ); }); Вот в принципе и все не сложные операции. Для индикации изменения давления "к дождю" необходимо придумать какую-нибудь формулу, скорее всего уже есть наработки в интернете. Вопрос в том, какие параметры потребуются для этого. Лично мне в голову приходит мысль смотреть в логи с датчика давления и влажности, а также рассчитывать абсолютную влажность по ним и смотреть на тенденцию изменения абсолютной и относительной влажности. Для индикации можно использовать значение в процентах или придумать свою условную шкалу, но в любом случае придется рассчитать её приделы. В общем, нужна формула. Для LCD дисплея 20х4 Вам понадобится i2c конвертер, возможно он уже распаян на нем, в противном случае у Вас не хватит свободных портов для подключения. Далее все еще проще, скорее всего у Вас уже имеется библиотека для работы с ним, и наверное это она https://github.com/fdebrabander/Arduino-LiquidCrystal-I2C-library Подключаем библиотеку и объявляем дисплей (например в основном .ino файле), убедитесь, что i2c адрес для дисплея никем не занят. #include <LiquidCrystal_I2C.h> LiquidCrystal_I2C lcd(0x27, 20, 4); Производим инициализацию дисплея, например, в конце функции setup. lcd.begin(); lcd.backlight(); Теперь создадим в планировщике задачу которая будет обновлять данные на дисплее каждые 30 секунд, опишем её сразу после инициализации дисплея. cron.add(cron::time_30s, [&](){ lcd.clear(); lcd.printf("Temperature %.1f C", sensors.get("out_temperature")); }); Первой строкой мы очищаем дисплей, а второй выводим данные. Естественно, что Вы можете изменять позицию курсора. Сразу отмечу, что я не проверял этот код, я просто опираюсь на описания библиотеки (LiquidCrystal_I2C.h) и её "родителя" (print.h) которые посмотрел на GitHub https://github.com/fdebrabander/Arduino-LiquidCrystal-I2C-library/blob/master/LiquidCrystal_I2C.h https://github.com/esp8266/Arduino/blob/master/cores/esp8266/Print.h Но общий смысл должен быть понятен. Что касаемо Blink, то мне с ним не приходилось сталкиваться, дать какие-то рекомендации со своей стороны затрудняюсь, нужно изучать API сервиса. Но я рад, что есть люди @Вадим Гречухин которые это уже реализовали. Перед установкой датчиков стоит обработать все открытые и подвергающиеся воздействия окружающей среды элементы диэлектрическим лаком. Сами сенсоры на время обработки лаком стоит закрыть чем ни будь чтобы не вывести их из строя. Обязательно убедитесь в их работоспособности после этих операций. Я старался размещать датчики сенсорами вниз, особенно фото датчики чтобы уменьшить количество оседаемой на них пыли, у меня её много т.к. проживаю чуть ли не в интимной близости от дороги. Ну а в будке, как мне кажется, лучше размещать их в средней части, на небольшом удалении друг от друга чтобы они не могли влиять на показания соседей. Все это мое субъективное мнение, основанное только на личном опыте. Далеко не факт, что он верный. Из достоверных источников известно точно, что сама будка должна быть расположена в теневой зоне на высоте пары метров от земли и не иметь вблизи других строений. В моем случае нарушены все эти правила т.к будка закреплена на мачте параболической антенны далеко даже не на первом этаже многоэтажного дома на расстояние 20 сантиметров от стены этого дома да еще и на солнечной стороне. В общем все это влияет на показания, но самое критичное это солнечная сторона, старайтесь найти место в тени на весь световой день. Скорее всего сенсоры вышли из строя. У меня аналогичная ситуация произошла с датчиком влажности и датчиком загрязнения воздуха после того как весной (в очередной раз) будка была повреждена и перестала выполнять свои прямые обязанности по защите датчиков. Судя по повреждениям, во время оттепели что-то прилетает с верхних этажей и уничтожает крайний конусный элемент будки. Следом после этого датчики заливает водой... Имейте это в виду при выборе места установки.
  2. @Nickson2006 доброе время суток. Скорее всего Вы не залили файлы из каталога data во flash память. Просмотрите еще раз инструкцию, после прошивки скетча необходимо залить файлы с помощью дополнительного плагина для Arduino IDE.
  3. @post125 доброе время суток. Все очень просто, по умолчанию никаких облаков нет, они подгружаются скриптом через 2 секунды после полной загрузки страницы. Ищите строку setTimeout(loadSky, 2000); Эта строка находится внутри функции sensorsStructure, которая в свою очередь отвечает за первоначальную отрисовку всех сенсоров. Просто замените эту строку на это $(".loading").fadeOut(2000); Это уберет анимацию загрузки страницы. После этих не хитрых операций Вы можете удалить фай cloud.css.gz и его распакованный вариант cloud.css (если он имеется) из флеш памяти микроконтроллера.
  4. @Astron рад, что у Вас все получилось. Да, раздел управления GPIO делался по просьбе людей где-то в этой теме и работает при условии, что датчики с нужными внутренними идентификаторами будут описаны. Его можно расширять, в этом нет ничего сложного, но проблема в количестве свободных портов микроконтроллера, внешне кажется, что ног много, но по факту, большая часть задействована под внутренние нужды микроконтроллера. Эту проблему можно решить, используя i2c расширители портов, но как мне кажется, лучше перейти на ESP32, это будет дешевле и сам контроллер намного интереснее. В текущем проекте много проблем как в логике и коде, так и в концепции в целом, все это делалось одновременно с изучением основ C++, JS и т.д. На данный момент много кода переписано под ESP32, а точнее написано с нуля, но с оглядкой на опыт использования ESP8266. Я постараюсь учесть пожелания всех, кто отписывался в теме, но точно могу сказать, что реализовываться они будут под ESP32. Под ESP8266 глобальных исправлений, скорее всего не будет, но раз интерес к уже опубликованному коду еще не угас, то я постараюсь снять ряд видеороликов в которых покажу как работают разные программные модули проекта и как вносить свои изменения чтобы людям было проще разбираться в моей писанине видя всю картину в целом и не испытывая при этом нравственных страданий. Ну и естественно, что тема поддержки новичков будет продолжать функционировать. Да, это уже учтено и новый интерфейс содержит статическую картинку в фоне, а на маленьких разрешениях только градиентный фон не напрягающий глаза. Кстати, в этом нет ничего сложного, все необходимое уже реализовано, и добавить данный функционал можно буквально за 15-20 минут. Самое сложное, это как оформить красиво картинку на дисплее. Постараюсь найти какой ни-ть i2c дисплей и показать как это сделать. PS: спасибо за поддержку, это очень важно для меня!
  5. Доброе время суток @Astron Тут Вы поспешили, смотрите в чем дело Это сообщение сформировано самой библиотекой и говорит, что вы пытаетесь использовать динамический буфер, который не поддерживается в 6-ой и более старших версиях. Именно по этой причине я указал в инструкции, что необходимо использовать 5-ую версию. Так что вы практически все сделали правильно. На всякий случай, я сформировал архив с Portable Arduino IDE v1.8.9 для Windows с поддержкой ESP8266 и ESP32, а также установленными FS плагинами для загрузки файлов web сервера, базовым набором библиотек (не все нужно обновлять) и скетчем с проектом метеостанции. Я мог что-то упустить, но надеюсь у Вас все получится. Скачать можно тут: https://yadi.sk/d/lXNS4_juPVPYwA
  6. @Astron Эти ошибки явно связаны со средой. Эта ошибка может появится только при использовании ESP8266 выше версии 2.5.0 т.к в библиотеки ES8266WebServer данный метод был переименован в responseCodeToString. По этому, Вы скорее всего ошиблись в версиях или используете не оригинальный репозиторий. Исправить это можно уйдя на более низкую версию или заменив строку String codeTranslate(int code) { return ESP8266WebServer::_responseCodeToString(code); } на эту String codeTranslate(int code) { return ESP8266WebServer::responseCodeToString(code); } Вот репозиторий который использую я http://arduino.esp8266.com/stable/package_esp8266com_index.json Esptool, это утилита, идущая в комплекте со всеми исходниками необходимыми для разработки и загрузки скетчей в Arduino IDE. А Вы производили установку дополнения ESP8266FS. Следовательно, ошибка вызвана неработоспособностью самой среды Arduino IDE, возможно часть файлов было потеряно или их не было вовсе. Попробуйте сделать Portable версию Aduino IDE чтобы не быть привязанным к файлам, раскиданным по системе. Для этого: Скачайте последнюю ZIP версию Arduino IDE, на момент публикации данного поста, это 1.8.9 https://www.arduino.cc/download_handler.php?f=/arduino-1.8.9-windows.zip Распакуйте содержимое архива в любое удобное Вам место. Зайдите в корневой каталог всех извлеченных файлов (пусть он называется \arduino-1.8.9) и создайте там каталог с именем portable Файл плагина ESP8266FS разместить по такому адресу \arduino-1.8.9\portable\sketchbook\tools\ESP8266FS\tool\esp8266fs.jar Все необходимые библиотеки размещайте тут \arduino-1.8.9\portable\sketchbook\libraries А исходники проектов тут \arduino-1.8.9\portable\sketchbook Теперь запустите Arduino IDE и в настройках добавьте репозиторий ESP8266 http://arduino.esp8266.com/stable/package_esp8266com_index.json Добавьте в среду разработки поддержку ESP8266, как описывалось в инструкции или внесите изменения, о которых я говорил чуть выше чтобы все собиралось в версиях 2.5.1+
  7. @qastron Только что проверил в Arduino IDE 1.8.9 с установленной поддержкой ESP8266 2.5.0 и последней версией (на 03.06.2019) библиотеки PubSubClient. Все собирается. Ошибки довольно странные т.к. указывают на не существующие или ошибочные методы. При это в оригинале библиотеки все используемые методы присутствуют. На всякий случай скину библиотеку PubSubClient установленную у меня. Убедитесь, что Вы используете именно ее. pubsubclient-master.zip
  8. Хорошая задачка по поиску подводного камня. Проблема носит синтаксический характер и скрыта в JavaScript. Вот кусочек кода из оригинального файла: $("div.settings div.menuList").click(function() { var id = $(this).attr("id"); $(".settings .menuList[id != " + id + "]").hide(); $(".settings .menuList#" + id).css({"cursor": "default"}); $(".settings #sl-" + id).show(); if(id.match(/(global|mqtt|thingspeak|narodmon|gpio)/i)) { if(id === "gpio") apiGetGPIO(); $(".settings input[type='submit']").show(); } else if(id === "firmware") selfCheckSupportMD5(); else if(id === "spiffs") apiGetSPIFFS(); else if(id === "system") apiGetSystemInfo(); else if(id === "i2c") apiGetI2CInfo(); }); А вот как выглядит регулярное выражение из Вашего файла для этого блока кода id.match(/(global|mqtt|thingspeak|narodmon|gpio|)/i) Проблема в лишнем символе " | " (вертикальная черта) в конце выражения. Этот символ, используемый в рамках круглых скобок обозначает логическое ИЛИ. В итоге Ваш шаблон закончен с логической ошибкой и сводит с ума обработчик. Вернитесь к оригинальному коду или допишите шаблон, полагаю, что Вы хотели сделать это: id.match(/(global|mqtt|thingspeak|narodmon|gpio|scaleW)/i) Последний вариант гарантирует, что в вашей ветке настроек веса появится кнопка сохранить. Если использовать встроенную кнопку сохранения через регулярное выражение, описанное ранее, то по её нажатию данные с формы будут переданы в переменную конфига которую Вы уже инициализировали. Отсюда (index.htm) <input type="text" id="scaleW_checkW" autocomplete="off" placeholder="5"> Сюда (esp8266) conf.add("scaleW_checkW"); Далее в коде микроконтроллера Вы можете работать с этой переменной. Я также заметил, что добавили кнопку "Откалибровать весы". <div class="button" id="calibration">Откалибровать весы</div> Если Вы хотите на неё повесить дополнительное действие, то необходимо организовать обработку события клика по ней $("#sl-scaleW #calibration").click(function() { }); В данном блоке реализуете обработку нажатия. Например, Вы можете выполнить AJAX запрос к WEB серверу ESP8266 и выполнить все необходимые действия на стороне микроконтроллера. В данном случае, возможно, Вам потребуется обновить значение эталонного веса, а только после этого произвести калибровку. Но, тут уже Вы сами решаете, что и как делать.
  9. @post125 попробуйте запустить скрипт в консольном режиме zabbixMqttClient.py window В выводе получите описания этапов подключения, например connecting... connected client id: server.local subscribe on "$SYS/#" subscribe on "kitsum/espWeatherStation/#" subscribe on "kitsum/serverRoom/#" Далее должен начаться вывод сообщений, полученных от брокера 2019-05-30 07:52:46 $SYS/broker/messages/sent 216459 (host: broker, key: topic[messages/sent]) server response: ... 2019-05-30 07:52:46 $SYS/broker/publish/messages/sent 34211 (host: broker, key: topic[publish/messages/sent]) server response: ... 2019-05-30 07:52:46 $SYS/broker/bytes/sent 1809731 (host: broker, key: topic[bytes/sent]) server response: ... 2019-05-30 07:52:46 $SYS/broker/publish/bytes/sent 206329 (host: broker, key: topic[publish/bytes/sent]) server response: ... Больше информации можно получить, включив вывод логов paho, для этого надо найти строку #mqttc.on_log = on_log Приводим её к такому виду mqttc.on_log = on_log Также можно модифицировать скрипт чтобы увидеть не стандартное поведение zabbix_sender Ищем блок if error == None: output = re.search('"processed: (\d+); failed: (\d+); total: (\d+); seconds spent: (\d+\.\d+)"', out$ if output.group(2) != '0': answer = 'failed (time ' + output.group(4) + ')' elif output.group(1) != '0': answer = 'processed (time ' + output.group(4) + ')' else: answer = output.group(0) alert(time.strftime('%Y-%m-%d %H:%M:%S ', time.localtime()) + msg.topic + ' ' + str(msg.payload.decode('utf-8')) + ' (host: ' + host + ', key: ' + key + ') server response: ' + answer) else: alert(error) return 0 Приводим его к такому виду if error == None: output = re.search('"processed: (\d+); failed: (\d+); total: (\d+); seconds spent: (\d+\.\d+)"', out$ if output == None: alert('zabbix_sender error!') return 0 elif output.group(2) != '0': answer = 'failed (time ' + output.group(4) + ')' elif output.group(1) != '0': answer = 'processed (time ' + output.group(4) + ')' else: answer = output.group(0) alert(time.strftime('%Y-%m-%d %H:%M:%S ', time.localtime()) + msg.topic + ' ' + str(msg.payload.decode('utf-8')) + ' (host: ' + host + ', key: ' + key + ') server response: ' + answer) else: alert(error) return 0 Также стоит попробовать отправить данные через zabbix_sender руками. zabbix_sender -vv -z 127.0.0.1 -p 1883 -s broker -k topic[my/test/path] -o 555 Где: 127.0.0.1 - имя или ip mqtt адрес сервера 1883 - порт mqtt сервера broker - имя узла в zabbix на который адресовано сообщение topic[my/test/path] - это ключ элемента данных который присутствует в шаблоне или создан руками 555 - любое значение для теста
  10. @post125 Это полный текст сообщения, которое Вы получаете? Скорее всего есть еще что-то. На крайний случай, на время теста, Вы можете добавить задачу в планировщик пользователя root. sudo crontab -e
  11. Под любой вариант Ubuntu, будь то сервер или рабочая станция, zabbix можно поставить из готовых пакетов. Более того, zabbix не использует GUI, а весь интерфейс реализован на уровне web сервера. Возможны оба варианта, но если уже есть функционирующий MQTT брокер и его работа Вас устраивает, то и смысла переносить его нет. Вы всегда успеете это сделать позже. Если использовать самую распространенную библиотеку "PubSubClient" для реализации MQTT протокола под ESP на C++, то пользователю предоставлен метод publish в качестве третьего параметра принимающий булево значение (true) для установки флага -r. Это требование необходимо выполнить только в том случае, если организовывать сбор данных в Zabbix через внешнюю проверку. В web интерфейсе Zabbix визуальных изменений не произойдет. В зависимости от варианта, выбранного Вами для реализации MQTT протокола в Zabbix, Вам становятся доступны специальные ключи, принимающий в качестве параметра имя топика откуда необходимо забирать данные. Я советую Вам использовать последний вариант, в котором Zabbix ловит данные через ловушки (zabbix trapper), а транзитом данных от MQTT брокера до Zabbix занимается демон, написанный на Python. Прочитайте внимательно выделенный пост выше. Если будут сложности, то пишите.
  12. Это уже зависит от Ваших возможностей по обслуживанию и развитию системы. Операционная система только фундамент, все остальное ставится поверх, будь то готовый софт или Ваши собственные скрипты и программы. В данном случае мне удобнее было бы работать с Ubuntu, но Вам, возможно, проще будет с Windows или с чем-то еще.
  13. Доброе время суток. @post125 за проверку пароля отвечает функция в файле index.htm function checkPassw(passw) { return passw.match(/^.{5,63}$/); } Что соответствует любой строке, содержащей от 5 до 63 символов.
  14. Если Вам нужен графический интерфейс, то ставьте Raspbian, но если интерфейс нужен для установки софта или Вы будите пользоваться им очень редко, то ставьте Ubuntu Server. Для установки операционной системы действуйте по мануалу на официальном сайте Малины. Вам понадобится монитор и другая периферия на этом этапе. В процессе первого запуска, скорее всего у Вас спросят какой пароль задать пользователю root, обязательно запомните этот пароль. Возможно на каких-то этапах установки Вам будут предлагать доставить софт или произвести начальную конфигурацию, если не знаете, что от Вас хотят или сомневаетесь, то оставляйте все по умолчанию. После того как все будет готово, Вас должно выбросить в интерфейс операционной системы. Я пойду по более сложному пути и буду думать, что графической оболочки у Вас нет, а, следовательно, Вас встретит черный экран с предложением ввести логин и пароль. Для Raspbian по умолчанию используется логин "pi" пароль "raspbery", для Ubuntu Server логин и пароль "ubuntu", возможно первая буква заглавная. Нам необходимо выяснить какой ip адрес был присвоен малине в домашней сети (если Вы не задали его руками при установке системы). Сделать это можно следующей командой. ifconfig Если в выводе слишком много информации и Вы теряетесь, то можно убрать лишнее и оставить только данные по локальной сети. Для этого ведите следующую команду в которой укажите первые два октета Вашей сети. Скорее всего это 192.168 ifconfig | grep 192.168 Теперь Вы должны явно видеть выделенный малине ip адрес, мы будем использовать его для подключения с других устройств. Но если Вы используете DHCP, то данный ip рано или поздно изменится. Самым лучшим вариантом будет зайти в панель управления домашнего маршрутизатора и в настройках DHCP сервера закрепить за малиной данный адрес. Можно указать статический ip в самой малине, но тогда можно поиметь горя в будущем, в общем, сетью должен управлять Ваш маршрутизатор, а не рядовые хосты. Теперь мы можем подключиться к будущему серверу с домашнего компьютера. Подключаемся по SSH с помощью любого удобного клиента, например Putty. https://www.raspberrypi.org/documentation/remote-access/ssh/windows.md Авторизуемся под встроенным пользователем о котором упоминалось ранее и под которым Вы уже заходили на предыдущем этапе. Повышаем себе привилегии до root sudo su Обновляем информацию об актуальных пакетах apt update Обновляем имеющиеся в системе пакеты apt upgrade Устанавливаем MQTT брокер apt install mosquitto Брокер должен начать работать сразу после установки на начальной конфигурации, для домашнего сервера это вполне достаточно. Проверить статус брокера можно так: /etc/init.d/mosquitto status Теперь пора поднять web сервер, для дома прекрасно подойдет Apache, а заодно сразу поставим PHP и модуль позволяющий добавить в Apache поддержку .php скриптов. apt install apache2 php libapache2-mod-php Перезапустим web сервер /etc/init.d/apache2 restart Для установки MySQL сервера выполните следующую команду apt install mysql-server Во время установки СУБД Вас попросят задать пароль для основного пользователя root, этот пользователь не связан с ОС (просто одинаковые имена) и пароль распространяется только на MySQL. Для удобства можно указать тот же пароль, что используется системным пользователем root, это противоречит политике безопасности, но для теста малины вполне сойдет. Также я очень советую доставить Midnight Commander дабы чувствовать себя человеком при навигации по каталогам системы apt install mc Главное помните, что Midnight Commander обладает правами того пользователя, из-под которого запущен. В связи с этим, если вы захотите редактировать файлы конфигурации или выполнять иные задачи требующие права пользователя root, то всегда запускайте mc через sudo. sudo mc Вас попросят ввести пароль пользователя root и mc запустится от его имени. Для теста можно перейти в домашний каталог web сервера и создать там тестовый php скрипт. Все это сделать можно через Midnight Commander или выполнив следуюoe. командe echo "<?PHP phpinfo();" > /var/www/html/test.php Теперь перейдите в браузере на страницу http://server_ip/test.php чтобы убедиться в работоспособности. Кажется, на этом все, возможно я что-то пропустил, но это уже мелочи. Если будет нужно, то могу в свободно время снять видео как все это развернуть, вроде где-то валялась Raspbery PI 2. В общем пишите если будут вопросы. Если не определитесь какую систему сбора и анализа данных использовать, то можете посмотреть в сторону Zabbix. Тут на форуме есть тема, в которой описано как подружить MQTT и Zabbix. Но в любом случае, для начала посмотрите на другие системы, все-таки Zabbix это серверное решение и очень плотно работает с MySQL, что не очень хорошо для флешки которая используется малиной. Также можно использовать какой ни-ть HDD или SSD формата 2.5 дюйма место SD карты, но это уже отдельная история.
  15. @post125 Менее каменистый путь это брокер на Малине или любой аналогичный вариант. Ставите Ubuntu и одной командой устанавливаете MQTT сервер. В таком случает получаете полноценную систему с возможностью наращивать функционал, в том числе и запись данных куда угодно, хоть в СУБД. Заодно можно систему визуализации добавить, да и вообще, что угодно. А маршрутизатор, как не крути, это чисто транспортный узел со всеми вытекающими ограничениями по железу, а следовательно, и по программной части. Думаю, что вариант с OpenWRT + MQTT больше подойдет для удаленных систем, например, гаража.
  16. Контроллер жалуется, что не может найти точку доступа, но находит её при сканировании эфира. К сожалению, нет какой-либо дополнительной информации чтобы можно было явно указать на проблему. Попробуйте следующее: попытайтесь подключиться к другой точке доступа, например, можно использовать в этой роли сотовый телефон попробуйте сменить имя своей точки доступа, на время теста И еще, очень важное замечание. Обязательно перейдите на WAP2-PSK шифрование. И не скидывайте в открытый доступ скриншоты с настройками доступа к Вашему маршрутизатору. В данном случае, Вы засветили свой ключ доступа, и его обязательно нужно менять.
  17. т.к нет обратной связи для понимания когда, в данных режимах, необходимо подавать сигнал на открытие механического замка, то стоит отказаться от попытки удержания замка в открытом состоянии. Как-никак это делалось для электромагнита. В Вашем случае открыть дверь можно как мастер ключом после выхода из режима программирования, так и любым из добавленных ключей. В случае отсутствия ключей в памяти, что может быть только при монтаже замка, проблема решается при добавлении первого ключа, что опять же делается при монтаже замка. Так что просто удалите код отвечающий за управление замком в этих режимах и попробуйте на практике как это будет работать.
  18. Да, удалите в основном файле строку cron.add(cron::time_1m, [&](){ sensors.checkLine(); }, true); Всю инициализацию датчиков проведите самостоятельно без использования соответствующих функций при описании датчиков. Или замените указанную выше строку на разовый вызов метода checkLine sensors.checkLine(); Проверка датчиков на шине проводится через определение доступности адресов датчиков Wire.beginTransmission(sensor->address); /* ... */ sensor->status = (Wire.endTransmission() == 0); Скорее всего вы получаете не все данные при запросе данных для комплексного суточного графика в следствии чего json строка считается поврежденной и график не строится. А корень проблемы в том, что в данном случае контроллер передает данные по всем сенсорам, для которых активно ведение лога. Связано это с тем, что изначально не было графиков по конкретным сенсорам, существовал только комплексный график, соответственно и данные отдавались все и сразу. Получается, что Вы нашли придел для размера передаваемого объекта с данными. Часть кода уже переписана и прекрасно работает, но есть технические нюансы, из-за которых я не могу назвать какие-то конкретные строки. Ну и опять же, все приходится делать только в свободное время.
  19. Проверьте повторно этот код, возможно, как раз в конце суток, Вы пытаетесь писать за приделы массива, что приводит к краху. Работа index.htm не может влиять на микроконтроллер т.к весь код этого файла обрабатывается на стороне браузера. Контроллер принимает участие только в ответах на запросы API. По умолчанию контроллер полностью работает в автономном режиме, без подключения WiFi, сторонних сервисов, и внешней логики. Ставим в гараж, накидываем датчики, нагрузку, двигаем ползунки в web интерфейсе и в бой. Все остальное уже последствия различной степени интеграции в пользовательскую кухню. На данном этапе существования проекта можно легко добавить практически все идеи, описываемые в этой теме.
  20. А вот это оставлять так не стоит. Не пытались выяснить через какое время после старта это происходит или после какого события? Обратите внимание, что в Serial монитор периодически сбрасываются данные о свободной памяти (freeHeap), это значение не должно постоянно уменьшаться. Обычно, после выполнения всех заданий в планировщике это значение останавливается на фиксированной отметке и больше не уменьшается. Также для отслеживания памяти через web можно добавить программный сенсор. sensors.add(new knob_t(0, 81920, "1", "RAM", "Byte"), device::in, "ram", [&](){ return 81920 - ESP.getFreeHeap(); }); Это первое с чего стоит начать. Также можете отключать разные части кода и смотреть на работу в течении суток. И помните, что в ESP8266 v2.4.1 для Arduino IDE имеется утечка памяти. Я рекомендую использовать 2.4.2. В общем, не бросайте эту ситуацию. Лично я использую Zabbix, передача данных идет через MQTT. Но думаю, что стоит отталкиваться от того, на чем вы можете развернуть какую-либо систему мониторинга. Эту возможность планировалось реализовать в текущей версии, но выход ESP32 перебросил свободное время хобби микроконтроллеров на себя. Это вполне возможно в будущем.
  21. Вы вполне можете так сделать, но, как и нулевые значения, это по-прежнему будут не настоящие данные на протяжении всего начального суточного графика. Тут уже кому, что больше нравится. Лично для меня логически более понятно, когда данные начинают поступать от 0, особенно если отображаются несколько источников информации, я сразу вижу, когда контроллер начал его наполнение. Если требуется более детально рассмотреть динамику изменений и при этом исключить влияние на масштабирование со стороны пограничных значений, я использую выделение нужной области, в народе просто зуммирование. Выделение интересующей нас области Результат зуммирования Смотрите как будет удобнее лично Вам.
  22. Проблема действительно была в коде и искать её нужно в файле sensors.h, нас интересует метод clear String sensors::clear(float value) { if ((int)value == 0) return "0"; else if (value - (int)value == 0) return String((int)value); else return String(value); } Данный код используется для обрезания лишних нулей после запятой при формировании json объекта с данными журнала для их передачи в web интерфейс. Смысл был в уменьшении размера передаваемого объекта. Проблема крылась в самой первой проверке, её не смогут пройти числа меньше от -1 до 1 из-за приведения их к целому значению. Замените его на более явный и понятный код String sensors::clear(float value) { if (value == 0) return "0"; if (value - (int16_t)value == 0) return String((int16_t)value); return String(value); } Вот пример его работы на числах близких к нулю 0.000000, 0 0.123400, 0.12 0.500000, 0.50 -0.900000, -0.90 1.000000, 1 0.987654, 0.99 9.000000, 9 Есть другие, более лучшие способы обработки, постараюсь их рассмотреть в следующей версии проекта, но есть вероятность, что в этой части кода не будет нужды. Уверяю Вас, что все графики масштабируются одинаково. Все дело в том, что: Фактические показания барометра всегда находятся в узкой полосе по оси Y относительно лимитов этой оси График автоматически масштабируется под минимальные и максимальные значения переданных ему данных Изначально массив с данными для графика заполнен нулевыми значениями во всем своем диапазоне Получается, что пока все нулевые значения не будут вытеснены из массива, а это первые 24 часа работы, график будет строиться с учетом этих значений т.к для него это живые данные, хотя в человеческом понимании (появление значений по оси Y отличных от 0) это дает ошибочное чувство, что график только начал наполняться и визуально изменяться. Даже если Вы перезапустите контроллер и откроете график до того, как контроллер начнет перезапись массива с логами, Вы увидите полностью построенный график, но просто все значения по оси Y равны 0. Обратите внимание, как выглядит график давления после первых суток работы. Вот пример на сайте разработчика плагина, который демонстрирует динамическое обновление данных на графике, но, если Вы уделите просмотру этого демо чуть больше времени, то заметите как происходит автоматическое масштабирование графика по шкале Y в зависимости от минимального и максимального значения в переданном массиве данных. https://www.highcharts.com/demo/live-data Возможно в API можно найти свойство, отвечающее за выход значения по оси Y за определенные лимиты и регулирующее поведение графика, но это мои догадки, ответ нужно искать в документации от Highcharts.
  23. @init.d У Вас ряд явных проблем при работе с массивом. for (int i = 0; i < KEYS; i++) kst = DIS; // состояние датчика задается константой Место присвоения значения конкретному элементу массива, тут произведена попытка записать в память, где находится весь массив, значение константы DIS. Место этого код должен был выглядеть следующим образом. for (int i = 0; i < KEYS; i++) kst[i] = DIS; Вообще два одинаковых цикла for можно и одним заменить, но если честно, то наполнение массива kst не имеет смысла т.к массив был заполнен при объявлении, просто заполните его нужными значениями. boolean kst[KEYS] = {0}; Та же самая ситуация происходит и в цикле loop - двойные циклы и попытка перезаписать память, значениями которых там быть не должно. for (int i = 0; i < 3; i++) kst = ex1.digitalRead(i); for (int i = 0; i < 3; i++){ if (kst[i] == ACT) { // если хотя бы 1 ключ активен String state = "Active"; Serial.println (state); } } А хотели, видимо, получить это bool state = false; // Водим bool переменную чтобы не спамить Active в Serial если более одного порта активно for (int i = 0; i < KEYS; i++) { // Все делаем в одном цикле с лимитом указанным к KEYS kst[i] = ex1.digitalRead(i); // Пишем состояние порта в соответствующий элемент массива if (kst[i] == ACT) state = true; // Меняем состояние если хотябы один порт в состоянии ACT } if (state) Serial.println("Active"); // Сообщаем о состоянии ACT Ну и все в таком духе. А если по нормальному, то надо использовать прерывания, PCF8575 их поддерживает. Но опять вопрос, зачем нужен этот расширитель портов, если не задействованы стандартные порты микроконтроллера (в данном случае UNO). Выкидываем PCF и используем прерывания на стандартных портах. Почитать о них можно тут https://www.arduino.cc/reference/en/language/functions/external-interrupts/attachinterrupt/ И в таком случае код будет выглядеть совсем иначе, а работать намного быстрее и стабильнее. Единственный минус, это всего два порта поддерживающих прерывания у UNO.
  24. Доброе время суток. Не совсем понятно, эти проблемы и на локальных графиках тоже или только на народном мониторинге? Если на локальных тоже провалы, значит вы в течении какого-то времени получаете такие данные т.к на наполнение медианного фильтра требуется время, а это не менее трех запросов данных (по умолчанию размер буфера фильтра равен пяти). Тут не все так просто, гарантировать целостность данных нельзя т.к никаких проверок попросту нет. Наличие данных, похожих на настоящие, в одном элементе массива не дает гарантии, что в других элементах все также хорошо. Для повышения надежности используется механизм проверки контрольной суммы, алгоритм выбирается в зависимости от конкретной ситуации. Например, мы отправляем 1 байт полезной нагрузки, пусть это будет чисто 123 или ‭01111011‬ в двоичной системе счисления, следом за ним отправляем еще 1 байт который будет представлять из себя контрольную сумму. Содержимое второго байта зависит от алгоритма, который мы выбираем, пусть это будет битовая операция "И" (AND) и по завершению расчетов мы хотим видеть 0, что будет сигналом о корректности данных. Таким образом второй байт должен содержать такую последовательность бит, чтобы в ходе операции получить значение 0. В нашем примере, второй бит должен содержать число 132 или ‭10000100‬ в двоичной системе счисления. На принимающей стороне производим расчет: 01111011 & 10000100 = 00000000 Также доступны и другие битовые операции https://ru.wikipedia.org/wiki/Битовые_операции Для каждой конкретной задачи требуется свой алгоритм, конечно не факт, что это будут именно битовые операции. Используется только медианный фильтр. Чтобы отключить его, перейдите в файл sensors.h и найдите строку medianFilter_t lastDimension; Замените её на float lastDimension; Медианный фильтр использует float для хранения значений, на практике он нужен только чтобы сгладит пиковые выбросы, спровоцированные кратковременными внешними изменениями в среде в которой находятся сенсоры. В последних исходниках проекта имеются только сжатые файлы, но поддерживаются оба варианта. Приоритет отдается сжатому файлу т.к его транспортировка требует меньше времени. Вы можете удалить архивную копию, но предварительно убедитесь, что на контроллере имеется его распакованный вариант, иначе могут быть небольшие трудности. Явного округления в коде не предусматривалось, давайте искать. Для начала, стоит посмотреть на данные передаваемые от контроллера в web интерфейс. Сделать это можно пройдя по ссылке http://espws.local/api/sensors/log (espws.local стоит заменить на ip адрес если пытаетесь открыть ссылку с устройства не поддерживающего mDNS или имеющего проблемы при работе с зоной .local). Еще более наглядный вариант, это зайти на главную страницы web сервера микроконтроллера, открыть режим разработчика в браузере (раздел network), далее открыть график и поймать запрос на страницу /api/sensors/log. В окне ответа сервера ищем вкладку с предварительным просмотром, там будет разобранный json ответ с данными. Просмотрите эти данные и попытайтесь определить, получаете ли Вы эти значения уже округленными или это происходит во время создания самого графика. На народном мониторинге используется тот же самый плагин для создания графиков, что и в проекте метеостанции. Вы можете изменять визуализацию как Вам больше нравится, но в приделах возможностей самого плагина. Вот ссылка на сайт разработчика https://www.highcharts.com/ они предоставляют очень наглядные демо и всю документацию для создания собственной визуализации данных, это могут быть не только графики. Постараюсь в ближайшее время воссоздать Вашу проблему с округлением, но с разовыми пропажами данных все сложнее т.к это касается непосредственно Вашего кода.
  25. Скорее всего Вы достигли придела в шаге шкалы для библиотеки Knob. Возможно Вам подойдет иной вариант отображения состояния батареи, например, в процентах отображающих полезную емкость аккумулятора, только не от 0 до maxV, а от минимально напряжения питания контроллера до maxV. Например, для ESP8266 эти границы будут от 2.2V до 3.6V, что дает нам (3.6V-2.2V)/100 = 0.014V на 1% шкалы сенсора. В таком случае, когда мы видим, что осталось 5% полезной емкости аккумулятора, приходит точное понимание, что аккумулятор нужно заменить. Возможно Вы найдете более информативный способ визуализировать изменение тысячных долей. На всякий случай скину ссылки на сам плагин Knob: Примеры: http://anthonyterrien.com/demo/knob/ Исходники: https://github.com/aterrien/jQuery-Knob
×
×
  • Create New...