Официальный сайт компании Arduino по адресу arduino.cc
Термостат OpenTherm на ESP8266
- Войдите или зарегистрируйтесь, чтобы получить возможность отправлять комментарии
Сб, 20/04/2019 - 16:43
Приветствую. Делаю термостат для своего газового котла BAXI SLiM, работающий по протоколу OpenTherm. Всё вроде работает. Но веб страница долго грузится. Может есть какие замечения? Как ускорить процесс загрузки? С MQTT проблем нет.
#include <ESP8266WiFi.h> #include <PubSubClient.h> #include <WiFiClient.h> #include <ESP8266WebServer.h> #include <ESP8266mDNS.h> #include <OneWire.h> #include <DallasTemperature.h> #include <OpenTherm.h> //https://github.com/ihormelnyk/opentherm_library/ //OpenTherm input and output wires connected to 4 and 5 pins on the OpenTherm Shield const int inPin = 4; //D2 const int outPin = 5; //D1 #define ONE_WIRE_BUS 14 //D5 Data wire is connected to 14 pin on the OpenTherm Shield #define BUILTIN_LED 2 //D4 Встроенный LED const char* ssid = "*******"; const char* password = "**********"; const char* mqtt_server = "***********"; const int mqtt_port = 12345; const char* mqtt_user = "*********"; const char* mqtt_password = "********"; ESP8266WebServer server(80); //Server on port 80 OneWire oneWire(ONE_WIRE_BUS); DallasTemperature sensors(&oneWire); OpenTherm ot(inPin, outPin); WiFiClient espClient; PubSubClient client(espClient); char buf[50]; float sp = 20.00, //set point pv = 0, //current temperature pv_last = 0, //prior temperature ierr = 0, //integral error dt = 0, //time between measurements op = 0; //PID controller output unsigned long ts = 0, new_ts = 0; //timestamp bool enableCentralHeating = true; bool enableHotWater = true; bool enableCooling = false; int timer_off = 180; //Задержка выключения реле насоса СО 3 мин. const char HTTP_HTML[] PROGMEM = "<!DOCTYPE html>\ <html>\ <head>\ <link rel=\"icon\" type=\"image/jpg\" href=\"\">\ <title>Термостат</title>\ <meta charset=\"utf-8\">\ <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">\ <script>\ window.setInterval(\"update()\", 2000);\ function update(a,b,c,d,e,f,g,h){\ var a = new XMLHttpRequest();\ a.open(\"GET\", \"/temp\", true);\ a.onreadystatechange = function() {\ if (a.readyState != XMLHttpRequest.DONE || a.status != 200) return;\ document.getElementById('temp').innerHTML = a.responseText;\ };\ a.send();\ var b = new XMLHttpRequest();\ b.open(\"GET\", \"/boilertemp\", true);\ b.onreadystatechange = function() {\ if (b.readyState != XMLHttpRequest.DONE || b.status != 200) return;\ document.getElementById('boilertemp').innerHTML = b.responseText;\ };\ b.send();\ var c = new XMLHttpRequest();\ c.open(\"GET\", \"/dhwttemp\", true);\ c.onreadystatechange = function() {\ if (c.readyState != XMLHttpRequest.DONE || c.status != 200) return;\ document.getElementById('dhwttemp').innerHTML = c.responseText;\ };\ c.send();\ var d = new XMLHttpRequest();\ d.open(\"GET\", \"/outsidetemp\", true);\ d.onreadystatechange = function() {\ if (d.readyState != XMLHttpRequest.DONE || d.status != 200) return;\ document.getElementById('outsidetemp').innerHTML = d.responseText;\ };\ d.send();\ var e = new XMLHttpRequest();\ e.open(\"GET\", \"/chon\", true);\ e.onreadystatechange = function() {\ if (e.readyState != XMLHttpRequest.DONE || e.status != 200) return;\ document.getElementById('chon').innerHTML = e.responseText;\ };\ e.send();\ var f = new XMLHttpRequest();\ f.open(\"GET\", \"/hwon\", true);\ f.onreadystatechange = function() {\ if (f.readyState != XMLHttpRequest.DONE || f.status != 200) return;\ document.getElementById('hwon').innerHTML = f.responseText;\ };\ f.send();\ var g = new XMLHttpRequest();\ g.open(\"GET\", \"/flameon\", true);\ g.onreadystatechange = function() {\ if (g.readyState != XMLHttpRequest.DONE || g.status != 200) return;\ document.getElementById('flameon').innerHTML = g.responseText;\ };\ g.send();\ var h = new XMLHttpRequest();\ h.open(\"GET\", \"/getadc\", true);\ h.onreadystatechange = function() {\ if (h.readyState != XMLHttpRequest.DONE || h.status != 200) return;\ document.getElementById('getadc').innerHTML = h.responseText;\ };\ h.send();\ }\ </script>\ </head>\ <body style=\"text-align:center\">\ <div style=\"line-height:0.5\">\ <h1><font color=\"#1c6b72\"size=\"5\"face=\"Verdana\">Термостат</font></h1>\ <img src=\"\"><p>\ <font size=\"3\"face=\"Verdana\">OpenTherm</font><p>\ <hr align=\"center\" size=\"3\" width=\"290px\" color=\"#1c6b72\"><br/><br/>\ <font color=\"#1c6b72\"size=\"8\"face=\"Verdana\"><span id=\"temp\">{0}</span>°C</font><p><br/><br/>\ <font size=\"4\"face=\"Verdana\">Котёл, подача:  <span id=\"boilertemp\">{2}</span>°C</font><p><br/>\ <font size=\"4\"face=\"Verdana\">Горячая вода:   <span id=\"dhwttemp\">{3}</span>°C</font><p><br/>\ <font size=\"4\"face=\"Verdana\">На улице:   <span id=\"outsidetemp\">{4}</span>°C</font><p><br/>\ <font size=\"4\"face=\"Verdana\">Модуляция:   <span id=\"getadc\">{8}</span>%</font><p>\ <hr align=\"center\" size=\"3\" width=\"290px\" color=\"#1c6b72\"><br/>\ <form method=\"post\">\ <font size=\"4\"face=\"Verdana\">Значение уставки,°C: <b><input type=\"number\" min=\"10\" max=\"30\" step=\"0.1\" name=\"sp\" value=\"{1}\" style=\"font-size:17px; width:65px; background-color:#E1E1E1\"></b></font><br/><br/><br/>\ <input type=\"submit\"value=\"Записать\" style=\"font-size:16px; width:100px\">\ <form>\ </p>\ <hr align=\"center\" size=\"3\" width=\"290px\" color=\"#1c6b72\"><br/>\ <font size=\"3\"face=\"Verdana\">Статус СО:   <b><span id=\"chon\">{5}</span></b></font><p><br/>\ <font size=\"3\"face=\"Verdana\">Статус ГВС:   <b><span id=\"hwon\">{6}</span></b></font><p><br/>\ <font size=\"3\"face=\"Verdana\">Статус горелки:  <b><span id=\"flameon\">{7}</span></b></font><p>\ </div>\ </body>\ </html>"; void handleInterrupt() { ot.handleInterrupt(); } float getTemp() { return sensors.getTempCByIndex(0); } float getBoilerTemp() { return ot.getBoilerTemperature(); } float getDHWTemp() { unsigned long request26 = ot.buildRequest(OpenThermRequestType::READ, OpenThermMessageID::Tdhw, 0); unsigned long respons26 = ot.sendRequest(request26); uint16_t dataValue26 = respons26 & 0xFFFF; float result26 = dataValue26 / 256; return result26; } float getOutsideTemp() { unsigned long request27 = ot.buildRequest(OpenThermRequestType::READ, OpenThermMessageID::Toutside, 0); unsigned long respons27 = ot.sendRequest(request27); uint16_t dataValue27 = respons27 & 0xFFFF; if (dataValue27 > 32768) { //negative float result27 = -(65536 - dataValue27) / 256; return result27; } else { //positive float result27 = dataValue27 / 256; return result27; } } int getADC() { delay(100); int ar = analogRead(A0); ar = map(ar, 26, 1023, 0, 100); return ar; } unsigned long getCentralHeatingEnabled() { unsigned long response = ot.setBoilerStatus(enableCentralHeating, enableHotWater, enableCooling); OpenThermResponseStatus responseStatus = ot.getLastResponseStatus(); if (responseStatus = OpenThermResponseStatus::SUCCESS) return ot.isCentralHeatingEnabled(response); } unsigned long getHotWaterEnabled() { unsigned long response = ot.setBoilerStatus(enableCentralHeating, enableHotWater, enableCooling); OpenThermResponseStatus responseStatus = ot.getLastResponseStatus(); if (responseStatus = OpenThermResponseStatus::SUCCESS) return ot.isHotWaterEnabled(response); } unsigned long getFlameOn() { unsigned long response = ot.setBoilerStatus(enableCentralHeating, enableHotWater, enableCooling); OpenThermResponseStatus responseStatus = ot.getLastResponseStatus(); if (responseStatus = OpenThermResponseStatus::SUCCESS) return ot.isFlameOn(response); } float pid(float sp, float pv, float pv_last, float& ierr, float dt) { float Kc = 10.0; // K / %Heater float tauI = 50.0; // sec float tauD = 1.0; // sec // PID coefficients float KP = Kc; float KI = Kc / tauI; float KD = Kc * tauD; // upper and lower bounds on heater level float ophi = 100; float oplo = 0; // calculate the error float error = sp - pv; // calculate the integral error ierr = ierr + KI * error * dt; // calculate the measurement derivative float dpv = (pv - pv_last) / dt; // calculate the PID output float P = KP * error; //proportional contribution float I = ierr; //integral contribution float D = -KD * dpv; //derivative contribution float op = P + I + D; // implement anti-reset windup if ((op < oplo) || (op > ophi)) { I = I - KI * error * dt; // clip output op = max(oplo, min(ophi, op)); } ierr = I; Serial.println("Заданное значение температуры в помещкнии = " + String(sp) + " °C"); Serial.println("Текущее значение температуры в помещкнии = " + String(pv) + " °C"); Serial.println("Выхов ПИД регулятора = " + String(op)); Serial.println("Время между измерениями = " + String(dt) + "; ПИД коэффициенты: П = " + String(P) + "; И = " + String(I) + "; Д = " + String(D)); return op; } //=============================================================== // Эта процедура выполняется при открытии IP-адреса в браузере //=============================================================== void handleRoot() { digitalWrite(BUILTIN_LED, 1); if (server.method() == HTTP_POST) { for (uint8_t i = 0; i < server.args(); i++) { if (server.argName(i) == "sp") { sp = server.arg(i).toFloat(); } } } String page = FPSTR(HTTP_HTML); page.replace("{0}", String(getTemp())); page.replace("{1}", String((float)sp)); page.replace("{2}", String(getBoilerTemp())); page.replace("{3}", String(getDHWTemp())); page.replace("{4}", String(getOutsideTemp())); page.replace("{5}", String(getCentralHeatingEnabled() ? "on" : "off")); page.replace("{6}", String(getHotWaterEnabled() ? "on" : "off")); page.replace("{7}", String(getFlameOn() ? "on" : "off")); page.replace("{8}", String(getADC())); server.send(200, "text/html", page); digitalWrite(BUILTIN_LED, 0); } void handleGetTemp() { //digitalWrite(BUILTIN_LED, 1); server.send(200, "text/plain", String(getTemp())); //digitalWrite(BUILTIN_LED, 0); } void handleGetBoilerTemp() { //digitalWrite(BUILTIN_LED, 1); server.send(200, "text/plain", String(getBoilerTemp())); //digitalWrite(BUILTIN_LED, 0); } void handleGetDHWTemp() { //digitalWrite(BUILTIN_LED, 1); server.send(200, "text/plain", String(getDHWTemp())); //digitalWrite(BUILTIN_LED, 0); } void handleGetOutsideTemp() { //digitalWrite(BUILTIN_LED, 1); server.send(200, "text/plain", String(getOutsideTemp())); //digitalWrite(BUILTIN_LED, 0); } void handleGetCentralHeatingEnabled() { //digitalWrite(BUILTIN_LED, 1); server.send(200, "text/plain", String(getCentralHeatingEnabled() ? "on" : "off")); //digitalWrite(BUILTIN_LED, 0); } void handleGetHotWaterEnabled() { //digitalWrite(BUILTIN_LED, 1); server.send(200, "text/plain", String(getHotWaterEnabled() ? "on" : "off")); //digitalWrite(BUILTIN_LED, 0); } void handleGetFlameOn() { //digitalWrite(BUILTIN_LED, 1); server.send(200, "text/plain", String(getFlameOn() ? "on" : "off")); //digitalWrite(BUILTIN_LED, 0); } void handleADC() { //digitalWrite(BUILTIN_LED, 1); server.send(200, "text/plain", String(getADC())); //digitalWrite(BUILTIN_LED, 0); } //============================================================== // SETUP //============================================================== void setup_wifi() { delay(10); //Connect to Wi-Fi Network Serial.println(); Serial.print("Connecting to "); Serial.println(ssid); WiFi.mode(WIFI_STA); WiFi.begin(ssid, password); //Connect to your Wi-Fi router // Wait for connection while (WiFi.status() != WL_CONNECTED) { delay(500); Serial.print("."); } //If connection successful show IP address in serial monitor Serial.println(""); Serial.println("WiFi connected"); Serial.println("IP address: "); Serial.println(WiFi.localIP()); } void setup(void) { pinMode(BUILTIN_LED, OUTPUT); digitalWrite(BUILTIN_LED, 0); Serial.begin(115200); setup_wifi(); if (MDNS.begin("thermostat")) { Serial.println("MDNS responder started"); } //Initialize Webserver server.on("/", handleRoot); // Ответ сервера на запрос главной страницы server.on("/temp", handleGetTemp); // Ответ сервера на запрос температуры в помещении server.on("/boilertemp", handleGetBoilerTemp); // Ответ сервера на запрос температуры подачи котла server.on("/dhwttemp", handleGetDHWTemp); // Ответ сервера на запрос температуры горячей воды server.on("/outsidetemp", handleGetOutsideTemp); // Ответ сервера на запрос уличной температуры server.on("/chon", handleGetCentralHeatingEnabled); // Ответ сервера на запрос состояния СО server.on("/hwon", handleGetHotWaterEnabled); // Ответ сервера на запрос состояния ГВС server.on("/flameon", handleGetFlameOn); // Ответ сервера на запрос состояния горелки server.on("/getadc", handleADC); //Reads ADC function server.begin(); Serial.println("HTTP server started"); //============================================================== // Init DS18B20 sensor //============================================================== sensors.begin(); sensors.requestTemperatures(); sensors.setWaitForConversion(false); //switch to async mode pv, pv_last = sensors.getTempCByIndex(0); ts = millis(); //============================================================== // Init OpenTherm Controller //============================================================== ot.begin(handleInterrupt); //============================================================== // Init MQTT Client //============================================================== client.setServer(mqtt_server, mqtt_port); client.setCallback(callback); } void publish_temperature() { Serial.println("MQTT, Current Room Temperature, °C = " + String(pv)); String(pv).toCharArray(buf, 50); client.publish("pv", buf); } void publish_boilertemp() { Serial.println("MQTT, CH Temperature, °C = " + String(getBoilerTemp())); String(getBoilerTemp()).toCharArray(buf, 50); client.publish("cht", buf); } void publish_dhwtemp() { Serial.println("MQTT, DHW Temperature, °C = " + String(getDHWTemp())); String(getDHWTemp()).toCharArray(buf, 50); client.publish("dhwt", buf); } void publish_outtemp() { Serial.println("MQTT, Outside Temperature, °C = " + String(getOutsideTemp())); String(getOutsideTemp()).toCharArray(buf, 50); client.publish("outt", buf); } void publish_statusCH() { Serial.println("MQTT, Status Central Heating = " + String(getCentralHeatingEnabled())); String(getCentralHeatingEnabled()).toCharArray(buf, 50); client.publish("sch", buf); } void publish_ADC() { Serial.println("MQTT, Relative Modulation Level, % = " + String(getADC())); String(getADC()).toCharArray(buf, 50); client.publish("adc", buf); } void publish_statusDWH() { Serial.println("MQTT, Status Hot Water = " + String(getHotWaterEnabled())); String(getHotWaterEnabled()).toCharArray(buf, 50); client.publish("sdwh", buf); } void publish_statusFlame() { Serial.println("MQTT, Status Central Heating = " + String(getFlameOn())); String(getFlameOn()).toCharArray(buf, 50); client.publish("sfl", buf); } void publish_setBoilerTemperature() { Serial.println("MQTT, PID Controller Output, % = Set Point CH Temperature, °C = " + String(op)); String(op).toCharArray(buf, 50); client.publish("op", buf); } void callback(char* topic, byte* payload, unsigned int length) { if (strcmp(topic, "sp") != 0) return; String str = String(); for (int i = 0; i < length; i++) { str += (char)payload[i]; } Serial.println("MQTT, Setpoint Room Temperature = " + str); sp = str.toFloat(); } void reconnect() { while (!client.connected()) { Serial.print("Attempting MQTT connection..."); // Attempt to connect if (client.connect("ESP8266Client", mqtt_user, mqtt_password)) { Serial.println("connected"); // Once connected, publish an announcement... publish_temperature(); publish_boilertemp(); publish_dhwtemp(); publish_outtemp(); publish_ADC(); publish_statusCH(); publish_statusDWH(); publish_statusFlame(); publish_setBoilerTemperature(); // ... and resubscribe client.subscribe("sp"); } else { Serial.print("failed, rc ="); Serial.print(client.state()); Serial.println(" try again in 5 seconds"); // Wait 5 seconds before retrying delay(5000); } } } //============================================================== // LOOP //============================================================== void loop(void) { new_ts = millis(); if (new_ts - ts > 1000) { unsigned long response = ot.setBoilerStatus(enableCentralHeating, enableHotWater, enableCooling); OpenThermResponseStatus responseStatus = ot.getLastResponseStatus(); if (responseStatus = OpenThermResponseStatus::SUCCESS) { //Serial.println("Error: Invalid boiler response " + String(response, HEX)); Serial.println("Текущий статус системы отопления: " + String(ot.isCentralHeatingEnabled(response) ? "on" : "off")); Serial.println("Текущий статус горячей воды: " + String(ot.isHotWaterEnabled(response) ? "on" : "off")); Serial.println("Текущий статус горелки: " + String(ot.isFlameOn(response) ? "on" : "off")); Serial.println("Индикация состояния неисправности: " + String(ot.isFault(response) ? "fault" : "no fault")); Serial.println("Диагностическая индикация: " + String(ot.isDiagnostic(response) ? "diagnostics" : "no diagnostics")); } if (responseStatus == OpenThermResponseStatus::NONE) { Serial.println("Error: OpenTherm is not initialized"); } else if (responseStatus == OpenThermResponseStatus::INVALID) { Serial.println("Error: Invalid response " + String(response, HEX)); } else if (responseStatus == OpenThermResponseStatus::TIMEOUT) { Serial.println("Error: Response timeout"); } pv = sensors.getTempCByIndex(0); dt = (new_ts - ts) / 1000.0; ts = new_ts; if (responseStatus == OpenThermResponseStatus::SUCCESS) { op = pid(sp, pv, pv_last, ierr, dt); //Set CH Temperature ot.setBoilerTemperature(op); pv_last = pv; sensors.requestTemperatures(); //async temperature request } //Print Temperature Serial.println("Текущая температура контура СО = " + String(getBoilerTemp()) + " °C"); Serial.println("Текущая температура контура ГВС = " + String(getDHWTemp()) + " °C"); Serial.println("Температура на улице = " + String(getOutsideTemp()) + " °C"); Serial.println("Текущий уровень модуляции горелки = " + String (getADC()) + " %"); Serial.println("ADC = " + String (analogRead(A0))); publish_temperature(); publish_boilertemp(); publish_dhwtemp(); publish_outtemp(); publish_ADC(); publish_statusCH(); publish_statusDWH(); publish_statusFlame(); publish_setBoilerTemperature(); } //MQTT Loop if (!client.connected()) { reconnect(); } client.loop(); server.handleClient(); //handle http requests }
Посмотрел по диагонали, возможно не прав, у вас получение температуры в самом выводе html странички.
Вообще все плохо, получение статусов котла температуры и прочих данных должно быть неблокирующим кодом по таймерам, а вывод в страничку из переменных а не динамически получать. Кстати зачем страничку в progmem? Неужели места в esp не хватило.
А в строке №241 эта самая страничка целиком закачивается в RAM
И на это тоже тратится драгоценное процессорное время
строка 223 и 224 орфографические ошибки, это, что бросилось в глаза
строка 223 и 224 орфографические ошибки, это, что бросилось в глаза
Спасибо, исправил.
Давно такого не видел, чтоб статику, по определению размещенную в PROGMEM, там ей и место, перегружать в стринг и в этом стринге реплейсить.
Для чего настраивать интервал, чтоб каждые 2 секунды асинхронно выполнялись куча запросов (броузер их параллельно ведь выполняет).
/temp /boilertemp /dhwttemp /outsidetemp /chon /hwon /flameon /getadc
ни ужели нельзя обойтись одним запросом, получать всё необходимое сразу.
Давно такого не видел, чтоб статику, по определению размещенную в PROGMEM, там ей и место, перегружать в стринг и в этом стринге реплейсить.
Для чего настраивать интервал, чтоб каждые 2 секунды асинхронно выполнялись куча запросов (броузер их параллельно ведь выполняет).
/temp /boilertemp /dhwttemp /outsidetemp /chon /hwon /flameon /getadc
ни ужели нельзя обойтись одним запросом, получать всё необходимое сразу.
Алексей., пробовал и одним запросом, как выше написали, но переменные не обновляются без перезагрузки страницы...
Что значит не обновляются? Есп их не отправляет? А по отдельности отправляет? Целую страницу отправляет, а десяток переменных не может? Да быть такого не может. Вы увеличиваете нагрузку, отправляя с броузера сразу несколько запросов, по одному для каждого параметра.
Приветствую. Делаю термостат для своего газового котла BAXI SLiM, работающий по протоколу OpenTherm. Всё вроде работает. Но веб страница долго грузится. Может есть какие замечения? Как ускорить процесс загрузки? С MQTT проблем нет.
Здравствуйте
Загрузил ваш скейтч
У меня котел не отключает помпу отопления. У Вас все нормально с этим
И возможно ли дописать скейтч для регулирования температуры горячей воды(можно только через mqtt)
Спасибо
VOVA_iS - не надо этот код брать, он откровенно плохо написан. Не шутите с газом.
VOVA_iS - не надо этот код брать, он откровенно плохо написан. Не шутите с газом.
Тут криво веб сервер написан
По работе котла код скопирован с примера библиотеки OpenTherm.
Mqtt работает хорошо
VOVA_iS, да, помпа молотит постоянно, повесил на термостат своё реле для её управления. Горячую воду то же сделал. Вообще много чего добавил и переделал, и продолжаю делать...
P.S. Да, сервер кривой :-(
VOVA_iS, от веб сервера решил отказаться, оставлю только конфигурационную страничку и всё, а для рулёжки достаточно mqtt.
VOVA_iS, да, помпа молотит постоянно, повесил на термостат своё реле для её управления. Горячую воду то же сделал. Вообще много чего добавил и переделал, и продолжаю делать...
P.S. Да, сервер кривой :-(
про помпу можно подробней:)?
скетч можно с горячей водой?
VOVA_iS, от веб сервера решил отказаться, оставлю только конфигурационную страничку и всё, а для рулёжки достаточно mqtt.
Да я согласен по MQTT управления выше крыши. И умный дом можно добавить полноценно
про помпу можно подробней:)?
скетч можно с горячей водой?
Для помпы использую библиотеку TimingRelay.h:
В setup:
В loop реле работает по условию:
С ГВС не так всё просто... Можно и по играться уставкой температуры ГВС. :-) При работе от внешнего термостата, а рекомендованный это QAA73, именно он задаёт режим работы ГВС, т.е. когда включить по таймеру (1 раз в неделю) прогрев системы до 65 гр., чтобы убить бацилы. При отсутствии внешнего это делает автоматика котла. Если в термостате этого нет, то ручками и по календарю. У меня пока не получилось автоматом, а может и получилось, полноценно не проверял, здесь, на форуме помогли.
Вот по порядку:
В reconnect добавить строку
В loop
А как параметр spdhw передаётся в котёл? не вижу
А как параметр spdhw передаётся в котёл? не вижу
Извиняюсь!
В loop после 466 строки (код в начале) добавить:
Ещё вырежу всю веб морду:-)
спасибо завтра попробую скомпилировать скетч
VOVA_iS, есть ещё одна задумка. В слимах по ОТ не передаётся уровень модуляции горелки. Можно сделать используя у ESP 8266 А0.
Интересная идея:-) только я на сколько помню на А0 может быть максимум 3.3v
надо сначала замеры сделать у котла:-)
VOVA_iS, вполне достаточно. С котла 9,10 контакты А5 с катушки модуляции идёт ШИМ. Надо ставить операционник, преобразовать ШИМ в линейные 0-5В, короче с него через делитель на А0. Программно потребуется ремапинг АЦП.
про помпу можно подробней:)?
скетч можно с горячей водой?
Для помпы использую библиотеку TimingRelay.h:
В setup:
В loop реле работает по условию:
Не могу скомпилировать идут ошибки
в коде данные переменные status_flame status_dhw не опредлены
Всё верно, статусы не задекларированы, вставьте это в loop выше индикации и состояния насоса:
Условия для индикации можно отключить, а то опять ругаться будет при компиляции...
я так понимаю нужно это
Заменить на это
Да, верно.
Скомпилировал код выше.
Увы температура горяей воды не передается.
Посмотрите где может быть ошибка?
ОК! Посмотрю. У меня сейчас проблема после обновления менеджера плат до версии 2.5.2. esp-шка ругается в мониторе "ISR not in IRAM!", ребутится постоянно. До обновления была версия 2.5.0. Копать дальше? Или откатиться и забыть? После декодирования стека пишет:
ОК! Посмотрю. У меня сейчас проблема после обновления менеджера плат до версии 2.5.2. esp-шка ругается в мониторе "ISR not in IRAM!", ребутится постоянно. До обновления была версия 2.5.0. Копать дальше? Или откатиться и забыть? После декодирования стека пишет:
Откатись и забудь.
У меня тоже после обновления много каких проектов перестало собираться.
Откатился и все норм
Это понятно, но всё же не случайно это, что то с прерываниями не то... Где то надо добавить ICACHE_RAM_ATTR перед функцией ISR, что бы правильно было.
Хотя ругани в сети по этому поводу полно...:-)
Так будет проще, выкладываю один из рабочих кодов без веб страницы, сервер из кода сами удалите и принты в монитор тоже, не нужны они для работы с MQTT.
P.S.
Так и есть, атрибут ICACHE_RAM_ATTR теперь обязателен в коде, функциях, вызываемых по прерываниям. Проблема работы с 2.5.2 разрешилась.
Проблему с помпой решил немного по другому
в секцию loop добавил
После добавления этого кода котёл по достижению заданной температуры выключает режим отопления и соответсвенно насос, через заданное время в настройке котла
Проблема с горячей водой так и осталось. Регулирование нету
даже в логах пишется что заданная температура 0
но если изменить код вот так
То заданная температура отображается(температура гвс установлена до подключение платы), но не устанавливается
VOVA_iS, с помпой понятно, а я то начал городить огород с доп. реле, простенько, сам бы не догадался...
Пишет 0, но только когда котёл не подключен к термостату, когда же котёл представится термостату и тот ответит ему, то всё ОК.
P.S.
Вообще библиотека ОТ неплохая, но попытка скрестить ужа и ежа провалилась, я про web и mqtt в одном флаконе. Жуткие тормоза при совместной работе двух библиотек, причём только web. Приглядываюсь к такому решению https://www.youtube.com/watch?v=QQ7apgKBgNM
VOVA_iS, с помпой понятно, а я то начал городить огород с доп. реле, простенько, сам бы не догадался...
Пишет 0, но только когда котёл не подключен к термостату, когда же котёл представится термостату и тот ответит ему, то всё ОК.
Котёл с термостатом представились друг другу.
А проблема что все данные с котла приходят кроме установленной температуры ГВС
да не нужен этот web
mqtt более чем достаточно
VOVA_iS, с помпой понятно, а я то начал городить огород с доп. реле, простенько, сам бы не догадался...
Пишет 0, но только когда котёл не подключен к термостату, когда же котёл представится термостату и тот ответит ему, то всё ОК.
Котёл с термостатом представились друг другу.
А проблема что все данные с котла приходят кроме установленной температуры ГВС
да не нужен этот web
mqtt более чем достаточно
А в mqtt брокере переменная spdhw видна?
но если изменить код вот так
1
float
setDHWTemp() {
2
unsigned
long
request56 = ot.buildRequest(OpenThermRequestType::WRITE_ACK, OpenThermMessageID::TdhwSet, hex56);
3
unsigned
long
respons56 = ot.sendRequest(request56);
4
uint16_t dataValue56 = respons56 & 0xFFFF;
5
float
result56 = dataValue56 / 256;
6
return
result56;
Тогда видна
В протоколе ОТ ID Msg 56 имеет | R W | DHW setpoint | f8.8 | 0..127 | Domestic hot water temperature setpoint (°C)
Можно прочитать, что записалось и вывести в монитор, отдельно я этого не делал, просто проверял ранее и всё было ОК. Точно не помню, как переменную обзывал...
VOVA_iS, а как определили, что в котёл не ушла уставка?
С этим кодом работы помпы ОТ отключается...т.е. интерфейс активируется только при этом условии, а если ОТ не активен, то целевую температуру контур ГВС не примет, стало быть, котёл работает в этом промежутке, в обычном режиме, ON/OFF (перемычка). Хотя, могу и ошибаться, погонять надо...
VOVA_iS, а как определили, что в котёл не ушла уставка?
Банально включаю воду и смотрю на сколько нагрел котел. Было 38 ставлю 45. Все равно греет до 38.
С этим кодом работы помпы ОТ отключается...т.е. интерфейс активируется только при этом условии, а если ОТ не активен, то целевую температуру контур ГВС не примет, стало быть, котёл работает в этом промежутке, в обычном режиме, ON/OFF (перемычка). Хотя, могу и ошибаться, погонять надо...
Это решение было найдено после проблемы с регулировкой температуры.
С параметром READ и WRITE_ACK установленную температуру читает.
Вообщем добавил свою переменную в MQTT
Она почему-то в консоли появляется один раз.
Лог с WRITE
VOVA_iS, а как определили, что в котёл не ушла уставка?
Банально включаю воду и смотрю на сколько нагрел котел. Было 38 ставлю 45. Все равно греет до 38.
Гистерезис большой, не зацикливаюсь. Я сейчас откопал свой рабочий код без WEB, внёс в него вашу правку помпы, проверил работает. Только обновите библиотеку ОТ. Там слово Enable заменено на Active (код правленный).
Ещё, в приложении (не знаю каким пользуетесь) целевую уставку температуры в помещении делайте не более 30 гр., а целевую уставку ГВС не менее 32 гр. Условие описано в этом куске кода, строка 367:
P.S. Кстати, у вас BAXI SLiM? В интерфейсной плате к LMU33 котла назначение реле какое? Есть предположения? Я интерфейсную плату сам делал, вместо реле повесил светодиод, для индикации, зажигается только при работе контура СО, на ГВС не активируется. При отключении СО, сразу гаснет.
Вообщем такая же фигня... Не проходит установка температуры.
Может проблема с версией термостата.
У вас что выдает в логах по этим параметрам. Думаю может он представиться нормально не может. Либо представился без регулировки ГВС.
И откуда можно узнать какие варианты есть представления для котла?
Ещё, в приложении (не знаю каким пользуетесь) целевую уставку температуры в помещении делайте не более 30 гр., а целевую уставку ГВС не менее 32 гр. Условие описано в этом куске кода, строка 367:
P.S. Кстати, у вас BAXI SLiM? В интерфейсной плате к LMU33 котла назначение реле какое? Есть предположения? Я интерфейсную плату сам делал, вместо реле повесил светодиод, для индикации, зажигается только при работе контура СО, на ГВС не активируется. При отключении СО, сразу гаснет.
У меня KOREASTAR Premium. В этом вопросе не помогу.
Загрузите мой последний код, посмотрим, что у вас. У меня котел представляется 1.51. если не ошибаюсь. просто я принты отключил.
Ещё, в приложении (не знаю каким пользуетесь) целевую уставку температуры в помещении делайте не более 30 гр., а целевую уставку ГВС не менее 32 гр. Условие описано в этом куске кода, строка 367:
P.S. Кстати, у вас BAXI SLiM? В интерфейсной плате к LMU33 котла назначение реле какое? Есть предположения? Я интерфейсную плату сам делал, вместо реле повесил светодиод, для индикации, зажигается только при работе контура СО, на ГВС не активируется. При отключении СО, сразу гаснет.
У меня KOREASTAR Premium. В этом вопросе не помогу.
Понял, тогда и с уровнем модуляции вам разбираться надо. Мой котёл по ОТ этот ID Msg "RelModLevel" не отдаёт, потому сам вычисляю, а не читаю его из котла.