From 740b3d3ad265f8b4594d96d86cc0a36507226c7c Mon Sep 17 00:00:00 2001 From: ym1026 <1539963735@qq.com> Date: Wed, 17 Sep 2025 15:08:13 +0800 Subject: [PATCH 1/5] =?UTF-8?q?=E5=BC=B9=E7=AA=97=E6=95=B0=E6=8D=AE?= =?UTF-8?q?=E5=88=B7=E6=96=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- web/src/components/Home/Modal.vue | 27 +++++++++++++++++++++------ web/src/views/sub/Home.vue | 1 - 2 files changed, 21 insertions(+), 7 deletions(-) diff --git a/web/src/components/Home/Modal.vue b/web/src/components/Home/Modal.vue index b913c7e..ecfcb49 100644 --- a/web/src/components/Home/Modal.vue +++ b/web/src/components/Home/Modal.vue @@ -60,6 +60,8 @@ export default { }, data() { return { + refreshInterval:null, + modalInfo: {}, list: [ { @@ -118,16 +120,29 @@ export default { return this.list.filter((_, index) => index % 2 !== 0).slice(0, 3) // 右列取前3个奇数索引 } }, + beforeUnmount() { + if(this.refreshInterval){ + clearInterval(this.refreshInterval) + this.refreshInterval=null + } + }, async mounted() { - await Promise.all([ - this.getStatTotalList(), - this.queryStationInfo(), - this.queryStationData(), - this.getStatDayList(1) - ]) + await this.loadAllData() + this.refreshInterval=setInterval(async()=>{ + await this.loadAllData() + },30000) //30s刷新一次 + }, methods: { + async loadAllData(){ + await Promise.all([ + this.getStatTotalList(), + this.queryStationInfo(), + this.queryStationData(), + this.getStatDayList(1) + ]) + }, // 查询系统累计统计信息 async getStatTotalList() { try { diff --git a/web/src/views/sub/Home.vue b/web/src/views/sub/Home.vue index 145918b..be6a01d 100644 --- a/web/src/views/sub/Home.vue +++ b/web/src/views/sub/Home.vue @@ -111,7 +111,6 @@ export default { } }, beforeUnmount() { - console.log('2222') if(this.refreshInterval){ clearInterval(this.refreshInterval) this.refreshInterval=null From 680be6155f9de79a058249e2b27b38b88a4b9a9c Mon Sep 17 00:00:00 2001 From: ym1026 <1539963735@qq.com> Date: Wed, 17 Sep 2025 15:18:32 +0800 Subject: [PATCH 2/5] =?UTF-8?q?=E5=9C=B0=E5=9B=BEzoom=E4=BF=AE=E6=94=B9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- web/src/components/Home/Map.vue | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/web/src/components/Home/Map.vue b/web/src/components/Home/Map.vue index 9acf054..86a8523 100644 --- a/web/src/components/Home/Map.vue +++ b/web/src/components/Home/Map.vue @@ -58,7 +58,7 @@ export default { data() { return { center: [112.870000,34.180000], // 默认中心点(河南) - zoom: 12, + zoom: 6, map: null, currentMarker: {}, showCtrModal: false, From 27ca787fc59876101588f970524f9c4121230dd1 Mon Sep 17 00:00:00 2001 From: ym1026 <1539963735@qq.com> Date: Wed, 17 Sep 2025 15:31:35 +0800 Subject: [PATCH 3/5] =?UTF-8?q?=E6=9C=8D=E5=8A=A1=E7=AE=A1=E7=90=86?= =?UTF-8?q?=E5=B8=83=E5=B1=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- web/src/components/Home/Alarm.vue | 8 +++++++- web/src/components/Home/Charge.vue | 1 - web/src/views/system/service.vue | 7 +++---- web/vue.config.js | 2 +- 4 files changed, 11 insertions(+), 7 deletions(-) diff --git a/web/src/components/Home/Alarm.vue b/web/src/components/Home/Alarm.vue index 1ac9f99..6b8da47 100644 --- a/web/src/components/Home/Alarm.vue +++ b/web/src/components/Home/Alarm.vue @@ -93,8 +93,11 @@ export default { }, mounted() {}, beforeUnmount() { - this.faultChart = null window.removeEventListener('resize', this.handleResize) + if (this.faultChart) { + this.faultChart.dispose() + this.faultChart = null + } }, methods: { @@ -130,6 +133,9 @@ export default { drawLineChart(activeKey) { this.getChargeData(activeKey) + if(this.faultChart){ + this.faultChart.dispose() + } const chartDom = document.getElementById('alarm-chart') let faultChart = this.$echarts.init(chartDom) this.faultChart = faultChart diff --git a/web/src/components/Home/Charge.vue b/web/src/components/Home/Charge.vue index 6d56dc0..a839e57 100644 --- a/web/src/components/Home/Charge.vue +++ b/web/src/components/Home/Charge.vue @@ -154,7 +154,6 @@ export default { }, drawLineChart(activeKey) { - console.log(this.$refs.charge) // const labelCount = Math.floor(500 / 30); this.getChargeData(activeKey) if(this.chargeChart){ diff --git a/web/src/views/system/service.vue b/web/src/views/system/service.vue index 18e5072..54d576f 100644 --- a/web/src/views/system/service.vue +++ b/web/src/views/system/service.vue @@ -58,7 +58,7 @@ export default { props: {}, data() { return { - tableData:[], + tableData: [], tableOption: { select: false @@ -243,10 +243,9 @@ export default { diff --git a/web/vue.config.js b/web/vue.config.js index 00aa3e2..56dc2c6 100644 --- a/web/vue.config.js +++ b/web/vue.config.js @@ -14,7 +14,7 @@ module.exports = defineConfig({ proxy: { '/api': { // 代理前缀,可以自定义(如 '/api') - target: 'http://192.168.0.187:19801', // 目标服务器地址 + target: 'http://192.168.0.187:19800', // 目标服务器地址 changeOrigin: true, // 是否改变请求源(跨域必备) pathRewrite: { '^/api': '' // 重写路径,去掉 '/api' 前缀 From 94d6d8a8dbd1411d792171e5a3a79d5bc0b83a47 Mon Sep 17 00:00:00 2001 From: ym1026 <1539963735@qq.com> Date: Wed, 17 Sep 2025 15:34:01 +0800 Subject: [PATCH 4/5] =?UTF-8?q?=E8=BF=90=E8=A1=8C=E4=BF=A1=E6=81=AF?= =?UTF-8?q?=E9=BB=98=E8=AE=A4=E6=95=B0=E6=8D=AE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- web/src/components/Home/Modal/OperationalInfo.vue | 2 +- web/src/components/Home/Operational.vue | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/web/src/components/Home/Modal/OperationalInfo.vue b/web/src/components/Home/Modal/OperationalInfo.vue index f931c4c..b9f6fc8 100644 --- a/web/src/components/Home/Modal/OperationalInfo.vue +++ b/web/src/components/Home/Modal/OperationalInfo.vue @@ -88,7 +88,7 @@ export default { handler(newVal, oldVal) { if (newVal !== oldVal) { this.list.forEach((item) => { - item.value = this.propsTotal[item.key] + item.value = this.propsTotal[item.key]||0 }) } }, diff --git a/web/src/components/Home/Operational.vue b/web/src/components/Home/Operational.vue index 06a651a..4617190 100644 --- a/web/src/components/Home/Operational.vue +++ b/web/src/components/Home/Operational.vue @@ -74,7 +74,7 @@ export default { const dates = data.map((item) => item.station_name) const values = [] keys.forEach((item, index) => { - values[index] = data.map((dataValue) => dataValue[keys[index]]) + values[index] = data.map((dataValue) => dataValue[keys[index]])||0 }) return { From c55da0bddcbfb00afb7d6270ef48b9b3b51f6917 Mon Sep 17 00:00:00 2001 From: lixiaoyuan Date: Wed, 17 Sep 2025 19:55:59 +0800 Subject: [PATCH 5/5] =?UTF-8?q?=E8=B0=83=E8=AF=95=E4=BF=AE=E6=94=B9MQTT?= =?UTF-8?q?=E9=80=9A=E8=AE=AF=E9=97=AE=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- bin/Release/assets/config/app.json | 32 +- bin/Release/assets/config/regaddrs.json | 20 ++ bin/Release/assets/config/regaddrsShow.json | 17 +- src/app/Application.cpp | 52 ++- src/app/Config.cpp | 82 ++++- src/app/Config.h | 43 ++- src/app/Device.cpp | 66 ++-- src/app/Device.h | 2 + src/app/Station.cpp | 261 +++++++++------ src/app/Station.h | 46 +-- src/common/Spdlogger.cpp | 8 +- src/database/Dao.cpp | 1 + src/database/DaoEntity.cpp | 11 +- src/main.cpp | 5 +- src/protocol/HttpEntity.cpp | 140 +++++--- src/protocol/HttpEntity.h | 1 + src/protocol/MqttEntity.cpp | 349 ++++++++------------ src/protocol/MqttEntity.h | 19 +- 18 files changed, 661 insertions(+), 494 deletions(-) diff --git a/bin/Release/assets/config/app.json b/bin/Release/assets/config/app.json index 97f616b..a27c91c 100644 --- a/bin/Release/assets/config/app.json +++ b/bin/Release/assets/config/app.json @@ -1,8 +1,34 @@ { + "debug":0, "launchdate": "2025-09-01", + "weburl": "http://127.0.0.1:19601/", "database": {"host": "localhost", "port": 3306, "user": "root", "passwd": "123456", "dbname": "ess"}, - "token":"", - "http": {"port": 19801}, + + "http": {"token":0, "port": 19801, "encryption":0, "encryptKey":""}, "mqtt": {"host":"mqtt://43.136.119.46:6203","username":"jsyhsec","password":"123456"}, - "weburl": "http://127.0.0.1:19601/" + "topic": { + "EMS_YX": {"deviceType":101, "polling":0, "enabled": 1}, + "EMS_YC": {"deviceType":101, "polling":0, "enabled": 1}, + "EMS_YT": {"deviceType":101, "polling":0, "enabled": 1}, + "PCS_YX": {"deviceType":102, "polling":1, "enabled": 1}, + "PCS_YC": {"deviceType":102, "polling":1, "enabled": 1}, + "PCU_YX": {"deviceType":103, "polling":0, "enabled": 1}, + "PCU_YC": {"deviceType":103, "polling":0, "enabled": 1}, + "BMS_YX": {"deviceType":104, "polling":0, "enabled": 1}, + "BMS_YC": {"deviceType":104, "polling":0, "enabled": 1}, + "BCU_YC": {"deviceType":105, "polling":1, "enabled": 1}, + "BCU_YX": {"deviceType":105, "polling":1, "enabled": 1}, + "MEM_YC": {"deviceType":3, "polling":0, "enabled": 1}, + "TH_YC": {"deviceType":10, "polling":1, "enabled": 1}, + "Fire40_YX": {"deviceType":7, "polling":1, "enabled": 1}, + "Cooling_YC": {"deviceType":14, "polling":1, "enabled": 1}, + "Cooling_YX": {"deviceType":14, "polling":1, "enabled": 1}, + "Gateway_YX": {"deviceType":15, "polling":1, "enabled": 1}, + "Gateway_YC": {"deviceType":15, "polling":1, "enabled": 1}, + "Charger_YC": {"deviceType":106, "polling":0, "enabled": 1} + }, + "view": {"latitude":0,"longitude":0,"altitude":0}, + "video": { + "1":{"host":"", "port":9000, "user":"", "passwd":""} + } } \ No newline at end of file diff --git a/bin/Release/assets/config/regaddrs.json b/bin/Release/assets/config/regaddrs.json index d56984d..2f7ce5a 100644 --- a/bin/Release/assets/config/regaddrs.json +++ b/bin/Release/assets/config/regaddrs.json @@ -866,5 +866,25 @@ {"key": "40021", "datatype": "uint16", "remark": "自定时间段"}, {"key": "40038", "datatype": "uint16", "remark": "其他参数"} ] + }, + "Charger_YC":{ + "addr":[ + {"key": "11", "datatype": "uint16", "remark": "枪1:状态"}, + {"key": "12", "datatype": "uint16", "remark": "枪1:需求电压"}, + {"key": "13", "datatype": "uint16", "remark": "枪1:需求电流"}, + {"key": "14", "datatype": "uint16", "remark": "枪1:需求功率"}, + {"key": "15", "datatype": "uint16", "remark": "枪1:输出电压"}, + {"key": "16", "datatype": "uint16", "remark": "枪1:输出电流"}, + {"key": "17", "datatype": "uint16", "remark": "枪1:输出功率"}, + {"key": "18", "datatype": "uint16", "remark": "枪1:功率限值"}, + {"key": "21", "datatype": "uint16", "remark": "枪2:状态"}, + {"key": "22", "datatype": "uint16", "remark": "枪2:需求电压"}, + {"key": "23", "datatype": "uint16", "remark": "枪2:需求电流"}, + {"key": "24", "datatype": "uint16", "remark": "枪2:需求功率"}, + {"key": "25", "datatype": "uint16", "remark": "枪2:输出电压"}, + {"key": "26", "datatype": "uint16", "remark": "枪2:输出电流"}, + {"key": "27", "datatype": "uint16", "remark": "枪2:输出功率"}, + {"key": "28", "datatype": "uint16", "remark": "枪2:功率限值"} + ] } } \ No newline at end of file diff --git a/bin/Release/assets/config/regaddrsShow.json b/bin/Release/assets/config/regaddrsShow.json index 5cf7f6b..94941fe 100644 --- a/bin/Release/assets/config/regaddrsShow.json +++ b/bin/Release/assets/config/regaddrsShow.json @@ -96,14 +96,15 @@ "Charger": { "deviceType":106, "addrYC":[ - ["需求电压", "31071", "0.0", " V"], - ["需求电流", "31073", "0.0", " A"], - ["需求功率", "31075", "0.0", " kW"], - ["功率限值", "31077", "0.0", " kW"], - ["输出电压", "31079", "0.0", " V"], - ["输出电流", "31081", "0.0", " A"], - ["输出功率", "31083", "0.0", " kW"] + ["工作状态", "11", "空闲", ""], + ["需求电压", "12", "0.0", " V", "0.1"], + ["需求电流", "13", "0.0", " A", "0.01"], + ["需求功率", "14", "0.0", " kW", "0.1"], + ["输出电压", "15", "0.0", " V", "0.1"], + ["输出电流", "16", "0.0", " A", "0.01"], + ["输出功率", "17", "0.0", " kW", "0.1"], + ["功率限值", "18", "0.0", " kW", "0.1"], ], - "addrCurve": ["31079", "31081", "31083"] + "addrCurve": ["15", "16", "17"] } } \ No newline at end of file diff --git a/src/app/Application.cpp b/src/app/Application.cpp index 3d44d33..cb88ed7 100644 --- a/src/app/Application.cpp +++ b/src/app/Application.cpp @@ -16,6 +16,11 @@ void Application::init() { // 初始化系统配置,读取配置文件 Config::init("assets/config/app.json"); + if (Config::option.debug) + { + spdlog::set_level(spdlog::level::debug); // 设置全局日志等级为 debug + spdlog::debug("[app] spdlog debug enable."); + } // MQTT 数据结构 REGAddr::load("assets/config/regaddrs.json"); @@ -77,8 +82,8 @@ void Application::runThreadMain() if (!this->isInit) { continue; } } - static TimeTick ttMqtt(1); // 检查 场站的 MQTT 连接 - if (ttMqtt.elapse(20)) + static TimeTick ttMqtt; // 检查 场站的 MQTT 连接 + if (ttMqtt.elapse(30)) { auto& optionMqtt = Config::option.mqtt; if (!optionMqtt.host.empty()) @@ -86,22 +91,21 @@ void Application::runThreadMain() for (auto& item : appdata.mapStation) { auto& station = item.second; - if (station && station->isOpen) + if (station) { - // 该函数检查连接状态,若已经连接,则无操作;若未连接,则进行连接操作 - item.second->initMqtt(); - // 召测 - item.second->polling(); + if (station->isOpen) + { + // 该函数检查连接状态,若已经连接,则无操作;若未连接,则进行连接操作 + item.second->initMqtt(); + // 召测 + item.second->polling(); + } + // 检查设备的在线状态 + station->checkDevice(); } } } } - - static TimeTick ttData(1); // 检查数据 - if (ttData.elapse(20)) - { - //appdata.initFromDB(); - } } } @@ -110,20 +114,14 @@ void Application::runThreadStat() int nCachePos = 0; while (!isQuit) { - int64_t tTime = Utils::time(); - int64_t tDate = Utils::date(); - int64_t delta = tTime-tDate; - int n = delta / 600; - int offset = delta % 600; - bool flagStore = (delta >=0 && delta < 86400 && offset <= 10 && n != nCachePos); - if (flagStore) + static TimeTick ttStat(1); + if(ttStat.elapse(10)) { - nCachePos = n; - std::string dt = Utils::dateStr(tDate); - // // 设备历史数据(电压、电流、功率),存储到 history_day + // 设备历史数据(电压、电流、功率),存储到 history_day + // 统计数据,存储到 stat_station for (auto item: appdata.mapStation) { - item.second->writeRuntimeData(dt, nCachePos); + item.second->writeStatistic(); } } else @@ -131,12 +129,6 @@ void Application::runThreadStat() //spdlog::info("保存历史数据倒计时: {}", 600 - offset); } - // 统计计算,存储到 stat_station - for (auto& station : appdata.mapStation) - { - - } - std::this_thread::sleep_for(std::chrono::seconds(1)); } } \ No newline at end of file diff --git a/src/app/Config.cpp b/src/app/Config.cpp index e4129f6..b18e153 100644 --- a/src/app/Config.cpp +++ b/src/app/Config.cpp @@ -5,6 +5,7 @@ #include "common/JsonN.h" #include "common/Spdlogger.h" #include "AppData.h" +#include "protocol/MqttEntity.h" AppOption Config::option; @@ -19,49 +20,106 @@ bool Config::init(std::string filename) } spdlog::info("[config] load config file success, filename={}", filename); + JSON::read(jsonroot, "debug", option.debug); + JSON::read(jsonroot, "weburl", option.webSrvUrl); + JSON::read(jsonroot, "launchdate", option.lunchDate); + + if (jsonroot.contains("database")) { - njson json = jsonroot.at("database"); + njson& json = jsonroot.at("database"); JSON::read(json, "host", option.database.host); JSON::read(json, "port", option.database.port); JSON::read(json, "user", option.database.user); JSON::read(json, "passwd", option.database.passwd); JSON::read(json, "dbname", option.database.dbname); - spdlog::info("[config] parse database success. host={}", option.database.host); } else { - spdlog::info("[config] parse database failed: not found."); + spdlog::error("[config] parse database failed: not found."); } if (jsonroot.contains("http")) { - njson json = jsonroot.at("http"); - std:string token; - JSON::read(json, "token", token); - option.http.useToken = !token.empty(); + njson& json = jsonroot.at("http"); + JSON::read(json, "token", option.http.useToken); JSON::read(json, "port", option.http.port); + JSON::read(json, "encryption", option.http.encryption); + JSON::read(json, "encryptKey", option.http.encryptKey); } else { - spdlog::info("[config] parse http failed: not found."); + spdlog::error("[config] parse http failed: not found."); } if (jsonroot.contains("mqtt")) { - njson json = jsonroot.at("mqtt"); + njson& json = jsonroot.at("mqtt"); JSON::read(json, "host", option.mqtt.host); JSON::read(json, "username", option.mqtt.username); JSON::read(json, "password", option.mqtt.password); } else { - spdlog::info("[config] parse mqtt failed: not found."); + spdlog::error("[config] parse mqtt failed: not found."); } - JSON::read(jsonroot, "weburl", option.webSrvUrl); - JSON::read(jsonroot, "launchdate", option.lunchDate); + if (jsonroot.contains("view")) + { + njson& json = jsonroot["view"]; + JSON::read(json, "latitude", option.view.latitude); + JSON::read(json, "longitude", option.view.longitude); + JSON::read(json, "altitude", option.view.altitude); + } + else + { + spdlog::error("[config] parse view failed: not found."); + } + if (jsonroot.contains("video")) + { + njson& json = jsonroot["video"]; + for (auto& item: json.items()) + { + auto& key = item.key(); + auto& jsonItem = item.value(); + auto& itemVideo = option.mapVideo[key]; + JSON::read(jsonItem, "host", itemVideo.host); + JSON::read(jsonItem, "port", itemVideo.port); + JSON::read(jsonItem, "user", itemVideo.user); + JSON::read(jsonItem, "passwd", itemVideo.passwd); + } + } + else + { + spdlog::error("[config] parse video failed: not found."); + } + + if (jsonroot.contains("topic")) + { + njson& json = jsonroot["topic"]; + for (auto& item: json.items()) + { + auto& key = item.key(); + auto& jsonItem = item.value(); + auto& info = MqttClient::s_mapTopicInfo[key]; + info.name = key; + JSON::read(jsonItem, "deviceType", info.deviceType); + JSON::read(jsonItem, "polling", info.polling); + JSON::read(jsonItem, "enabled", info.enabled); + } + } return true; +} + + +AppOption::VideoInfo* Config::getVideoInfo(std::string name) +{ + auto iter = option.mapVideo.find(name); + if (iter!=option.mapVideo.end()) + { + return &(iter->second); + } + return nullptr; } \ No newline at end of file diff --git a/src/app/Config.h b/src/app/Config.h index f800504..ad6bd91 100644 --- a/src/app/Config.h +++ b/src/app/Config.h @@ -1,23 +1,27 @@ #pragma once +#include #include -struct DatabaseOption -{ - std::string host; - int port; - std::string user; - std::string passwd; - std::string dbname; -}; - struct AppOption { - DatabaseOption database; + int debug {0}; + std::string webSrvUrl; + std::string lunchDate; struct { - bool useToken {true}; + std::string host; + int port; + std::string user; + std::string passwd; + std::string dbname; + } database; + + struct { + int useToken {1}; int port {0}; + int encryption {1}; + std::string encryptKey; } http; struct { @@ -26,8 +30,20 @@ struct AppOption std::string password; } mqtt; - std::string webSrvUrl; - std::string lunchDate; + struct { + float latitude {0}; + float longitude {0}; + float altitude {0}; + } view; + + struct VideoInfo { + std::string host; + int port; + std::string user; + std::string passwd; + }; + + std::map mapVideo; }; @@ -36,6 +52,7 @@ class Config public: static bool init(std::string filename); + static AppOption::VideoInfo* getVideoInfo(std::string name); static AppOption option; }; \ No newline at end of file diff --git a/src/app/Device.cpp b/src/app/Device.cpp index 5fcce92..59654e5 100644 --- a/src/app/Device.cpp +++ b/src/app/Device.cpp @@ -256,6 +256,9 @@ void Device::storeDB(int npos) void Device::setParam(std::string k, int v) { + this->ts = Utils::time(); + online = 1; + float ratio = 1.0; auto iter = mapMyParams.find(k); if (iter != mapMyParams.end()) @@ -264,40 +267,62 @@ void Device::setParam(std::string k, int v) //spdlog::info("[device] set param: {} {}={}, ratio={}", iter->second->name, k, v, ratio); } + if (type == 106) // 充电桩2号枪,特殊数据格式 + { + if (k=="22") { ratio = 0.1; } + else if (k=="23") { ratio = 0.01; } + else if (k== "24") { ratio = 0.1; } + else if (k== "25") { ratio = 0.1; } + else if (k== "26") { ratio = 0.01; } + else if (k== "27") { ratio = 0.1; } + else if (k== "28") { ratio = 0.1; } + } + int precision = (ratio != 1.0f) ? 1 : 0; - mapParams[k] = Utils::toStr(v*ratio, precision); + std::string valStr = Utils::toStr(v*ratio, precision); + if (type == 106) // 充电桩状态,特殊数据格式 + { + if (k=="11" || k == "21") { + valStr = (valStr == "1" ? "充电" : "空闲"); + } + } + mapParams[k] = valStr; if (type == 3 ) // 电表 - { - if (k == "") this->err = v; + { + running = 1; } else if (type == 101) // EMS { + running = 1; } else if (type == 102) // PCS { - if (k == "0x1003") err = v; // 故障状态 R uint16 1故障,0正常 0 0x1003 - if (k == "0x1005") online = v; // 设备在线 R uint16 1在线,0无效 1 0x1005 - if (k == "0x1009") running = (v==1 || v==2); //充放状态 R uint16 0:待机, 1:充电, 2:放电, 3:搁置 0 0x1009 + if (k == "0x1003") err = v; // 故障状态 R uint16 1故障,0正常 0 0x1003 + else if (k == "0x1005") online = v; // 设备在线 R uint16 1在线,0无效 1 0x1005 + else if (k == "0x1009") running = (v==1 || v==2); //充放状态 R uint16 0:待机, 1:充电, 2:放电, 3:搁置 0 0x1009 } else if (type == 103) // PCU { - if (k == "0x1002") err = v; //故障状态 R uint16 1故障,0正常 0 0x1002 - if (k == "0x1004") online = v; //设备在线 R uint16 1在线,0无效 1 0x1004 - if (k == "0x1006") running = v; //启停状态 R uint16 1开机,0关机 1 0x1006 + if (k == "0x1002") err = v; //故障状态 R uint16 1故障,0正常 0 0x1002 + else if (k == "0x1004") online = v; //设备在线 R uint16 1在线,0无效 1 0x1004 + else if (k == "0x1006") running = v; //启停状态 R uint16 1开机,0关机 1 0x1006 } else if (type == 104) // BMS { - if (k == "0x004A") { err = (v==1); online = 1; } //运行状态 R uint16 0 运行状态 0-正常 1-告警 2-保护 0x004A - if (k == "0x004B") running = (v==1 || v==2); //充放电状态 R uint16 0 0-待机 1-充电 2-放电 0x004B + if (k == "0x004A") { err = (v==1); } //运行状态 R uint16 0 运行状态 0-正常 1-告警 2-保护 0x004A + else if (k == "0x004B") running = (v==1 || v==2); //充放电状态 R uint16 0 0-待机 1-充电 2-放电 0x004B } else if (type == 105) // BCU { - if (k == "0xA003") running = (v==0x33 || v==0x44); //蓄电池充放电状态 R uint16 "0x11开路,0x22待机,0x33充电,0x44放电" 34 0xA003 - if (k == "0xA004") err = (v==0x55); online=1; //电池组运行状态 R uint16 "0x11跳机,0x22待机,0x33放空,0x44充满,0x55预警,0x66正常" 102 0xA004 + if (k == "0xA003") { running = (v==0x33 || v==0x44); } //蓄电池充放电状态 R uint16 "0x11开路,0x22待机,0x33充电,0x44放电" 34 0xA003 + else if (k == "0xA004") { err = (v==0x55); } //电池组运行状态 R uint16 "0x11跳机,0x22待机,0x33放空,0x44充满,0x55预警,0x66正常" 102 0xA004 } else if (type == 106) // 充电桩 { + if (k == "21") { + running = (mapParams["11"] == "充电" || mapParams["21"] == "充电"); // 充电状态: 0:空闲,1:充电 + } } else if (type == 109) // 光伏板 { @@ -343,12 +368,13 @@ void Device::getRuntimeParams1(std::vector>& { if (type == 106) { - params.push_back({"需求电压", getParam("31072", "0.0") + " V"}); - params.push_back({"需求电流", getParam("31074", "0.0") + " A"}); - params.push_back({"需求功率", getParam("31076", "0.0") + " kW"}); - params.push_back({"功率限值", getParam("31078", "0.0") + " kW"}); - params.push_back({"输出电压", getParam("31080", "0.0") + " V"}); - params.push_back({"输出电流", getParam("31082", "0.0") + " A"}); - params.push_back({"输出功率", getParam("31084", "0.0") + " kW"}); + params.push_back({"工作状态", getParam("21", "空闲")}); + params.push_back({"需求电压", getParam("22", "0.0") + " V"}); + params.push_back({"需求电流", getParam("23", "0.0") + " A"}); + params.push_back({"需求功率", getParam("24", "0.0") + " kW"}); + params.push_back({"输出电压", getParam("25", "0.0") + " V"}); + params.push_back({"输出电流", getParam("26", "0.0") + " A"}); + params.push_back({"输出功率", getParam("27", "0.0") + " kW"}); + params.push_back({"功率限值", getParam("28", "0.0") + " kW"}); } } diff --git a/src/app/Device.h b/src/app/Device.h index d044dfd..d54aa2c 100644 --- a/src/app/Device.h +++ b/src/app/Device.h @@ -73,6 +73,8 @@ public: int online = 0; int running = 0; + int64_t ts {0}; + //std::map mapAttrs; Fields attrs; diff --git a/src/app/Station.cpp b/src/app/Station.cpp index b5eb7c2..1cff2b1 100644 --- a/src/app/Station.cpp +++ b/src/app/Station.cpp @@ -12,20 +12,6 @@ Station::Station() : stationId(0) { mqttCli = std::make_shared(); - - // 测试,设置默认值 - for (int i = 1; i<=5; i++) { - auto& unit = mapCoolingUnit[i]; - unit.powerOn = 1; - unit.mode = i%2; - } - - for (int i = 1; i<=5; i++) { - auto& unit = mapAircUnit[i]; - unit.powerOn = 1; - unit.temp = Utils::random(20, 40); - unit.hum = Utils::random(20, 80); - } } void Station::setFields(Fields& fields) @@ -33,12 +19,12 @@ void Station::setFields(Fields& fields) this->stationId = fields.get(DMStation::STATION_ID); this->name = fields.value(DMStation::NAME); this->capacity = fields.get(DMStation::CAPACITY); - this->workModeId = fields.get(DMStation::WORK_MODE); + this->workMode = fields.get(DMStation::WORK_MODE); this->code = fields.value(DMStation::CODE); this->status = fields.get(DMStation::STATUS); this->operationDate = fields.value(DMStation::OPERATION_DATE); this->isOpen = fields.get(DMStation::STATUS); - + this->launchDate = fields.value("operation_date"); this->policy.setFields(fields); } @@ -134,7 +120,7 @@ void Station::getDeviceByCategory(int category, std::vectorworkModeId = modeId; + this->workMode = modeId; std::string sql = SQL(SQL::TYPE::update).table(DMStation::TABLENAME) .update(DMStation::WORK_MODE, std::to_string(modeId)) .where(DMStation::STATION_ID + "=" + std::to_string(stationId)).str(); @@ -157,71 +143,6 @@ void Station::setPolicy(int policyId) } } -static std::string MapValueToJson(int npos, std::map& mapV) -{ - njson jsonarray = njson::array(); - for (int i = 0; i<=npos; i++) - { - jsonarray.push_back(mapV[i]); - } - return jsonarray.dump(); -} - -void Station::writeRuntimeData(std::string dt, int npos) -{ - auto dao = DaoEntity::create("history_day"); - for (auto iter = mapDevice.begin(); iter!=mapDevice.end(); ++iter) - { - auto device = iter->second; - if (device->cache(npos)) - { - Fields fields; - fields.set("dt", dt); - fields.set("station_id", this->stationId); - fields.set("device_id", device->deviceId); - fields.set("datatype", 1); - fields.set("value", MapValueToJson(npos, device->mapCacheVoltage)); - DAO::insertRuntimeData(dao, fields); - - fields.set("datatype", 2); - fields.set("value", MapValueToJson(npos, device->mapCacheCurrent)); - DAO::insertRuntimeData(dao, fields); - - fields.set("datatype", 3); - fields.set("value", MapValueToJson(npos, device->mapCachePower)); - DAO::insertRuntimeData(dao, fields); - - spdlog::info("[device] write runtime date to database, deviceId={}", device->deviceId); - } - } -} - -void Station::writeStatistic(std::string dt) -{ - Fields fields; - fields.set("storage_elect_in", statData.totalElectIn); - fields.set("storage_elect_out", statData.totalElectOut); - //fields.set("storage_num_in", statData.totalElectIn); - //fields.set("storage_num_out", 0); - //fields.set("storage_num_err", 0); - //fields.set("storage_t_in", 0); - //fields.set("storage_t_out", 0); - //fields.set("storage_usage", 0); - //fields.set("solar_elect_gen", 0); - //fields.set("solar_elect_grid", 0); - //fields.set("solar_num_err", 0); - //fields.set("solar_t", 0); - //fields.set("solar_usage", 0); - //fields.set("charge_elect", 0); - //fields.set("charge_num", 0); - //fields.set("charge_num_err", 0); - //fields.set("charge_t", 0); - //fields.set("charge_usage", 0); - fields.set("income_elect", statData.totalIncome); - //fields.set("income_charge", 0); - //fields.set("usage_rate", 0); -} - void Station::initMqtt() { if (status!=0 && mqttCli) @@ -249,7 +170,7 @@ void Station::setGarewayWorkMode() njson json; json["ts"] = Utils::time(); json["no"] = 1; // 设备编号 - json["40001"] = this->workModeId; + json["40001"] = this->workMode; if (policy.type == 1) { @@ -267,19 +188,43 @@ void Station::setGarewayWorkMode() mqttCli->publish("Gateway_YT", text); } - -void Station::setRuntimeData(string addr, int val) +void Station::checkDevice() { - if (addr == "0x110E") { statData.dayElectIn = val; } //日充电电量 R uint32 1kWh 0 0x110E - else if (addr == "0x1110") { statData.dayElectOut = val; } //日放电电量 R uint32 1kWh 0 0x1110 - else if (addr == "0x1112") { statData.dayIncomeIn = val; } //日充电费用 R uint32 1RMB 0 0x1112 - else if (addr == "0x1114") { statData.dayIncomeOut = val; } //日放电费用 R uint32 1RMB 0 0x1114 - else if (addr == "0x1116") { statData.dayIncome = val; } //日收益 R int32 1RMB 0 0x1116 - else if (addr == "0x112C") { statData.totalElectIn = val; } //总充电电量 R uint32 1kWh 6659(0x112D) 0x112C - else if (addr == "0x112E") { statData.totalElectOut = val; } //总放电电量 R uint32 1kWh 4925(0x112F) 0x112E - else if (addr == "0x1130") { statData.totalIncomeIn = val; } //总充电费用 R uint32 1RMB 6605(0x1131) 0x1130 - else if (addr == "0x1132") { statData.totalIncomeOut = val; } //总放电费用 R uint32 1RMB 4949(0x1133) 0x1132 - else if (addr == "0x1134") { statData.totalIncome = val; } //总收益 R int32 1RMB -1 0x1134 + for (auto& item: mapDevice) + { + auto& device = item.second; + if (device) + { + if (Utils::time() - device->ts > 60*6) + { + device->online = 0; + } + } + } +} + +void Station::setRuntimeData(int deviceNo, string addr, int val) +{ + if (deviceNo == 1) + { + if (addr == "0x000B") { this->voltage = val; } // A相电压 R uint32 1V 0x000B + if (addr == "0x0011") { this->current = val; } // A相电流 R int32 1A 0x0011 + if (addr == "0x0011") { this->power = val; } // 三相总有功 R int32 1kW 0x0023 + } + else if (deviceNo == 2) + { + statData.ts = Utils::time(); + if (addr == "0x002F") { statData.dayElectIn = val; } //日充电电量 R uint32 1kWh 0x002F + else if (addr == "0x0031") { statData.dayElectOut = val; } //日放电电量 R uint32 1kWh 0x0031 + else if (addr == "0x0033") { statData.dayIncomeIn = val; } //日充电费用 R uint32 1RMB 0x0033 + else if (addr == "0x0035") { statData.dayIncomeOut = val; } //日放电费用 R uint32 1RMB 0x0035 + else if (addr == "0x0037") { statData.dayIncome = val; } //日收益 R int32 1RMB 0x0037 + else if (addr == "0x004D") { statData.totalElectIn = val; } //总充电电量 R uint32 1kWh 0x004D + else if (addr == "0x004F") { statData.totalElectOut = val; } //总放电电量 R uint32 1kWh 0x004F + else if (addr == "0x0051") { statData.totalIncomeIn = val; } //总充电费用 R uint32 1RMB 0x0051 + else if (addr == "0x0053") { statData.totalIncomeOut = val; } //总放电费用 R uint32 1RMB 0x0053 + else if (addr == "0x0055") { statData.totalIncome = val; } //总收益 R int32 1RMB 0x0055 + } } void Station::setTHData(int deviceNo, string addr, int val) @@ -287,8 +232,16 @@ void Station::setTHData(int deviceNo, string addr, int val) auto& unit = mapTempHumUnit[deviceNo]; if (addr == "0x0001") { ; } //所属通道号 R uint16 1 0x0001 else if (addr == "0x0002") { ; } //所属温湿度号 R uint16 1~10 0x0002 - else if (addr == "0x0003") { unit.temp = float(val) * 0.1; } //温度 R int16 0.1℃ 0x0003 - else if (addr == "0x0004") { unit.hum = float(val) * 0.1; } //湿度 R int16 0.1℃ 0x0004 + else if (addr == "0x0003") //温度 R int16 0.1℃ 0x0003 + { + unit.temp = float(val) * 0.1; + if (deviceNo == 1) temperature = unit.temp; + } + else if (addr == "0x0004") //湿度 R int16 0.1℃ 0x0004 + { + unit.hum = float(val) * 0.1; + if (deviceNo == 1) humidity = unit.hum; + } } void Station::setFire40Data(int deviceNo, string addr, int val) @@ -328,8 +281,8 @@ void Station::setCoolingData(int deviceNo, string addr, int val) if (addr == "0x1001") { ; } //所属通道号 R uint16 1 0x1001 else if (addr == "0x1002") { ; }// 所属冷机号 R uint16 1~10 0x1002 - else if (addr == "0x1003") { unit.powerOn = val; }// 开关 R uint16 0:关机,1:开机 0x1003 - else if (addr == "0x1004") { ; }// 采样模式 R uint16 0-出水温度 1-电芯温度 0x1004 + else if (addr == "0x1003") { coolingStatus = unit.powerOn = val; }// 开关 R uint16 0:关机,1:开机 0x1003 + else if (addr == "0x1004") { unit.mode = val; }// 采样模式 R uint16 0-出水温度 1-电芯温度 0x1004 else if (addr == "0x1005") { unit.cooling = val; }// 制冷状态 R uint16 0:关闭, 1:启动 0x1005 else if (addr == "0x1006") { unit.heating = val; }// 制热状态 R uint16 0:关闭, 1:启动 0x1006 else if (addr == "0x1007") { unit.highTemp = val; }// 高温告警 R uint16 0:正常,1:告警 0x1007 @@ -340,4 +293,112 @@ void Station::setCoolingData(int deviceNo, string addr, int val) else if (addr == "0x100C") { ; }// 出水温度传感器 R uint16 0:正常,1:告警 0x100C else if (addr == "0x100D") { ; }// 进水压力传感器 R uint16 0:正常,1:告警 0x100D else if (addr == "0x100E") { ; }// 出水压力传感器 R uint16 0:正常,1:告警 0x100E -} \ No newline at end of file +} + +void Station::setWorkModeFromGateway(int mode) +{ + if (mode != this->workMode) + { + //std::string sql = "update station set work_mode='" + std::to_string(mode) + "'"; + //auto ret = DaoEntity::execOnce(sql); + //if (ret) + //{ + // spdlog::info("[station] wrok_mode is diffent with gateway, update success.[{}]-[{}]", workMode, mode); + // this->workMode = mode; + //} + //else + //{ + // spdlog::error("[station] wrok_mode is diffent with gateway, update failed.[{}]-[{}]", workMode, mode); + //} + + //this->setGarewayWorkMode(); + } +} + + +static std::string MapValueToJson(int npos, std::map& mapV) +{ + njson jsonarray = njson::array(); + for (int i = 0; i<=npos; i++) + { + jsonarray.push_back(mapV[i]); + } + return jsonarray.dump(); +} + +void Station::writeStatistic() +{ + auto dao = DaoEntity::create("history_day"); + + std::string dt = Utils::dateStr(); + int64_t tTime = Utils::time(); + int64_t tDate = Utils::date(); + int npos = (tTime-tDate) / 600; + + for (auto iter = mapDevice.begin(); iter!=mapDevice.end(); ++iter) + { + auto device = iter->second; + if (device->cache(npos)) + { + Fields fields; + fields.set("dt", dt); + fields.set("station_id", this->stationId); + fields.set("device_id", device->deviceId); + fields.set("datatype", 1); + fields.set("value", MapValueToJson(npos, device->mapCacheVoltage)); + DAO::insertRuntimeData(dao, fields); + + fields.set("datatype", 2); + fields.set("value", MapValueToJson(npos, device->mapCacheCurrent)); + DAO::insertRuntimeData(dao, fields); + + fields.set("datatype", 3); + fields.set("value", MapValueToJson(npos, device->mapCachePower)); + DAO::insertRuntimeData(dao, fields); + //spdlog::info("[device] write runtime date to database, deviceId={}", device->deviceId); + } + } + + if (statData.ts != 0) + { + Fields fields; + fields.set("dt", Utils::dateStr(statData.ts)); + fields.set("station_id", this->stationId); + fields.set("device_id", 0); + fields.set("elect_in", statData.dayElectIn); + fields.set("elect_out", statData.dayElectOut); + fields.set("income_in", statData.dayIncomeIn); + fields.set("income_out", statData.dayIncomeOut); + fields.set("income", statData.dayIncome); + //fields.set("num_in", ""); + //fields.set("num_out", ""); + //fields.set("num_err", ""); + //fields.set("t_in", ""); + //fields.set("t_out", ""); + //fields.set("usage_rate", ""); + fields.set("elect_in_total", statData.totalElectIn); + fields.set("elect_out_total", statData.totalElectOut); + fields.set("income_in_total", statData.totalIncomeIn); + fields.set("income_out_total", statData.totalIncomeOut); + fields.set("income_total", statData.totalIncome); + + dao->setTableName("stat_storage"); + std::vector vecKeys = { + "elect_in", "elect_out", "num_in", "num_out", "num_err", "t_in", "t_out", "usage_rate", "income_in", "income_out", + "elect_in_total", "elect_out_total", "income_in_total", "income_out_total" + }; + dao->duplicateUpdate(fields, vecKeys); + + { + Fields fields; + fields.set("dt", Utils::dateStr(statData.ts)); + fields.set("station_id", this->stationId); + fields.set("device_id", 0); + fields.set("storage_elect_in", statData.dayElectIn); + fields.set("storage_elect_out", statData.dayElectOut); + fields.set("income_elect", statData.dayIncome); + DAO::insertStatStation(dao, fields); + } + + } +} diff --git a/src/app/Station.h b/src/app/Station.h index e14a641..db2838f 100644 --- a/src/app/Station.h +++ b/src/app/Station.h @@ -109,17 +109,18 @@ public: void setWorkMode(int modeId); void setPolicy(int policyId); - void writeRuntimeData(std::string dt, int npos); - void writeStatistic(std::string dt); - void initMqtt(); void polling(); void setGarewayWorkMode(); + void checkDevice(); - void setRuntimeData(string addr, int val); + void setRuntimeData(int deviceNo, string addr, int val); void setTHData(int deviceNo, string addr, int val); void setFire40Data(int deviceNo, string addr, int val); void setCoolingData(int deviceNo, string addr, int val); + void setWorkModeFromGateway(int mode); + + void writeStatistic(); public: int stationId {}; @@ -129,14 +130,13 @@ public: int status {0}; std::string operationDate; SysPolicy policy; + std::string launchDate {}; bool isConnected {false}; - int workModeId {}; // 运行模式 + int workMode {}; // 运行模式 int runPolicyId {}; // 运行策略 - - /////////////////////////////////////////////////////////////////////////////////////////////// /// === 系统统计 === // 累计发电量,单位:kWh @@ -185,6 +185,12 @@ public: double temperature {}; // 湿度 double humidity {}; + int aircStatus {0}; + int coolingStatus {0}; + double voltage {0}; + double current {0}; + double power {0}; + double powerFactor {0}; /////////////////////////////////////////////////////////////////////////////////////////////// /// === 设备信息 === @@ -207,25 +213,25 @@ public: struct { - int64_t ts; + int64_t ts {0}; - double totalElectIn; //总充电电量 R uint32 1kWh 6659(0x112D) 0x112C - double totalElectOut; //总放电电量 R uint32 1kWh 4925(0x112F) 0x112E - double totalIncomeIn; //总充电费用 R uint32 1RMB 6605(0x1131) 0x1130 - double totalIncomeOut; //总放电费用 R uint32 1RMB 4949(0x1133) 0x1132 - double totalIncome; //总收益 R int32 1RMB -1 0x1134 + float totalElectIn {0.0}; //总充电电量 R uint32 1kWh 6659(0x112D) 0x112C + float totalElectOut {0.0}; //总放电电量 R uint32 1kWh 4925(0x112F) 0x112E + float totalIncomeIn {0.0}; //总充电费用 R uint32 1RMB 6605(0x1131) 0x1130 + float totalIncomeOut {0.0}; //总放电费用 R uint32 1RMB 4949(0x1133) 0x1132 + float totalIncome {0.0}; //总收益 R int32 1RMB -1 0x1134 //储能充放电时段hh R uint16 时 336 0x01 0x121C //储能充放电时段mm R uint16 分 0 0x01 0x121D //储能充放电时段ss R uint16 秒 0 0x01 0x121E - double totalDurationIn; - double totalDurationOut; + float totalDurationIn {0.0}; + float totalDurationOut {0.0}; - double dayElectIn; // 日充电电量 R uint32 1kWh 0 0x110E - double dayElectOut; // 日放电电量 R uint32 1kWh 0 0x1110 - double dayIncomeIn; // 日充电费用 R uint32 1RMB 0 0x1112 - double dayIncomeOut; // 日放电费用 R uint32 1RMB 0 0x1114 - double dayIncome; // 日收益 R int32 1RMB 0 0x1116 + float dayElectIn {0.0}; // 日充电电量 R uint32 1kWh 0 0x110E + float dayElectOut {0.0}; // 日放电电量 R uint32 1kWh 0 0x1110 + float dayIncomeIn {0.0}; // 日充电费用 R uint32 1RMB 0 0x1112 + float dayIncomeOut {0.0}; // 日放电费用 R uint32 1RMB 0 0x1114 + float dayIncome {0.0}; // 日收益 R int32 1RMB 0 0x1116 } statData; diff --git a/src/common/Spdlogger.cpp b/src/common/Spdlogger.cpp index a9b45bd..61c674b 100644 --- a/src/common/Spdlogger.cpp +++ b/src/common/Spdlogger.cpp @@ -21,7 +21,7 @@ void Spdlogger::init(spdlog::level::level_enum log_level, std::string filename) // 创建控制台接收器 auto consoleSink = std::make_shared(); - consoleSink->set_level(spdlog::level::info); // 设置控制台日志等级 + //consoleSink->set_level(log_level); // 设置控制台日志等级 //consoleSink->set_pattern("[%T] [%^%l%$] %v"); // 设置日志格式 // 创建文件接收器 @@ -31,7 +31,7 @@ void Spdlogger::init(spdlog::level::level_enum log_level, std::string filename) // 每日文件sink(可选,每天生成新文件) auto dailySink = std::make_shared("logs/ess.log", 0, 0); - dailySink->set_level(spdlog::level::debug); + //dailySink->set_level(log_level); //dailySink->set_pattern("[%Y-%m-%d %H:%M:%S.%e] [%l] %v"); // 创建一个多重接收器的 logger @@ -40,8 +40,8 @@ void Spdlogger::init(spdlog::level::level_enum log_level, std::string filename) // 设置全局 logger spdlog::set_default_logger(logger); - spdlog::set_level(spdlog::level::debug); // 设置全局日志等级为 debug - spdlog::flush_on(spdlog::level::info); // 开启日志刷新 + spdlog::set_level(log_level); // 设置全局日志等级为 debug + spdlog::flush_on(log_level); // 开启日志刷新 } void Spdlogger::drop() diff --git a/src/database/Dao.cpp b/src/database/Dao.cpp index d41476b..2bd79b6 100644 --- a/src/database/Dao.cpp +++ b/src/database/Dao.cpp @@ -644,6 +644,7 @@ Errcode DAO::insertStatStation(std::shared_ptr dao, Fields& fields) { // 根据主键(dt、station_id、category),写入或更新数据 if (!dao) { dao = DaoEntity::create("stat_station"); } + else { dao->setTableName("stat_station"); } std::vector vecKeys = { "storage_elect_in", "storage_elect_out", diff --git a/src/database/DaoEntity.cpp b/src/database/DaoEntity.cpp index 6771809..9715e13 100644 --- a/src/database/DaoEntity.cpp +++ b/src/database/DaoEntity.cpp @@ -129,21 +129,18 @@ int DaoEntity::duplicateUpdate(Fields& fields, const vector& keys) string val; for (auto& item : fields.map()) { - if (!key.empty()) - { - key += ","; val += ","; - } + if (!key.empty()) { key += ","; val += ","; } key += (item.first); val += ("'" + item.second + "'"); } string str; for (auto& k : keys) { - if (!str.empty()) + if (fields.contains(k)) { - str += ","; + if (!str.empty()) { str += ","; } + str += (k + "='" + fields.value(k) + "'"); } - str += (k + "='" + fields.value(k) + "'"); } string sql = "INSERT INTO " + tableName + "(" + key + ") VALUES (" + val + ") ON duplicate KEY UPDATE " + str; return this->db->exec(sql); diff --git a/src/main.cpp b/src/main.cpp index 980dfd8..98ce03c 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -124,14 +124,15 @@ int main(int argc, char** argv) // 设置控制台输入为 UTF-8 编码(如果需要输入中文) SetConsoleCP(CP_UTF8); // 初始化日志 - Spdlogger::init(spdlog::level::debug, ""); + Spdlogger::init(spdlog::level::info, ""); spdlog::info("[main] start ... ======================================================================"); - qputenv("QTWEBENGINE_CHROMIUM_FLAGS", "--ignore-gpu-blacklist --enable-gpu-rasterization --enable-native-gpu-memory-buffers --num-raster-threads=4"); // 运行后台服务 Application::instance().init(); + + while (1) { std::this_thread::sleep_for(std::chrono::milliseconds(100)); }; // 启动 PV 服务主线程 diff --git a/src/protocol/HttpEntity.cpp b/src/protocol/HttpEntity.cpp index 7a2ba32..79bdb46 100644 --- a/src/protocol/HttpEntity.cpp +++ b/src/protocol/HttpEntity.cpp @@ -128,6 +128,8 @@ public: static std::map g_mapHttpHandlerGet = { + {"/queryBaseinfo", HandlerOptions(&HttpEntity::logqueryBaseinfoin, {})}, + {"/login", HandlerOptions(&HttpEntity::login, {DMUser::ACCOUNT, DMUser::PASSWD})}, {"/queryUserList", HandlerOptions(&HttpEntity::queryUserList, {})}, {"/deleteUser", HandlerOptions(&HttpEntity::deleteUser, { DMUser::USER_ID})}, @@ -280,6 +282,7 @@ void HttpEntity::runHandler(std::string name, const HandlerOptions& handler, con jsonresp["errmsg"] = ErrcodeStr(errcode) + (errmsg.empty() ? "" : (":"+errmsg)); resp.set_content(jsonresp.dump(), "text/plain; charset=utf-8"); resp.status = 200; + spdlog::info("[http] request: {}, response: {}.", name, int(errcode)); } void HttpEntity::registGet(std::string name, void (HttpEntity::* func)(const httplib::Request& req, httplib::Response& resp)) @@ -287,6 +290,18 @@ void HttpEntity::registGet(std::string name, void (HttpEntity::* func)(const htt this->httpsvr.Get(name, std::bind(func, this, std::placeholders::_1, std::placeholders::_2)); } +Errcode HttpEntity::logqueryBaseinfoin(const httplib::Request& req, njson& json, std::string& errmsg) +{ + json["data"] = { + {"encryption", Config::option.http.encryption}, + {"encryptKey", Config::option.http.encryptKey}, + {"latitude", Config::option.view.latitude}, + {"longitude", Config::option.view.longitude}, + {"altitude", Config::option.view.altitude} + }; + return Errcode::OK; +} + Errcode HttpEntity::login(const httplib::Request& req, njson& json, std::string& errmsg) { std::string userId; @@ -602,12 +617,13 @@ Errcode HttpEntity::insertStation(const httplib::Request& req, njson& json, std: Errcode HttpEntity::updateStation(const httplib::Request& req, njson& json, std::string& errmsg) { Fields params; - GetRequestParam(req, {"station_id", "name", "address", "lon", "lat", "tel", "capacity", "status", "work_mode", "policy_id"}, params); + GetRequestParam(req, {"station_id", "name", "address", "lon", "lat", "tel", "capacity", "status", "work_mode", "policy_id", "operation_date"}, params); std::string stationId = params.value("station_id"); params.check("capacity", "", "0.0"); params.check("lon", "", "0.0"); params.check("lat", "", "0.0"); params.check("status", "", "1"); + params.check("policy_id", "", "NULL"); Errcode err = DAO::updateStationById(params); if (err == Errcode::OK) { @@ -621,16 +637,11 @@ Errcode HttpEntity::updateStation(const httplib::Request& req, njson& json, std: } else { - if (result.size() > 0) + auto station = Application::data().getStation(Utils::toInt(stationId)); + if (result.size() > 0 && station) { - auto station = Application::data().getStation(Utils::toInt(stationId)); - if (station) - { - station->setFields(result[0]); - station->setGarewayWorkMode(); - } + station->setFields(result[0]); } - } } return err; @@ -681,6 +692,15 @@ Errcode HttpEntity::queryStationOverview(const httplib::Request& req, njson& jso njson jsonCharge = njson::parse(R"({"category":2, "count":0, "power":0.0})"); njson jsonSolar = njson::parse(R"({"category":3, "count":0, "power":0.0})"); njson jsonSecurity = njson::parse(R"({"category":4, "count":0, "power":0.0})"); + auto videoInfo = Config::getVideoInfo(stationId); + if (videoInfo) + { + jsonSecurity["host"] = videoInfo->host; + jsonSecurity["port"] = videoInfo->port; + jsonSecurity["user"] = videoInfo->user; + jsonSecurity["passwd"] = videoInfo->passwd; + } + for (auto& fields : result) { int category = fields.get("category"); @@ -743,16 +763,21 @@ Errcode HttpEntity::queryStationInfo(const httplib::Request& req, njson& json, s } Errcode HttpEntity::queryStationData(const httplib::Request& req, njson& json, std::string& errmsg) { - // 温度, 电压、电流、功率、功率因数、 + std::string stationId = req.get_param_value("station_id"); + auto station = Application::data().getStation(Utils::toInt(stationId)); njson jsondata; - jsondata["voltage"] = Utils::toStr(200.32); - jsondata["current"] = Utils::toStr(20.56); - jsondata["power"] = Utils::toStr(200.32); - jsondata["powerFactor"] = Utils::toStr(1); - jsondata["envTemp"] = Utils::toStr(200.32); - jsondata["envhum"] = Utils::toStr(200.32); - jsondata["aircStatus"] = Utils::toStr(1); - jsondata["coolingStatus"] = Utils::toStr(0); + if (station) + { + // 温度, 电压、电流、功率、功率因数、 + jsondata["voltage"] = station->voltage; + jsondata["current"] = station->current; + jsondata["power"] = station->power; + jsondata["powerFactor"] = station->powerFactor; + jsondata["envTemp"] = station->temperature; + jsondata["envhum"] = station->humidity; + jsondata["aircStatus"] = station->aircStatus; + jsondata["coolingStatus"] = station->coolingStatus; + } json["data"] = jsondata; return Errcode::OK; @@ -1106,9 +1131,9 @@ static std::string VerifyStatSqlCondition(Fields& params) return sqlCondition; } -static std::string GetRequestStatParams(const httplib::Request& req) +static std::string GetRequestStatParams(const httplib::Request& req, Fields& params) { - Fields params; + GetRequestParam(req, {"station_id", "category", "start_date", "end_date"}, params); VerifyRequstParamsStatDate(params); return VerifyStatSqlCondition(params); @@ -1116,7 +1141,8 @@ static std::string GetRequestStatParams(const httplib::Request& req) Errcode HttpEntity::queryStatTotal(const httplib::Request& req, njson& json, std::string& errmsg) { - std::string sqlCondition = GetRequestStatParams(req); + Fields params; + std::string sqlCondition = GetRequestStatParams(req, params); std::string sql = R"(SELECT SUM(ss.storage_elect_in) storage_elect_in, SUM(storage_elect_in) storage_elect_in, SUM(storage_elect_out) storage_elect_out, @@ -1135,12 +1161,22 @@ Errcode HttpEntity::queryStatTotal(const httplib::Request& req, njson& json, std SUM(income_charge) income_charge FROM stat_station ss)" + sqlCondition + ";"; + std::string stationId = params.value("station_id"); + + njson jsondata; + + auto station = Application::data().getStation(Utils::toInt(stationId)); + if (station) + { + jsondata["launch_date"] = station->launchDate; + } + std::vector result; DaoEntity::execOnce(sql, result); if (result.size() > 0) { auto& fields = result[0]; - njson jsondata; + // jsondata["launch_date"] = "2025-09-01"; //场站上线日期 // jsondata["station_id"] = station_id; jsondata["storage_elect_in"] = fields.value("storage_elect_in"); //储能充电电量(kWh),精度:0.001 @@ -1200,7 +1236,8 @@ Errcode HttpEntity::queryStatDetailList(const httplib::Request& req, njson& json std::vector result; auto err = DAO::queryStatStationList(pageinfo, params, result); - json["data"] = FieldsToJsonArray(result); + //json["data"] = FieldsToJsonArray(result); + HttpHelper::setPagination(pageinfo, result, json); return err; } @@ -1286,40 +1323,38 @@ Errcode HttpEntity::queryEnvironment(const httplib::Request& req, njson& json, s auto& unit = iter->second; njson node; node["pos"] = "#" + std::to_string(iter->first); - node["temp"] = unit.temp; - node["hum"] = unit.hum; + node["temp"] = Utils::toStr(unit.temp); + node["hum"] = Utils::toStr(unit.hum); nodearray.push_back(node); } jsondata["temp_hum"] = nodearray; } { //空调 auto& mapAircUnit = station->mapAircUnit; - AircUnit unitTmp; - AircUnit* unit = (mapAircUnit.size() > 0) ? &(mapAircUnit[0]) : &unitTmp; njson nodearray = njson::array(); - if (unit) + for (auto& item: mapAircUnit) { - nodearray.push_back({{"pos", "开关"}, {"status", unit->powerOn == 0 ? "关机" : "开机"}}); - nodearray.push_back({{"pos", "启动制冷指令"}, {"status", unit->cooling == 0 ? "启动" : "关闭"}}); - nodearray.push_back({{"pos", "启动送风指令"}, {"status", unit->airSupply == 0 ? "关闭" : "启动"}}); - nodearray.push_back({{"pos", "启动待机指令"}, {"status", unit->standby == 0 ? "关闭" : "启动"}}); - nodearray.push_back({{"pos", "启动加热指令"}, {"status", unit->heating == 0 ? "关闭" : "启动"}}); - nodearray.push_back({{"pos", "传感器故障"}, {"status", unit->sensorAlarm == 0 ? "正常" : "告警"}}); - nodearray.push_back({{"pos", "高低电压告警"}, {"status", unit->voltageAlarm == 0 ? "正常" : "告警"}}); - nodearray.push_back({{"pos", "高低温告警"}, {"status", unit->tempAlarm == 0 ? "正常" : "告警"}}); - nodearray.push_back({{"pos", "高低压告警"}, {"status", unit->pressureAlarm == 0 ? "正常" : "告警"}}); - nodearray.push_back({{"pos", "压缩机告警"}, {"status", unit->compressorAlarm == 0 ? "正常" : "告警"}}); - nodearray.push_back({{"pos", "当前温度"}, {"status", std::to_string(unit->temp) + "℃"}}); - nodearray.push_back({{"pos", "当前湿度"}, {"status", std::to_string(unit->hum) + "%"}}); + auto& unit = item.second; + nodearray.push_back({{"pos", "开关"}, {"status", unit.powerOn == 0 ? "关机" : "开机"}}); + nodearray.push_back({{"pos", "启动制冷指令"}, {"status", unit.cooling == 0 ? "启动" : "关闭"}}); + nodearray.push_back({{"pos", "启动送风指令"}, {"status", unit.airSupply == 0 ? "关闭" : "启动"}}); + nodearray.push_back({{"pos", "启动待机指令"}, {"status", unit.standby == 0 ? "关闭" : "启动"}}); + nodearray.push_back({{"pos", "启动加热指令"}, {"status", unit.heating == 0 ? "关闭" : "启动"}}); + nodearray.push_back({{"pos", "传感器故障"}, {"status", unit.sensorAlarm == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "高低电压告警"}, {"status", unit.voltageAlarm == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "高低温告警"}, {"status", unit.tempAlarm == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "高低压告警"}, {"status", unit.pressureAlarm == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "压缩机告警"}, {"status", unit.compressorAlarm == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "当前温度"}, {"status", std::to_string(unit.temp) + "℃"}}); + nodearray.push_back({{"pos", "当前湿度"}, {"status", std::to_string(unit.hum) + "%"}}); + break; } jsondata["airc"] = nodearray; } { // 消防 static std::map mapFireStatusDef = { {0, "正常"}, {1,"预警"}, {2,"火警"} }; - std::map mapStatusDef = {{0, "无效"}, {1, "掉线"}, {2, "正常"}, {3, "启动"}}; - auto& mapFire40Unit = station->mapFire40Unit; njson nodearray = njson::array(); for (auto iter = mapFire40Unit.begin(); iter!=mapFire40Unit.end(); ++iter) @@ -1338,20 +1373,19 @@ Errcode HttpEntity::queryEnvironment(const httplib::Request& req, njson& json, s } { // 冷机 auto& mapCoolingUnit = station->mapCoolingUnit; - CoolingUnit unitTmp; - CoolingUnit* unit = (mapCoolingUnit.size() > 0) ? &(mapCoolingUnit[0]) : &unitTmp; njson nodearray = njson::array(); - if (unit) + for (auto& item: mapCoolingUnit) { + auto& unit = item.second; njson node; - nodearray.push_back({{"pos", "开关"}, {"status", unit->powerOn == 0 ? "关机" : "开机"}}); - nodearray.push_back({{"pos", "采样模式"}, {"status", unit->mode == 0 ? "出水温度" : "电芯温度"}}); - nodearray.push_back({{"pos", "制冷状态"}, {"status", unit->cooling == 0 ? "关闭" : "启动"}}); - nodearray.push_back({{"pos", "制热状态"}, {"status", unit->heating == 0 ? "关闭" : "启动"}}); - nodearray.push_back({{"pos", "高温告警"}, {"status", unit->highTemp == 0 ? "正常" : "告警"}}); - nodearray.push_back({{"pos", "低温告警"}, {"status", unit->lowTemp == 0 ? "正常" : "告警"}}); - nodearray.push_back({{"pos", "高压告警"}, {"status", unit->highPressure == 0 ? "正常" : "告警"}}); - nodearray.push_back({{"pos", "低压告警"}, {"status", unit->lowPressure == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "开关"}, {"status", unit.powerOn == 0 ? "关机" : "开机"}}); + nodearray.push_back({{"pos", "采样模式"}, {"status", unit.mode == 0 ? "出水温度" : "电芯温度"}}); + nodearray.push_back({{"pos", "制冷状态"}, {"status", unit.cooling == 0 ? "关闭" : "启动"}}); + nodearray.push_back({{"pos", "制热状态"}, {"status", unit.heating == 0 ? "关闭" : "启动"}}); + nodearray.push_back({{"pos", "高温告警"}, {"status", unit.highTemp == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "低温告警"}, {"status", unit.lowTemp == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "高压告警"}, {"status", unit.highPressure == 0 ? "正常" : "告警"}}); + nodearray.push_back({{"pos", "低压告警"}, {"status", unit.lowPressure == 0 ? "正常" : "告警"}}); } jsondata["cooling"] = nodearray; } diff --git a/src/protocol/HttpEntity.h b/src/protocol/HttpEntity.h index edd8418..d01e1f8 100644 --- a/src/protocol/HttpEntity.h +++ b/src/protocol/HttpEntity.h @@ -29,6 +29,7 @@ public: //void onGet(const httplib::Request& req, httplib::Response& resp); + Errcode logqueryBaseinfoin(const httplib::Request& req, njson& json, std::string& errmsg); Errcode login(const httplib::Request& req, njson& json, std::string& errmsg); Errcode queryUserList(const httplib::Request& req, njson& json, std::string& errmsg); diff --git a/src/protocol/MqttEntity.cpp b/src/protocol/MqttEntity.cpp index b12b9e1..8b61fe3 100644 --- a/src/protocol/MqttEntity.cpp +++ b/src/protocol/MqttEntity.cpp @@ -9,6 +9,20 @@ #define TIMEOUT 10000L +bool MqttClient::load(std::string filename) +{ + njson jsonroot; + bool ret = JSON::load(filename, jsonroot); + if (!ret) + { + spdlog::error("[mqtt] load config file failed, filename={}", filename); + return false; + } + return true; +} + + +std::map MqttClient::s_mapTopicInfo; int MqttClient::init(string addr, string clientId, string username, string password) { @@ -25,7 +39,7 @@ int MqttClient::init(string addr, string clientId, string username, string passw this->clientId = clientId; //this->mapTopicInfo["EMS_YX"] = TopicInfo("EMS_YX", 101); - this->mapTopicInfo["EMS_YC"] = TopicInfo("EMS_YC", 101); + //this->mapTopicInfo["EMS_YC"] = TopicInfo("EMS_YC", 101); //this->mapTopicInfo["EMS_YT"] = TopicInfo("EMS_YT", 101); //this->mapTopicInfo["PCS_YX"] = TopicInfo("PCS_YX", 102, 1); //this->mapTopicInfo["PCS_YC"] = TopicInfo("PCS_YC", 102, 1); @@ -35,14 +49,14 @@ int MqttClient::init(string addr, string clientId, string username, string passw //this->mapTopicInfo["BMS_YC"] = TopicInfo("BMS_YC", 104); //this->mapTopicInfo["BCU_YX"] = TopicInfo("BCU_YX", 105, 1); //this->mapTopicInfo["BCU_YC"] = TopicInfo("BCU_YC", 105, 1); - //this->mapTopicInfo["MEM_YC"] = TopicInfo("MEM_YC", 3, 1); + //this->mapTopicInfo["MEM_YC"] = TopicInfo("MEM_YC", 3); // 不召测 //this->mapTopicInfo["TH_YC"] = TopicInfo("TH_YC", 10, 1); //this->mapTopicInfo["Fire40_YX"] = TopicInfo("Fire40_YX", 7, 1); - this->mapTopicInfo["Cooling_YC"] = TopicInfo("Cooling_YC", 14, 1); - this->mapTopicInfo["Cooling_YX"] = TopicInfo("Cooling_YX", 14, 1); + //this->mapTopicInfo["Cooling_YC"] = TopicInfo("Cooling_YC", 14, 1); + //this->mapTopicInfo["Cooling_YX"] = TopicInfo("Cooling_YX", 14, 1); //this->mapTopicInfo["Gateway_YX"] = TopicInfo("Gateway_YX", 15, 1); //this->mapTopicInfo["Gateway_YC"] = TopicInfo("Gateway_YC", 15, 1); - //this->mapTopicInfo["Charger_YC"] = TopicInfo("Charger_YC", 106, 1); + //this->mapTopicInfo["Charger_YC"] = TopicInfo("Charger_YC", 106); MQTTAsync_connectOptions option = MQTTAsync_connectOptions_initializer; MQTTAsync_message pubmsg = MQTTAsync_message_initializer; @@ -135,18 +149,21 @@ void MqttClient::subscribe() MQTTAsync_responseOptions options = MQTTAsync_responseOptions_initializer; options.onSuccess = funcSuccess; options.onFailure = funcFailure; - for (auto& item: mapTopicInfo) + for (auto& item: MqttClient::s_mapTopicInfo) { - std::string topic = "up/json/" + clientId + "/" + item.first; - options.context = (void*)&item.first; - int rc = MQTTAsync_subscribe(client, topic.data(), qos, &options); - if (rc != MQTTASYNC_SUCCESS) + if (item.second.enabled) { - spdlog::error("[mqtt] subscribe [{},{}] failed, err={}", topic, qos, rc); - } - else - { - spdlog::info("[mqtt] subscribe [{},{}] ", topic, qos); + std::string topic = "up/json/" + clientId + "/" + item.first; + options.context = (void*)&item.first; + int rc = MQTTAsync_subscribe(client, topic.data(), qos, &options); + if (rc != MQTTASYNC_SUCCESS) + { + spdlog::error("[mqtt] subscribe [{},{}] failed, err={}", topic, qos, rc); + } + else + { + spdlog::info("[mqtt] subscribe [{},{}] ", topic, qos); + } } } } @@ -199,10 +216,10 @@ int MqttClient::polling() return 0; } - for (auto& item: mapTopicInfo) + for (auto& item: MqttClient::s_mapTopicInfo) { auto& topicInfo = item.second; - if (topicInfo.polling) + if (topicInfo.polling && topicInfo.enabled) { std::vector> vecDevice; station->getDeviceByType(topicInfo.deviceType, vecDevice); @@ -268,96 +285,6 @@ void MqttClient::onConnectFaiure(MQTTAsync_failureData* resp) this->destory(); } - -void MqttClient::ParseArrivedMessage(njson& json, string clientId, string command, std::shared_ptr station) -{ - std::string stationNo = clientId; - - auto mapRegPtr = REGAddr::getRegMap(command); - if (!mapRegPtr) - { - spdlog::error("[mqtt] get register add info error, clientId={}, stationId={}, command={}", clientId, stationNo, command); - return; - } - - auto iterTopic = mapTopicInfo.find(command); - if (iterTopic == mapTopicInfo.end()) - { - spdlog::error("[mqtt] get topic info error, clientId={}, stationId={}, command={}", clientId, stationNo, command); - return; - } - TopicInfo& topicInfo = iterTopic->second; - - int deviceNo = -1; - JSON::read(json, "no", deviceNo); - auto device = station->getDeviceByType(topicInfo.deviceType, Utils::toStr(deviceNo)); - if (!device) - { - return; - } - for (auto& item: json.items()) - { - std::string key = item.key(); - if (key != "ts" && key != "no") - { - auto data = json.at(key); - if (data.is_array()) - { - std::string addrText; - auto iter = mapRegPtr->find(key); - for (int i = 0; iend()) - { - - auto addr = iter->first; - int val = data[i]; - device->setParam(addr, val); - spdlog::info("[mqtt] read [{}]={},{}", addr, val, iter->second.remark); - - if (command == "EMS_YC" && addr == "0x110C") - { - int a = 30; - a = 100; - } - - if (command == "EMS_YC") - { - station->setRuntimeData(addr, val); - } - else if (command == "Fire40_YX") - { - station->setFire40Data(deviceNo, addr, val); - } - else if (command == "TH_YC") - { - station->setTHData(deviceNo, addr, val); - } - else if (command == "Cooling_YX" || command == "Cooling_YC") - { - station->setCoolingData(deviceNo, addr, val); - } - else if (command == "Gateway_YX") - { - //if (key == "CDZ") "CDZ": 1, //充电桩 1:在线,0:离线 - //else if (key == "EMU") //储能 1:在线,0:离线 - } - ++iter; - } - } - } - else if (data.is_number()) - { - device->setParam(key, data.get()); - } - else if (data.is_string()) - { - device->setParam(key, Utils::toInt(data.get())); - } - } - } -} - int MqttClient::onMessageArrived(char* topic, int topicLen, MQTTAsync_message* msg) { std::string topicStr = topic; @@ -412,7 +339,7 @@ int MqttClient::onMessageArrived(char* topic, int topicLen, MQTTAsync_message* m } else { - ParseArrivedMessage(json, clientId, command, station); + ParseArrivedMessage(json, command, station); } // 必须释放消息内存! @@ -422,117 +349,119 @@ int MqttClient::onMessageArrived(char* topic, int topicLen, MQTTAsync_message* m } -string MQTT::pack(std::string name) +void MqttClient::ParseArrivedMessage(njson& json, string command, std::shared_ptr station) { - njson json; - json["ts"] = Utils::time(); - json["no"] = 1; + std::string stationNo = clientId; - if (name == "EMS_YC") + auto iterTopic = MqttClient::s_mapTopicInfo.find(command); + if (iterTopic == MqttClient::s_mapTopicInfo.end()) { - //A相电压 R uint32 1V 0x107E - //B相电压 R uint32 1V 0x1080 - //C相电压 R uint32 1V 0x1082 - //A相电流 R int32 1A 0x1084 - //B相电流 R int32 1A 0x1086 - //C相电流 R int32 1A 0x1088 - - //储能系统SOC R uint16 0.1 0x107A - //储能系统SOH R uint16 0.1 0x107B - - json["addr"] = {"0x107A", "0x107B", "0x107E", "0x1080", "0x1082", "0x1084", "0x1086", "0x1088"}; + spdlog::error("[mqtt] get topic info error, clientId={}, stationId={}, command={}", clientId, stationNo, command); + return; } - else if (name == "PCS_YC") + TopicInfo& topicInfo = iterTopic->second; + + int deviceNo = -1; + JSON::read(json, "no", deviceNo); + auto device = station->getDeviceByType(topicInfo.deviceType, Utils::toStr(deviceNo)); + if (!device) { - //总充电量 R uint32 1kWh 0x0003 - //总放电量 R uint32 1kWh 0x0005 - - //A相电压 R int16 1V 0x0010 - //B相电压 R int16 1V 0x0011 - //C相电压 R int16 1V 0x0012 - - //A相电流 R int16 1A 0x0019 - //B相电流 R int16 1A 0x001A - //C相电流 R int16 1A 0x001B - - //三相总有功功率 R int16 1kW 0x0025 - //三相总无功功率 R int16 1kVar 0x0026 - //三相总视在功率 R int16 1kVA 0x0027 - //三相总功率因数 R int16 1 0x0028 - - //充电功率 R int16 1kW 0x002C - //放电功率 R int16 1kW 0x002D - json["addr"] = {"0x0003", "0x0005", "0x0010", "0x0011", "0x0012", "0x0019", "0x001A", "0x001B", "0x0025", "0x0026", "0x0027", "0x0028", "0x002C", "0x002D"}; + spdlog::error("[mqtt] get device info error, clientId={}, stationId={}, command={}", clientId, stationNo, command); + return; } - else if (name == "PCU_YC") + + auto mapRegPtr = REGAddr::getRegMap(command); + if (!mapRegPtr) { - //PCS侧线A相电压 R int16 1v 0x0013 - //PCS侧线B相电压 R int16 1v 0x0014 - //PCS侧线C相电压 R int16 1v 0x0015 - - //PCS侧功率因数A R int16 1 0x0019 - //PCS侧功率因数B R int16 1 0x001A - //PCS侧功率因数C R int16 1 0x001B - - //PCS侧相电流A R int16 1A 0x001C - //PCS侧相电流B R int16 1A 0x001D - //PCS侧相电流C R int16 1A 0x001E - - //PCS侧三相总有功功率 R int16 1kW 0x0028 - //PCS侧三相总无功功率 R int16 1kVar 0x0029 - //PCS侧三相总视在功率 R int16 1kVA 0x002A - //PCS侧三相总功率因数 R int16 1 0x002B - - json["addr"] = {"0x0013", "0x0014", "0x0015", "0x1080", "0x1082", "0x1084", "0x1086", "0x1088"}; + spdlog::error("[mqtt] get register add info error, clientId={}, stationId={}, command={}", clientId, stationNo, command); + return; } - else if (name == "BMS_YC") + for (auto& item: json.items()) { - //SOC R uint16 0.1 0x0001 - //SOH R uint16 0.1 0x0002 - //电压 R uint32 0.1V 0x0003 - //电流 R int32 0.1A 0x0005 - //可充电量 R uint32 1kWh 0x0007 - //可放电量 R uint32 1kWh 0x0009 - //可充电状态 R uint16 1:可充电;0:不可充电 0x0047 - //可放电状态 R uint16 1:可放电;0:不可放电 0x0048 - //运行状态 R uint16 运行状态 0-正常 1-告警 2-保护 0x0049 - //充放电状态 R uint16 0-待机 1-充电 2-放电 0x004A + std::string key = item.key(); + if (key == "ts" || key == "no") + { + continue; + } + + auto data = json[key]; + if (data.is_array()) + { + if (command == "Charger_YC") + { + if (key == "1") key = "11"; + else if (key == "2") key = "21"; + } + std::string addrText; + auto iter = mapRegPtr->find(key); + if (iter != mapRegPtr->end()) + { + for (int i = 0; ifirst; + int val = data[i]; + device->setParam(addr, val); + spdlog::debug("[mqtt] read [{}]={}, {}", addr, val, iter->second.remark); - json["addr"] = {"0x0001", "0x0002", "0x0003", "0x0005", "0x0007", "0x0009", "0x0047", "0x0048", "0x0049", "0x004A"}; + if (command == "MEM_YC") { station->setRuntimeData(deviceNo, addr, val); } + else if (command == "Fire40_YX") { station->setFire40Data(deviceNo, addr, val); } + else if (command == "TH_YC") { station->setTHData(deviceNo, addr, val); } + else if (command == "Cooling_YX" || command == "Cooling_YC") { station->setCoolingData(deviceNo, addr, val); } + else if (command == "Gateway_YX") + { + //if (key == "CDZ") "CDZ": 1, //充电桩 1:在线,0:离线 + //else if (key == "EMU") //储能 1:在线,0:离线 + } + ++iter; + } + } + } + else if (data.is_number()) + { + device->setParam(key, data.get()); + } + else if (data.is_string()) + { + device->setParam(key, Utils::toInt(data.get())); + } + } +} +std::vector KEY_CHARGER_1 = {"31071", "31073", "31075", "31077", "31079", "31081", "31083"}; +std::vector KEY_CHARGER_2 = {"31072", "31074", "31076", "31078", "31080", "31082", "31084"}; +void MqttClient::ParseMessageCharge(njson& json, string command, std::shared_ptr station, std::shared_ptr device) +{ + if (json.contains("1")) + { + auto& jsondata = json["1"]; + if (jsondata.is_array()) + { + for (int i = 0; i(); + device->setParam(addr, val); + spdlog::info("[mqtt] read: 枪1 [{}]={}", addr, val); + } + } + } + } + if (json.contains("2")) + { + auto& jsondata = json["2"]; + if (jsondata.is_array()) + { + for (int i = 0; i(); + device->setParam(addr, val); + spdlog::info("[mqtt] read: 枪2 [{}]={}", addr, val); + } + } + } } - else if (name == "BCU_YC") - { - //电表类型 R uint16 "0:储能站总表 1:逆变前侧电表 2:逆变后侧电表 3:配电柜电表 4:并网口电表" 0x0008 - //A相电压 R uint32 1V 0x000B - //B相电压 R uint32 1V 0x000D - //C相电压 R uint32 1V 0x000F - //A相电流 R int32 1A 0x0011 - //B相电流 R int32 1A 0x0013 - //C相电流 R int32 1A 0x0015 - - //尖段电价 R uint32 1RMB 0x0027 - //峰段电价 R uint32 1RMB 0x0029 - //平段电价 R uint32 1RMB 0x002B - //谷段电价 R uint32 1RMB 0x002D - //日充电电量 R uint32 1kWh 0x002F - //日放电电量 R uint32 1kWh 0x0031 - //日充电费用 R uint32 1RMB 0x0033 - //日放电费用 R uint32 1RMB 0x0035 - //日收益 R int32 1RMB 0x0037 - - //总充电电量 R uint32 1kWh 0x004D - //总放电电量 R uint32 1kWh 0x004F - //总充电费用 R uint32 1RMB 0x0051 - //总放电费用 R uint32 1RMB 0x0053 - //总收益 R int32 1RMB 0x0055 - } - else if (name == "TH_YC") - { - //所属通道号 R uint16 1 0x0001 - //所属温湿度号 R uint16 1~10 0x0002 - //温度 R int16 0.1℃ 0x0003 - //湿度 R int16 0.1℃ 0x0004 - - } - return json.dump(); } diff --git a/src/protocol/MqttEntity.h b/src/protocol/MqttEntity.h index 908d0a2..96d10c4 100644 --- a/src/protocol/MqttEntity.h +++ b/src/protocol/MqttEntity.h @@ -7,13 +7,14 @@ #include "common/JsonN.h" class Station; - +class Device; struct TopicInfo { std::string name; int deviceType {0}; int polling {0}; // 召测 + int enabled {1}; TopicInfo() {}; TopicInfo(std::string name, int deviceType, int polling=0) :name(name), deviceType(deviceType), polling(polling) @@ -25,6 +26,8 @@ using namespace std; class MqttClient { public: + static bool load(std::string filename); + int init(string addr, string clientId, string username, string password); void destory(); @@ -39,7 +42,8 @@ public: void onConnectFaiure(MQTTAsync_failureData* resp); int onMessageArrived(char* topic, int len, MQTTAsync_message* msg); - void ParseArrivedMessage(njson& json, string clientId, string command, std::shared_ptr station); + void ParseArrivedMessage(njson& json, string command, std::shared_ptr station); + void ParseMessageCharge(njson& json, string command, std::shared_ptr station, std::shared_ptr device); public: // MQTT clientId (使用station 的 code) @@ -51,7 +55,7 @@ public: bool isConnected {false}; bool isSubscribed {false}; - std::map mapTopicInfo; + static std::map s_mapTopicInfo; }; @@ -71,12 +75,3 @@ public: #define TOPIC_PCS_YC "up/json/预制舱01/PCS_YC" #define TOPIC_PCS_YC "up/json/预制舱01/PCS_YC" - -class MQTT -{ -public: - - -public: - static string pack(std::string name); -}; \ No newline at end of file