diff --git a/src/app.cpp b/src/app.cpp index 4c096e83..4e3595d1 100644 --- a/src/app.cpp +++ b/src/app.cpp @@ -235,7 +235,7 @@ void app::regularTickers(void) { // Plugin ZeroExport #if defined(PLUGIN_ZEROEXPORT) - everySec(std::bind(&ZeroExportType::tickerSecond, &mZeroExport), "ZeroExport"); + everySec(std::bind(&ZeroExportType::tickSecond, &mZeroExport), "ZeroExport"); #endif // Plugin ZeroExport - Ende @@ -423,6 +423,10 @@ void app::tickMinute(void) { //----------------------------------------------------------------------------- void app::tickMidnight(void) { + #if defined(PLUGIN_ZEROEXPORT) + mZeroExport.tickMidnight(); + #endif /*defined(PLUGIN_ZEROEXPORT)*/ + uint32_t localTime = gTimezone.toLocal(mTimestamp); uint32_t nxtTrig = gTimezone.toUTC(localTime - (localTime % 86400) + 86400); // next midnight local time onceAt(std::bind(&app::tickMidnight, this), nxtTrig, "mid2"); diff --git a/src/config/settings.h b/src/config/settings.h index d898c4b4..1b70161b 100644 --- a/src/config/settings.h +++ b/src/config/settings.h @@ -204,7 +204,6 @@ typedef struct { enum class zeroExportState : uint8_t { INIT, WAIT, - PUBLISH, WAITREFRESH, GETINVERTERACKS, GETINVERTERDATA, @@ -213,9 +212,11 @@ enum class zeroExportState : uint8_t { CONTROLLER, PROGNOSE, AUFTEILEN, - SETLIMIT, - SETPOWER, SETREBOOT, + SETPOWER, + SETLIMIT, + PUBLISH, + EMERGENCY, FINISH, ERROR }; @@ -271,6 +272,9 @@ typedef struct { unsigned long limitTsp; float dcVoltage; bool state; + // + bool doReboot; + int8_t doPower; } zeroExportGroupInverter_t; typedef struct { @@ -296,12 +300,11 @@ typedef struct { uint16_t powerMax; // - zeroExportState stateLast; zeroExportState state; zeroExportState stateNext; unsigned long lastRun; unsigned long lastRefresh; -// bool waitForAck; + uint16_t sleep; float eSum; float eSum1; @@ -315,12 +318,12 @@ typedef struct { float Ki; float Kd; -float pm_P[5]; -float pm_P1[5]; -float pm_P2[5]; -float pm_P3[5]; -uint8_t pm_iIn = 0; -uint8_t pm_iOut = 0; +//float pm_P[5]; +//float pm_P1[5]; +//float pm_P2[5]; +//float pm_P3[5]; +//uint8_t pm_iIn = 0; +//uint8_t pm_iOut = 0; float pmPower; float pmPowerL1; @@ -697,6 +700,8 @@ class settings { mCfg.plugin.zeroExport.groups[group].inverters[inv].waitLimitAck = false; mCfg.plugin.zeroExport.groups[group].inverters[inv].waitPowerAck = false; mCfg.plugin.zeroExport.groups[group].inverters[inv].waitRebootAck = false; + mCfg.plugin.zeroExport.groups[group].inverters[inv].doReboot = false; + mCfg.plugin.zeroExport.groups[group].inverters[inv].doPower = -1; } // Battery mCfg.plugin.zeroExport.groups[group].battEnabled = false; @@ -714,6 +719,7 @@ class settings { mCfg.plugin.zeroExport.groups[group].state = zeroExportState::INIT; mCfg.plugin.zeroExport.groups[group].lastRun = 0; mCfg.plugin.zeroExport.groups[group].lastRefresh = 0; + mCfg.plugin.zeroExport.groups[group].sleep = 0; mCfg.plugin.zeroExport.groups[group].pmPower = 0; mCfg.plugin.zeroExport.groups[group].pmPowerL1 = 0; mCfg.plugin.zeroExport.groups[group].pmPowerL2 = 0; diff --git a/src/plugins/zeroExport/zeroExport.h b/src/plugins/zeroExport/zeroExport.h index 01343aef..054d897b 100644 --- a/src/plugins/zeroExport/zeroExport.h +++ b/src/plugins/zeroExport/zeroExport.h @@ -17,10 +17,6 @@ template -// TODO: Anbindung an MQTT für Logausgabe. -// TODO: Powermeter erweitern -// TODO: Der Teil der noch in app.cpp steckt komplett hier in die Funktion verschieben. - class ZeroExport { public: /** ZeroExport @@ -55,6 +51,10 @@ class ZeroExport { /** loop * Arbeitsschleife + * @param + * @returns + * @todo publish + * @todo emergency */ void loop(void) { if ((!mIsInitialized) || (!mCfg->enabled)) { @@ -63,76 +63,242 @@ class ZeroExport { mPowermeter.loop(); + unsigned long Tsp = millis(); + bool DoLog = false; + for (uint8_t group = 0; group < ZEROEXPORT_MAX_GROUPS; group++) { if (!mCfg->groups[group].enabled) { continue; } - bool flag = false; + // sleep + if (Tsp <= (mCfg->groups[group].lastRun + mCfg->groups[group].sleep)) { + continue; + } + mCfg->groups[group].sleep = 0; + + mLog["g"] = group; + mLog["s"] = (uint8_t)mCfg->groups[group].state; switch (mCfg->groups[group].state) { case zeroExportState::INIT: - flag = groupInit(group); + if (groupInit(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK -> Wait, WaitRefresh + mCfg->groups[group].state = zeroExportState::WAIT; + mCfg->groups[group].stateNext = zeroExportState::WAITREFRESH; + } else { + // Error -> Wait, Init + mCfg->groups[group].state = zeroExportState::WAIT; + mCfg->groups[group].stateNext = zeroExportState::INIT; + } break; case zeroExportState::WAIT: - flag = groupWait(group); - break; - case zeroExportState::PUBLISH: - flag = groupPublish(group); + if (groupWait(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK + if (mCfg->groups[group].state != mCfg->groups[group].stateNext) { + // OK -> Next + mCfg->groups[group].state = mCfg->groups[group].stateNext; + } else { + // OK -> Init + mCfg->groups[group].state = zeroExportState::INIT; + mCfg->groups[group].stateNext = zeroExportState::INIT; + } + } break; case zeroExportState::WAITREFRESH: - flag = groupWaitRefresh(group); + if (groupWaitRefresh(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK +#if defined(ZEROEXPORT_DEV_POWERMETER) + mCfg->groups[group].state = zeroExportState::GETPOWERMETER; + mCfg->groups[group].stateNext = zeroExportState::GETPOWERMETER; +#else + mCfg->groups[group].state = zeroExportState::GETINVERTERACKS; + mCfg->groups[group].stateNext = zeroExportState::GETINVERTERACKS; +#endif + } break; case zeroExportState::GETINVERTERACKS: - flag = groupGetInverterAcks(group); + if (groupGetInverterAcks(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::GETINVERTERDATA; + mCfg->groups[group].stateNext = zeroExportState::GETINVERTERDATA; + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 1000; + } break; case zeroExportState::GETINVERTERDATA: - flag = groupGetInverterData(group); + if (groupGetInverterData(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::BATTERYPROTECTION; + mCfg->groups[group].stateNext = zeroExportState::BATTERYPROTECTION; + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 500; + } break; case zeroExportState::BATTERYPROTECTION: - flag = groupBatteryprotection(group); + if (groupBatteryprotection(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::GETPOWERMETER; + mCfg->groups[group].stateNext = zeroExportState::GETPOWERMETER; + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 1000; + } break; case zeroExportState::GETPOWERMETER: - flag = groupGetPowermeter(group); + if (groupGetPowermeter(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK +#if defined(ZEROEXPORT_DEV_POWERMETER) + mCfg->groups[group].lastRefresh = millis(); + mCfg->groups[group].state = zeroExportState::PUBLISH; + mCfg->groups[group].stateNext = zeroExportState::PUBLISH; +#else + mCfg->groups[group].state = zeroExportState::CONTROLLER; + mCfg->groups[group].stateNext = zeroExportState::CONTROLLER; +#endif + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 3000; + } break; case zeroExportState::CONTROLLER: - flag = groupController(group); + if (groupController(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].lastRefresh = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::PROGNOSE; + mCfg->groups[group].stateNext = zeroExportState::PROGNOSE; + } else { + // Error - WaitRefresh + mCfg->groups[group].state = zeroExportState::WAITREFRESH; + mCfg->groups[group].stateNext = zeroExportState::WAITREFRESH; + } break; case zeroExportState::PROGNOSE: - flag = groupPrognose(group); + if (groupPrognose(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::AUFTEILEN; + mCfg->groups[group].stateNext = zeroExportState::AUFTEILEN; + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 500; + } break; case zeroExportState::AUFTEILEN: - flag = groupAufteilen(group); + if (groupAufteilen(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::SETREBOOT; + mCfg->groups[group].stateNext = zeroExportState::SETREBOOT; + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 500; + } break; - case zeroExportState::SETLIMIT: - flag = groupSetLimit(group); + case zeroExportState::SETREBOOT: + if (groupSetReboot(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].lastRefresh = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::SETPOWER; + mCfg->groups[group].stateNext = zeroExportState::SETPOWER; + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 1000; + } break; case zeroExportState::SETPOWER: - flag = groupSetPower(group); + if (groupSetPower(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].lastRefresh = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::SETLIMIT; + mCfg->groups[group].stateNext = zeroExportState::SETLIMIT; + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 1000; + } break; - case zeroExportState::SETREBOOT: - flag = groupSetReboot(group); + case zeroExportState::SETLIMIT: + if (groupSetLimit(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::PUBLISH; + mCfg->groups[group].stateNext = zeroExportState::PUBLISH; + } else { + // Error - Sleep + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].sleep = 1000; + } + break; + case zeroExportState::PUBLISH: + if (groupPublish(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::WAITREFRESH; + mCfg->groups[group].stateNext = zeroExportState::WAITREFRESH; + //} else { + // TODO: fehlt + } + break; + case zeroExportState::EMERGENCY: + if (groupEmergency(group, &Tsp, &DoLog)) { + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].lastRefresh = Tsp; + // OK + mCfg->groups[group].state = zeroExportState::INIT; + mCfg->groups[group].stateNext = zeroExportState::INIT; + //} else { + // TODO: fehlt + } break; case zeroExportState::FINISH: case zeroExportState::ERROR: default: + mCfg->groups[group].lastRun = Tsp; + mCfg->groups[group].lastRefresh = Tsp; + mCfg->groups[group].sleep = 1000; mCfg->groups[group].state = zeroExportState::INIT; mCfg->groups[group].stateNext = zeroExportState::INIT; - if (millis() > (mCfg->groups[group].lastRefresh + (mCfg->groups[group].refresh * 1000UL))) { - mCfg->groups[group].lastRefresh = millis(); - } break; } - if (flag) sendLog(); + if (mCfg->debug) { + unsigned long eTsp = millis(); + mLog["B"] = Tsp; + mLog["E"] = eTsp; + mLog["D"] = eTsp - Tsp; + } + + if (DoLog) sendLog(); + clearLog(); + DoLog = false; } + return; } - /** tickerSecond + /** tickSecond * Zeitimpuls */ - void tickerSecond() { - // TODO: Warten ob benötigt, ggf ein bit setzen, das in der loop() abgearbeitet wird. + void tickSecond() { if ((!mIsInitialized) || (!mCfg->enabled)) { return; } @@ -156,6 +322,23 @@ class ZeroExport { } } + /** tickerMidnight + * Zeitimpuls + */ + void tickMidnight(void) { + if ((!mIsInitialized) || (!mCfg->enabled)) { + return; + } + + // Reboot + for (uint8_t group = 0; group < ZEROEXPORT_MAX_GROUPS; group++) { + for (uint8_t inv = 0; inv < ZEROEXPORT_GROUP_MAX_INVERTERS; inv++) { + // Reboot + mCfg->groups[group].inverters[inv].doReboot = true; + } + } + } + /** resetWaitLimitAck * * @param @@ -177,10 +360,12 @@ class ZeroExport { mCfg->groups[group].inverters[inv].waitLimitAck = 0; mLog["w"] = 0; - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; + if (mCfg->debug) { + unsigned long eTsp = millis(); + mLog["B"] = bTsp; + mLog["E"] = eTsp; + mLog["D"] = eTsp - bTsp; + } sendLog(); return; } @@ -209,10 +394,12 @@ class ZeroExport { mCfg->groups[group].inverters[inv].waitPowerAck = 30; mLog["w"] = 30; - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; + if (mCfg->debug) { + unsigned long eTsp = millis(); + mLog["B"] = bTsp; + mLog["E"] = eTsp; + mLog["D"] = eTsp - bTsp; + } sendLog(); return; } @@ -241,10 +428,12 @@ class ZeroExport { mCfg->groups[group].inverters[inv].waitRebootAck = 30; mLog["w"] = 30; - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; + if (mCfg->debug) { + unsigned long eTsp = millis(); + mLog["B"] = bTsp; + mLog["E"] = eTsp; + mLog["D"] = eTsp - bTsp; + } sendLog(); return; } @@ -284,10 +473,13 @@ class ZeroExport { mLog["ivL"] = ivL; mLog["zeL"] = (uint16_t)mCfg->groups[group].inverters[inv].limit; mCfg->groups[group].inverters[inv].limit = ivL; - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; + + if (mCfg->debug) { + unsigned long eTsp = millis(); + mLog["B"] = bTsp; + mLog["E"] = eTsp; + mLog["D"] = eTsp - bTsp; + } sendLog(); return; } @@ -329,19 +521,18 @@ class ZeroExport { * @returns true/false * @todo getInverterById statt getInverterByPos, dann würde die Variable *iv und die Schleife nicht gebraucht. */ - bool groupInit(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); + bool groupInit(uint8_t group, unsigned long *tsp, bool *doLog) { + uint8_t result = false; + + if (mCfg->debug) mLog["t"] = "groupInit"; - mLog["t"] = "groupInit"; - mLog["g"] = group; + mCfg->groups[group].lastRun = *tsp; - mCfg->groups[group].stateLast = zeroExportState::INIT; + *doLog = true; // Init ivPointer for (uint8_t inv = 0; inv < ZEROEXPORT_GROUP_MAX_INVERTERS; inv++) { mIv[group][inv] = nullptr; - doLog = true; } // Search/Set ivPointer @@ -382,7 +573,7 @@ class ZeroExport { logObj["id"] = iv->id; mIv[group][inv] = mSys->getInverterByPos(i); - doLog = true; + result = true; } } @@ -393,197 +584,63 @@ class ZeroExport { mCfg->groups[group].inverters[inv].waitRebootAck = 0; } - // Next - mCfg->groups[group].state = zeroExportState::WAIT; - mCfg->groups[group].stateNext = zeroExportState::WAIT; + mCfg->groups[group].lastRefresh = *tsp; - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - mCfg->groups[group].lastRefresh = eTsp; - - return doLog; + return result; } /** groupWait - * pausiert die Gruppe + * pausiert die Gruppe bis die Wartezeit 60s seit dem lastRun abgelaufen ist. * @param group * @returns true/false */ - bool groupWait(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupWait"; - mLog["g"] = group; + bool groupWait(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupWait"; // Wait 60s - if (bTsp <= (mCfg->groups[group].lastRun + 60000UL)) { - return doLog; - } - - mCfg->groups[group].stateLast = zeroExportState::WAIT; - - doLog = true; - - // Next - if (mCfg->groups[group].state != mCfg->groups[group].stateNext) { - mCfg->groups[group].state = mCfg->groups[group].stateNext; - } else { - mCfg->groups[group].state = zeroExportState::WAITREFRESH; - mCfg->groups[group].stateNext = zeroExportState::WAITREFRESH; + if (*tsp <= (mCfg->groups[group].lastRun + 60000UL)) { + return false; } - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; + mCfg->groups[group].lastRun = *tsp; - mCfg->groups[group].lastRun = eTsp; - mCfg->groups[group].lastRefresh = eTsp; + *doLog = true; - return doLog; - } - - /** groupPublish - * - */ - bool groupPublish(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupPublish"; - mLog["g"] = group; - - mCfg->groups[group].stateLast = zeroExportState::PUBLISH; - - if (mMqtt->isConnected()) { - DynamicJsonDocument doc(512); - JsonObject obj = doc.to(); - - doLog = true; - - String gr; - - // Init - if (!mIsSubscribed) { - mIsSubscribed = true; - mMqtt->publish("zero/set/enabled", ((mCfg->enabled) ? dict[STR_TRUE] : dict[STR_FALSE]), false); - mMqtt->subscribe("zero/set/enabled", QOS_2); - - gr = "zero/set/groups/" + String(group) + "/enabled"; - mMqtt->publish(gr.c_str(), ((mCfg->groups[group].enabled) ? dict[STR_TRUE] : dict[STR_FALSE]) , false); - mMqtt->subscribe(gr.c_str(), QOS_2); - } - - mMqtt->publish("zero/state/enabled", ((mCfg->enabled) ? dict[STR_TRUE] : dict[STR_FALSE]), false); - gr = "zero/state/groups/" + String(group) + "/enabled"; - mMqtt->publish(gr.c_str(), ((mCfg->groups[group].enabled) ? dict[STR_TRUE] : dict[STR_FALSE]) , false); - - // if (mCfg->groups[group].publishPower) { - // mCfg->groups[group].publishPower = false; - obj["L1"] = mCfg->groups[group].pmPowerL1; - obj["L2"] = mCfg->groups[group].pmPowerL2; - obj["L3"] = mCfg->groups[group].pmPowerL3; - obj["Sum"] = mCfg->groups[group].pmPower; - - mMqtt->publish("zero/state/powermeter/P", doc.as().c_str(), false); - doc.clear(); - // } - - // if (mCfg->groups[group].pm_Publish_W) { - // mCfg->groups[group].pm_Publish_W = false; - // obj["todo"] = "true"; - // obj["L1"] = mCfg->groups[group].pm_P1; - // obj["L2"] = mCfg->groups[group].pm_P2; - // obj["L3"] = mCfg->groups[group].pm_P3; - // obj["Sum"] = mCfg->groups[group].pm_P; - // mMqtt->publish("zero/powermeter/W", doc.as().c_str(), false); - // doc.clear(); - // } - - for (uint8_t inv = 0; inv < ZEROEXPORT_GROUP_MAX_INVERTERS; inv++) { - } - } - - mCfg->groups[group].state = zeroExportState::WAITREFRESH; - mCfg->groups[group].stateNext = zeroExportState::WAITREFRESH; - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return doLog; + return true; } /** groupWaitRefresh - * pausiert die Gruppe + * pausiert die Gruppe bis die Wartezeit refresh seit dem lastRefresh abgelaufen ist. * @param group * @returns true/false */ - bool groupWaitRefresh(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupWaitRefresh"; - mLog["g"] = group; + bool groupWaitRefresh(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupWaitRefresh"; // Wait Refreshtime - if (bTsp <= (mCfg->groups[group].lastRefresh + (mCfg->groups[group].refresh * 1000UL))) { - return doLog; + if (*tsp <= (mCfg->groups[group].lastRefresh + (mCfg->groups[group].refresh * 1000UL))) { + return false; } - mCfg->groups[group].stateLast = zeroExportState::WAITREFRESH; - - doLog = true; - - // Next -#if defined(ZEROEXPORT_DEV_POWERMETER) - mCfg->groups[group].state = zeroExportState::GETPOWERMETER; - mCfg->groups[group].stateNext = zeroExportState::GETPOWERMETER; -#else - mCfg->groups[group].state = zeroExportState::GETINVERTERACKS; - mCfg->groups[group].stateNext = zeroExportState::GETINVERTERACKS; -#endif - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; + mCfg->groups[group].lastRun = *tsp; - mCfg->groups[group].lastRun = eTsp; + *doLog = true; - return doLog; + return true; } /** groupGetInverterAcks * aktualisiert die Gruppe mit den ACKs der Inverter für neue Limits * @param group * @returns true/false + * @todo siehe code */ - bool groupGetInverterAcks(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupGetInverterAcks"; - mLog["g"] = group; - - // Wait 250ms - if (mCfg->groups[group].stateLast == zeroExportState::GETINVERTERACKS) { - if (bTsp <= (mCfg->groups[group].lastRun + 250UL)) { - return doLog; - } - } + bool groupGetInverterAcks(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupGetInverterAcks"; - mCfg->groups[group].stateLast = zeroExportState::GETINVERTERACKS; + mCfg->groups[group].lastRun = *tsp; - doLog = true; + *doLog = true; // Wait Acks JsonArray logArr = mLog.createNestedArray("ix"); @@ -610,34 +667,24 @@ class ZeroExport { logObj["wL"] = mCfg->groups[group].inverters[inv].waitLimitAck; wait = true; } - // waitPowerAck - if (mCfg->groups[group].inverters[inv].waitPowerAck > 0) { - logObj["wP"] = mCfg->groups[group].inverters[inv].waitPowerAck; - wait = true; - } - // waitRebootAck - if (mCfg->groups[group].inverters[inv].waitRebootAck > 0) { - logObj["wR"] = mCfg->groups[group].inverters[inv].waitRebootAck; - wait = true; - } + // TODO: Remove -> is moved to groupSetPower + // // waitPowerAck + // if (mCfg->groups[group].inverters[inv].waitPowerAck > 0) { + // logObj["wP"] = mCfg->groups[group].inverters[inv].waitPowerAck; + // wait = true; + // } + // TODO: Remove -> is moved to groupSetReboot + // // waitRebootAck + // if (mCfg->groups[group].inverters[inv].waitRebootAck > 0) { + // logObj["wR"] = mCfg->groups[group].inverters[inv].waitRebootAck; + // wait = true; + // } } mLog["w"] = wait; - // Next - if (!wait) { - mCfg->groups[group].state = zeroExportState::GETINVERTERDATA; - mCfg->groups[group].stateNext = zeroExportState::GETINVERTERDATA; - } - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return doLog; + if (wait) return false; + return true; } /** groupGetInverterData @@ -645,16 +692,12 @@ class ZeroExport { * @param group * @returns true/false */ - bool groupGetInverterData(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); + bool groupGetInverterData(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupGetInverterData"; - mLog["t"] = "groupGetInverterData"; - mLog["g"] = group; + mCfg->groups[group].lastRun = *tsp; - mCfg->groups[group].stateLast = zeroExportState::GETINVERTERDATA; - - doLog = true; + *doLog = true; // Get Data JsonArray logArr = mLog.createNestedArray("ix"); @@ -682,23 +725,9 @@ class ZeroExport { float p = mIv[group][inv]->getChannelFieldValue(CH0, FLD_PAC, rec); // TODO: Save Hole Power für die Webanzeige - - // Get Limit - // uint16_t Limit = mIv[group][inv]->actPowerLimit; } - // Next - mCfg->groups[group].state = zeroExportState::BATTERYPROTECTION; - mCfg->groups[group].stateNext = zeroExportState::BATTERYPROTECTION; - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return doLog; + return true; } /** groupBatteryprotection @@ -706,18 +735,12 @@ class ZeroExport { * @param group * @returns true/false */ - bool groupBatteryprotection(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupBatteryprotection"; - mLog["g"] = group; + bool groupBatteryprotection(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupBatteryprotection"; - mCfg->groups[group].stateLast = zeroExportState::BATTERYPROTECTION; - mCfg->groups[group].state = zeroExportState::GETPOWERMETER; - mCfg->groups[group].stateNext = zeroExportState::GETPOWERMETER; + mCfg->groups[group].lastRun = *tsp; - doLog = true; + *doLog = true; // Protection if (mCfg->groups[group].battEnabled) { @@ -727,21 +750,21 @@ class ZeroExport { if (mCfg->groups[group].battVoltageOn <= mCfg->groups[group].battVoltageOff) { mCfg->groups[group].battSwitch = false; mLog["err"] = "Config - battVoltageOn(" + (String)mCfg->groups[group].battVoltageOn + ") <= battVoltageOff(" + (String)mCfg->groups[group].battVoltageOff + ")"; - return doLog; + return false; } // Config - parameter check if (mCfg->groups[group].battVoltageOn <= (mCfg->groups[group].battVoltageOff + 1)) { mCfg->groups[group].battSwitch = false; mLog["err"] = "Config - battVoltageOn(" + (String)mCfg->groups[group].battVoltageOn + ") <= battVoltageOff(" + (String)mCfg->groups[group].battVoltageOff + " + 1V)"; - return doLog; + return false; } // Config - parameter check if (mCfg->groups[group].battVoltageOn <= 22) { mCfg->groups[group].battSwitch = false; mLog["err"] = "Config - battVoltageOn(" + (String)mCfg->groups[group].battVoltageOn + ") <= 22V)"; - return doLog; + return false; } int8_t id = 0; @@ -859,31 +882,21 @@ class ZeroExport { mLog["sw"] = mCfg->groups[group].battSwitch; - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return doLog; + return true; } /** groupGetPowermeter * Holt die Daten vom Powermeter * @param group * @returns true/false + * @todo Eventuell muss am Ende geprüft werden ob die Daten vom Powermeter plausibel sind. */ - bool groupGetPowermeter(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupGetPowermeter"; - mLog["g"] = group; + bool groupGetPowermeter(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupGetPowermeter"; - mCfg->groups[group].stateLast = zeroExportState::GETPOWERMETER; + mCfg->groups[group].lastRun = *tsp; - doLog = true; + *doLog = true; bool result = false; result = mPowermeter.getData(mLog, group); @@ -892,31 +905,10 @@ class ZeroExport { (mCfg->groups[group].pmPower == 0) && (mCfg->groups[group].pmPower == 0) && (mCfg->groups[group].pmPower == 0)) { - return doLog; + return false; } - // TODO: eventuell muss hier geprüft werden ob die Daten vom Powermeter plausibel sind. - - // Next -#if defined(ZEROEXPORT_DEV_POWERMETER) - mCfg->groups[group].state = zeroExportState::PUBLISH; - mCfg->groups[group].stateNext = zeroExportState::PUBLISH; - mCfg->groups[group].lastRefresh = millis(); -#else - if (result) { - mCfg->groups[group].state = zeroExportState::CONTROLLER; - mCfg->groups[group].stateNext = zeroExportState::CONTROLLER; - } -#endif - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return doLog; + return true; } /** groupController @@ -924,16 +916,12 @@ class ZeroExport { * @param group * @returns true/false */ - bool groupController(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupController"; - mLog["g"] = group; + bool groupController(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupController"; - mCfg->groups[group].stateLast = zeroExportState::GETPOWERMETER; + mCfg->groups[group].lastRun = *tsp; - doLog = true; + *doLog = true; // Führungsgröße w in Watt float w = mCfg->groups[group].setPoint; @@ -967,19 +955,15 @@ class ZeroExport { (e1 < mCfg->groups[group].powerTolerance) && (e1 > -mCfg->groups[group].powerTolerance) && (e2 < mCfg->groups[group].powerTolerance) && (e2 > -mCfg->groups[group].powerTolerance) && (e3 < mCfg->groups[group].powerTolerance) && (e3 > -mCfg->groups[group].powerTolerance)) { - mCfg->groups[group].state = zeroExportState::WAITREFRESH; - mCfg->groups[group].stateNext = zeroExportState::WAITREFRESH; - mCfg->groups[group].lastRun = bTsp; mLog["tol"] = mCfg->groups[group].powerTolerance; - return doLog; + return false; } // Regler - // TODO: Regelparameter unter Advanced konfigurierbar? Aber erst wenn Regler komplett ingegriert. float Kp = mCfg->groups[group].Kp; float Ki = mCfg->groups[group].Ki; float Kd = mCfg->groups[group].Kd; - unsigned long Ta = bTsp - mCfg->groups[group].lastRefresh; + unsigned long Ta = *tsp - mCfg->groups[group].lastRefresh; mLog["Kp"] = Kp; mLog["Ki"] = Ki; mLog["Kd"] = Kd; @@ -1046,19 +1030,7 @@ class ZeroExport { mCfg->groups[group].grpPowerL2 = y2; mCfg->groups[group].grpPowerL3 = y3; - // Next - mCfg->groups[group].state = zeroExportState::PROGNOSE; - mCfg->groups[group].stateNext = zeroExportState::PROGNOSE; - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - mCfg->groups[group].lastRefresh = eTsp; - - return doLog; + return true; } /** groupPrognose @@ -1066,29 +1038,14 @@ class ZeroExport { * @param group * @returns true/false */ - bool groupPrognose(uint8_t group) { - bool result = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupPrognose"; - mLog["g"] = group; - - mCfg->groups[group].stateLast = zeroExportState::PROGNOSE; + bool groupPrognose(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupPrognose"; - result = true; + mCfg->groups[group].lastRun = *tsp; - // Next - mCfg->groups[group].state = zeroExportState::AUFTEILEN; - mCfg->groups[group].stateNext = zeroExportState::AUFTEILEN; + *doLog = true; - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return result; + return true; } /** groupAufteilen @@ -1096,22 +1053,29 @@ class ZeroExport { * @param group * @returns true/false */ - bool groupAufteilen(uint8_t group) { - bool result = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupAufteilen"; - mLog["g"] = group; + bool groupAufteilen(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupAufteilen"; - mCfg->groups[group].stateLast = zeroExportState::AUFTEILEN; + mCfg->groups[group].lastRun = *tsp; - result = true; + *doLog = true; float y = mCfg->groups[group].grpPower; float y1 = mCfg->groups[group].grpPowerL1; float y2 = mCfg->groups[group].grpPowerL2; float y3 = mCfg->groups[group].grpPowerL3; + // TDOD: nochmal durchdenken ... es muss für Sum und L1-3 sein + // uint16_t groupPmax = 0; + // for (uint8_t inv = 0; inv < ZEROEXPORT_GROUP_MAX_INVERTERS; inv++) { + // groupPmax = mCfg->groups[group].inverters[inv].limit; + // } + // int16_t groupPavail = mCfg->groups[group].powerMax - groupPmax; + // + // if ( y > groupPavail ) { + // y = groupPavail; + // } + mLog["y"] = y; mLog["y1"] = y1; mLog["y2"] = y2; @@ -1143,6 +1107,7 @@ class ZeroExport { // if (!cfgGroupInv->state) { // continue; // } + record_t<> *rec; rec = mIv[group][inv]->getRecordStruct(RealTimeRunData_Debug); cfgGroupInv->power = mIv[group][inv]->getChannelFieldValue(CH0, FLD_PAC, rec); @@ -1153,6 +1118,7 @@ class ZeroExport { ivId_Pmin[cfgGroupInv->target] = inv; // Hier kein return oder continue sonst dauerreboot } + if ((uint16_t)cfgGroupInv->power > ivPmax[cfgGroupInv->target]) { grpTarget[cfgGroupInv->target] = true; ivPmax[cfgGroupInv->target] = (uint16_t)cfgGroupInv->power; @@ -1236,37 +1202,25 @@ class ZeroExport { } } - // Next - mCfg->groups[group].state = zeroExportState::SETLIMIT; - mCfg->groups[group].stateNext = zeroExportState::SETLIMIT; - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return result; + return true; } - /** groupSetLimit + /** groupSetReboot * * @param group * @returns true/false */ - bool groupSetLimit(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); + bool groupSetReboot(uint8_t group, unsigned long *tsp, bool *doLog) { + bool result = true; - mLog["t"] = "groupSetLimit"; - mLog["g"] = group; + if (mCfg->debug) mLog["t"] = "groupSetReboot"; - mCfg->groups[group].stateLast = zeroExportState::SETLIMIT; + mCfg->groups[group].lastRun = *tsp; - // Set limit + JsonArray logArr = mLog.createNestedArray("ix"); for (uint8_t inv = 0; inv < ZEROEXPORT_GROUP_MAX_INVERTERS; inv++) { - mLog["i"] = inv; + JsonObject logObj = logArr.createNestedObject(); + logObj["i"] = inv; // Inverter not enabled -> ignore if (!mCfg->groups[group].inverters[inv].enabled) { @@ -1276,100 +1230,53 @@ class ZeroExport { if (mCfg->groups[group].inverters[inv].id <= 0) { continue; } - // Nothing todo - if (mCfg->groups[group].inverters[inv].limit == mCfg->groups[group].inverters[inv].limitNew) { + *doLog = true; + // Reset + if ((mCfg->groups[group].inverters[inv].doReboot) && (mCfg->groups[group].inverters[inv].waitRebootAck == 0)) { + result = false; + mCfg->groups[group].inverters[inv].doReboot = false; continue; } - // Abbruch weil Inverter nicht verfügbar - if (!mIv[group][inv]->isAvailable()) { + // Wait + if (mCfg->groups[group].inverters[inv].waitRebootAck > 0) { + result = false; continue; } - // Abbruch weil Inverter produziert nicht - if (!mIv[group][inv]->isProducing()) { + // Inverter nothing to do -> ignore + if (!mCfg->groups[group].inverters[inv].doReboot) { continue; } - // do - - // Set Power on/off - // TODO: testen - /* - if (mCfg->groups[group].inverters[inv].limitNew < mCfg->groups[group].inverters[inv].powerMin) { - if (mIv[group][inv]->isProducing()) { - mCfg->groups[group].inverters[inv].switch = false; - mCfg->groups[group].stateNext = zeroExportState::SETPOWER; - } - } else { - if (!mIv[group][inv]->isProducing()) { - mCfg->groups[group].inverters[inv].switch = true; - mCfg->groups[group].stateNext = zeroExportState::SETPOWER; - } - } - */ - - // Restriction LimitNew >= Pmin - if (mCfg->groups[group].inverters[inv].limitNew < mCfg->groups[group].inverters[inv].powerMin) { - mCfg->groups[group].inverters[inv].limitNew = mCfg->groups[group].inverters[inv].powerMin; - } - // Restriction LimitNew >= 2% - uint16_t power2proz = mIv[group][inv]->getMaxPower() / 100 * 2; - if (mCfg->groups[group].inverters[inv].limitNew < power2proz) { - mCfg->groups[group].inverters[inv].limitNew = power2proz; - } - - // Restriction LimitNew <= Pmax - if (mCfg->groups[group].inverters[inv].limitNew > mCfg->groups[group].inverters[inv].powerMax) { - mCfg->groups[group].inverters[inv].limitNew = mCfg->groups[group].inverters[inv].powerMax; - } - - // Reject limit if difference < 5 W - if ( - (mCfg->groups[group].inverters[inv].limitNew > (mCfg->groups[group].inverters[inv].limit + ZEROEXPORT_GROUP_WR_LIMIT_MIN_DIFF)) && - (mCfg->groups[group].inverters[inv].limitNew < (mCfg->groups[group].inverters[inv].limit - ZEROEXPORT_GROUP_WR_LIMIT_MIN_DIFF))) { - mLog["err"] = String("Diff < ") + String(ZEROEXPORT_GROUP_WR_LIMIT_MIN_DIFF) + String("W"); - return false; - } - - // Nothing todo - if (mCfg->groups[group].inverters[inv].limit == mCfg->groups[group].inverters[inv].limitNew) { + // Inverter not available -> ignore + if (!mIv[group][inv]->isAvailable()) { + result = false; + logObj["err"] = "is not Available"; continue; } - doLog = true; + result = false; + *doLog = true; - mCfg->groups[group].inverters[inv].limit = mCfg->groups[group].inverters[inv].limitNew; - mLog["zeL"] = (uint16_t)mCfg->groups[group].inverters[inv].limit; + logObj["act"] = String("reboot"); // wait for Ack - mCfg->groups[group].inverters[inv].waitLimitAck = 60; - mLog["wL"] = mCfg->groups[group].inverters[inv].waitLimitAck; + mCfg->groups[group].inverters[inv].waitRebootAck = 120; + logObj["wR"] = mCfg->groups[group].inverters[inv].waitRebootAck; // send Command DynamicJsonDocument doc(512); JsonObject obj = doc.to(); - obj["val"] = (uint16_t)mCfg->groups[group].inverters[inv].limit; obj["id"] = mCfg->groups[group].inverters[inv].id; obj["path"] = "ctrl"; - obj["cmd"] = "limit_nonpersistent_absolute"; + obj["cmd"] = "restart"; mApi->ctrlRequest(obj); - mLog["d"] = obj; + logObj["d"] = obj; } - // Next - mCfg->groups[group].state = zeroExportState::PUBLISH; - mCfg->groups[group].stateNext = zeroExportState::PUBLISH; - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return doLog; + return result; } /** groupSetPower @@ -1377,18 +1284,17 @@ class ZeroExport { * @param group * @returns true/false */ - bool groupSetPower(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); + bool groupSetPower(uint8_t group, unsigned long *tsp, bool *doLog) { + bool result = true; - mLog["t"] = "groupSetPower"; - mLog["g"] = group; + if (mCfg->debug) mLog["t"] = "groupSetPower"; - mCfg->groups[group].stateLast = zeroExportState::SETPOWER; + mCfg->groups[group].lastRun = *tsp; - // Set power + JsonArray logArr = mLog.createNestedArray("ix"); for (uint8_t inv = 0; inv < ZEROEXPORT_GROUP_MAX_INVERTERS; inv++) { - mLog["i"] = inv; + JsonObject logObj = logArr.createNestedObject(); + logObj["i"] = inv; // Inverter not enabled -> ignore if (!mCfg->groups[group].inverters[inv].enabled) { @@ -1399,123 +1305,261 @@ class ZeroExport { continue; } - // do - doLog = true; + if (mCfg->debug) *doLog = true; - // Abbruch weil Inverter nicht verfügbar + // Inverter not available -> ignore if (!mIv[group][inv]->isAvailable()) { - mLog["err"] = "is not Available"; + logObj["a"] = false; + result = false; continue; } - // Nothing todo - if (mCfg->groups[group].battSwitch == mIv[group][inv]->isProducing()) { - mLog["err"] = "battSwitch " + String(mCfg->groups[group].battSwitch) + " == isProducing()" + String(mIv[group][inv]->isProducing()); + // Reset + if ((mCfg->groups[group].inverters[inv].doPower != -1) && (mCfg->groups[group].inverters[inv].waitPowerAck == 0)) { + result = false; + mCfg->groups[group].inverters[inv].doPower = -1; + logObj["act"] = "done"; + continue; + } + + // Calculate + logObj["battSw"] = mCfg->groups[group].battSwitch; + logObj["ivL"] = mCfg->groups[group].inverters[inv].limitNew; + logObj["ivSw"] = mIv[group][inv]->isProducing(); + if ( + (mCfg->groups[group].battSwitch == true) && + (mCfg->groups[group].inverters[inv].limitNew > mCfg->groups[group].inverters[inv].powerMin) && + (mIv[group][inv]->isProducing() == false)) { + // On + mCfg->groups[group].inverters[inv].doPower = true; + logObj["act"] = "on"; + } + if ( + ( + (mCfg->groups[group].battSwitch == false) || + (mCfg->groups[group].inverters[inv].limitNew < mCfg->groups[group].inverters[inv].powerMin)) && + (mIv[group][inv]->isProducing() == true)) { + // Off + mCfg->groups[group].inverters[inv].doPower = false; + logObj["act"] = "off"; + } + + // Wait + if (mCfg->groups[group].inverters[inv].waitPowerAck > 0) { + logObj["w"] = mCfg->groups[group].inverters[inv].waitPowerAck; + result = false; + continue; + } + + // Nothing to do + if (mCfg->groups[group].inverters[inv].doPower == -1) { + logObj["act"] = "nothing to do"; continue; } - mLog["act"] = String("switch to: ") + String(mCfg->groups[group].battSwitch); + result = false; + *doLog = true; + + if (!mCfg->debug) logObj["act"] = mCfg->groups[group].inverters[inv].doPower; // wait for Ack mCfg->groups[group].inverters[inv].waitPowerAck = 120; - mLog["wP"] = mCfg->groups[group].inverters[inv].waitPowerAck; + logObj["wP"] = mCfg->groups[group].inverters[inv].waitPowerAck; // send Command DynamicJsonDocument doc(512); JsonObject obj = doc.to(); - obj["val"] = mCfg->groups[group].battSwitch; + obj["val"] = mCfg->groups[group].inverters[inv].doPower; obj["id"] = mCfg->groups[group].inverters[inv].id; obj["path"] = "ctrl"; obj["cmd"] = "power"; mApi->ctrlRequest(obj); - mLog["d"] = obj; + logObj["d"] = obj; } - // Next - mCfg->groups[group].state = zeroExportState::PUBLISH; - mCfg->groups[group].stateNext = zeroExportState::PUBLISH; - mCfg->groups[group].lastRefresh = millis(); - - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; - - mCfg->groups[group].lastRun = eTsp; - - return doLog; + return result; } - /** groupSetReboot + /** groupSetLimit * * @param group * @returns true/false */ - bool groupSetReboot(uint8_t group) { - bool doLog = false; - unsigned long bTsp = millis(); - - mLog["t"] = "groupSetReboot"; - mLog["g"] = group; + bool groupSetLimit(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupSetLimit"; - mCfg->groups[group].stateLast = zeroExportState::SETREBOOT; + mCfg->groups[group].lastRun = *tsp; - // Set reboot + // Set limit + JsonArray logArr = mLog.createNestedArray("ix"); for (uint8_t inv = 0; inv < ZEROEXPORT_GROUP_MAX_INVERTERS; inv++) { - mLog["i"] = inv; + JsonObject logObj = logArr.createNestedObject(); + logObj["i"] = inv; // Inverter not enabled -> ignore if (!mCfg->groups[group].inverters[inv].enabled) { continue; } + // Inverter not selected -> ignore if (mCfg->groups[group].inverters[inv].id <= 0) { continue; } + // if isOff -> Limit Pmin + if (!mIv[group][inv]->isProducing()) { + mCfg->groups[group].inverters[inv].limitNew = mCfg->groups[group].inverters[inv].powerMin; + } + + // Nothing todo + if (mCfg->groups[group].inverters[inv].limit == mCfg->groups[group].inverters[inv].limitNew) { + continue; + } + // Abbruch weil Inverter nicht verfügbar - // if (!mIv[group][inv]->isAvailable()) { - // continue; - // } + if (!mIv[group][inv]->isAvailable()) { + continue; + } + + // Abbruch weil Inverter produziert nicht + /// if (!mIv[group][inv]->isProducing()) { + /// continue; + /// } // do - doLog = true; - mLog["act"] = String("reboot"); + // Restriction LimitNew >= Pmin + if (mCfg->groups[group].inverters[inv].limitNew < mCfg->groups[group].inverters[inv].powerMin) { + mCfg->groups[group].inverters[inv].limitNew = mCfg->groups[group].inverters[inv].powerMin; + } + // Restriction LimitNew >= 2% + uint16_t power2proz = mIv[group][inv]->getMaxPower() / 100 * 2; + if (mCfg->groups[group].inverters[inv].limitNew < power2proz) { + mCfg->groups[group].inverters[inv].limitNew = power2proz; + } + + // Restriction LimitNew <= Pmax + if (mCfg->groups[group].inverters[inv].limitNew > mCfg->groups[group].inverters[inv].powerMax) { + mCfg->groups[group].inverters[inv].limitNew = mCfg->groups[group].inverters[inv].powerMax; + } + + // Reject limit if difference < 5 W + if ( + (mCfg->groups[group].inverters[inv].limitNew > (mCfg->groups[group].inverters[inv].powerMin + ZEROEXPORT_GROUP_WR_LIMIT_MIN_DIFF)) && + (mCfg->groups[group].inverters[inv].limitNew > (mCfg->groups[group].inverters[inv].limit + ZEROEXPORT_GROUP_WR_LIMIT_MIN_DIFF)) && + (mCfg->groups[group].inverters[inv].limitNew < (mCfg->groups[group].inverters[inv].limit - ZEROEXPORT_GROUP_WR_LIMIT_MIN_DIFF))) { + mLog["err"] = String("Diff < ") + String(ZEROEXPORT_GROUP_WR_LIMIT_MIN_DIFF) + String("W"); + return false; + } + + // Nothing todo + if (mCfg->groups[group].inverters[inv].limit == mCfg->groups[group].inverters[inv].limitNew) { + continue; + } + + *doLog = true; + + mCfg->groups[group].inverters[inv].limit = mCfg->groups[group].inverters[inv].limitNew; + mLog["zeL"] = (uint16_t)mCfg->groups[group].inverters[inv].limit; // wait for Ack - mCfg->groups[group].inverters[inv].waitRebootAck = 120; - mLog["wR"] = mCfg->groups[group].inverters[inv].waitRebootAck; + mCfg->groups[group].inverters[inv].waitLimitAck = 60; + mLog["wL"] = mCfg->groups[group].inverters[inv].waitLimitAck; // send Command DynamicJsonDocument doc(512); JsonObject obj = doc.to(); + obj["val"] = (uint16_t)mCfg->groups[group].inverters[inv].limit; obj["id"] = mCfg->groups[group].inverters[inv].id; obj["path"] = "ctrl"; - obj["cmd"] = "restart"; + obj["cmd"] = "limit_nonpersistent_absolute"; mApi->ctrlRequest(obj); mLog["d"] = obj; } - // Next - mCfg->groups[group].state = zeroExportState::PUBLISH; - mCfg->groups[group].stateNext = zeroExportState::PUBLISH; - mCfg->groups[group].lastRefresh = millis(); + return true; + } + + /** groupPublish + * + */ + bool groupPublish(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupPublish"; + + mCfg->groups[group].lastRun = *tsp; - unsigned long eTsp = millis(); - mLog["B"] = bTsp; - mLog["E"] = eTsp; - mLog["D"] = eTsp - bTsp; + *doLog = true; - mCfg->groups[group].lastRun = eTsp; + return true; - return doLog; + if (mMqtt->isConnected()) { + DynamicJsonDocument doc(512); + JsonObject obj = doc.to(); + + *doLog = true; + + // Init + if (!mIsSubscribed) { + mIsSubscribed = true; + mMqtt->publish("zero/set/enabled", ((mCfg->enabled) ? dict[STR_TRUE] : dict[STR_FALSE]), false); + mMqtt->subscribe("zero/set/enabled", QOS_2); + } + + mMqtt->publish("zero/state/enabled", ((mCfg->enabled) ? dict[STR_TRUE] : dict[STR_FALSE]), false); + + // if (mCfg->groups[group].publishPower) { + // mCfg->groups[group].publishPower = false; + obj["L1"] = mCfg->groups[group].pmPowerL1; + obj["L2"] = mCfg->groups[group].pmPowerL2; + obj["L3"] = mCfg->groups[group].pmPowerL3; + obj["Sum"] = mCfg->groups[group].pmPower; + mMqtt->publish("zero/state/powermeter/P", doc.as().c_str(), false); + doc.clear(); + // } + + // if (mCfg->groups[group].pm_Publish_W) { + // mCfg->groups[group].pm_Publish_W = false; + // obj["todo"] = "true"; + // obj["L1"] = mCfg->groups[group].pm_P1; + // obj["L2"] = mCfg->groups[group].pm_P2; + // obj["L2"] = mCfg->groups[group].pm_P3; + // obj["Sum"] = mCfg->groups[group].pm_P; + // mMqtt->publish("zero/powermeter/W", doc.as().c_str(), false); + // doc.clear(); + // } + + for (uint8_t inv = 0; inv < ZEROEXPORT_GROUP_MAX_INVERTERS; inv++) { + } + } + + return true; + } + + /** groupEmergency + * + * @param group + * @returns true/false + * @todo Hier ist noch keine Funktion + */ + bool groupEmergency(uint8_t group, unsigned long *tsp, bool *doLog) { + if (mCfg->debug) mLog["t"] = "groupEmergency"; + + mCfg->groups[group].lastRun = *tsp; + + *doLog = true; + + // if (!korrect) { + // do + // return; + // } + + return true; } /** sendLog - * Sendet das Log über Webserial und/oder MQTT + * Sendet den LogSpeicher über Webserial und/oder MQTT */ void sendLog(void) { if (mCfg->log_over_webserial) { @@ -1527,7 +1571,12 @@ class ZeroExport { mMqtt->publish("ze", mDocLog.as().c_str(), false); } } + } + /** clearLog + * Löscht den LogSpeicher + */ + void clearLog(void) { mDocLog.clear(); } diff --git a/src/web/RestApi.h b/src/web/RestApi.h index 0db1e3ee..045eb2d8 100644 --- a/src/web/RestApi.h +++ b/src/web/RestApi.h @@ -1158,7 +1158,6 @@ class RestApi { mConfig->plugin.zeroExport.groups[group].Ki = jsonIn[F("Ki")]; mConfig->plugin.zeroExport.groups[group].Kd = jsonIn[F("Kd")]; // Global - mConfig->plugin.zeroExport.groups[group].stateLast = zeroExportState::INIT; mConfig->plugin.zeroExport.groups[group].state = zeroExportState::INIT; mConfig->plugin.zeroExport.groups[group].stateNext = zeroExportState::INIT; mApp->saveSettings(false); // without reboot