@ -18,6 +18,8 @@
# endif
# endif
# include "../utils/dbg.h"
# include "../utils/dbg.h"
# include "../utils/ahoyTimer.h"
# include "../config/config.h"
# include <PubSubClient.h>
# include <PubSubClient.h>
# include "../defines.h"
# include "../defines.h"
# include "../hm/hmSystem.h"
# include "../hm/hmSystem.h"
@ -38,18 +40,34 @@ class mqtt {
~ mqtt ( ) { }
~ mqtt ( ) { }
void setup ( mqttConfig_t * cfg , const char * devname , HMSYSTEM * sys ) {
void setup ( mqttConfig_t * cfg , const char * devname , const char * version , HMSYSTEM * sys , uint32_t * utcT s ) {
DPRINTLN ( DBG_VERBOSE , F ( " mqtt.h:setup " ) ) ;
DPRINTLN ( DBG_VERBOSE , F ( " mqtt.h:setup " ) ) ;
mAddressSet = true ;
mAddressSet = true ;
mCfg = cfg ;
snprintf ( mDevName , DEVNAME_LEN , " %s " , devname ) ;
snprintf ( mDevName , DEVNAME_LEN , " %s " , devname ) ;
mCfg = cfg ;
mSys = sys ;
mSys = sys ;
mUtcTimestamp = utcTs ;
mClient - > setServer ( mCfg - > broker , mCfg - > port ) ;
mClient - > setServer ( mCfg - > broker , mCfg - > port ) ;
mClient - > setBufferSize ( MQTT_MAX_PACKET_SIZE ) ;
mClient - > setBufferSize ( MQTT_MAX_PACKET_SIZE ) ;
setCallback ( std : : bind ( & mqtt < HMSYSTEM > : : cbMqtt , this , std : : placeholders : : _1 , std : : placeholders : : _2 , std : : placeholders : : _3 ) ) ;
setCallback ( std : : bind ( & mqtt < HMSYSTEM > : : cbMqtt , this , std : : placeholders : : _1 , std : : placeholders : : _2 , std : : placeholders : : _3 ) ) ;
sendMsg ( " version " , version ) ;
sendMsg ( " device " , devname ) ;
sendMsg ( " uptime " , " 0 " ) ;
}
void loop ( ) {
if ( mAddressSet ) {
if ( ! mClient - > connected ( ) )
reconnect ( ) ;
mClient - > loop ( ) ;
if ( ! mSendList . empty ( ) )
sendIvData ( ) ;
}
}
}
void setCallback ( MQTT_CALLBACK_SIGNATURE ) {
void setCallback ( MQTT_CALLBACK_SIGNATURE ) {
@ -84,20 +102,15 @@ class mqtt {
return mClient - > connected ( ) ;
return mClient - > connected ( ) ;
}
}
void loop ( ) {
void payloadEventListener ( uint8_t cmd ) {
//DPRINT(F("m"));
mSendList . push ( cmd ) ;
if ( mAddressSet ) {
if ( ! mClient - > connected ( ) )
reconnect ( ) ;
mClient - > loop ( ) ;
}
}
}
uint32_t getTxCnt ( void ) {
uint32_t getTxCnt ( void ) {
return mTxCnt ;
return mTxCnt ;
}
}
void sendMqttDiscoveryConfig ( const char * topic , uint32_t invertval ) {
void sendMqttDiscoveryConfig ( const char * topic ) {
DPRINTLN ( DBG_VERBOSE , F ( " sendMqttDiscoveryConfig " ) ) ;
DPRINTLN ( DBG_VERBOSE , F ( " sendMqttDiscoveryConfig " ) ) ;
char stateTopic [ 64 ] , discoveryTopic [ 64 ] , buffer [ 512 ] , name [ 32 ] , uniq_id [ 32 ] ;
char stateTopic [ 64 ] , discoveryTopic [ 64 ] , buffer [ 512 ] , name [ 32 ] , uniq_id [ 32 ] ;
@ -131,7 +144,7 @@ class mqtt {
doc [ " unit_of_meas " ] = iv - > getUnit ( i , rec ) ;
doc [ " unit_of_meas " ] = iv - > getUnit ( i , rec ) ;
doc [ " uniq_id " ] = String ( iv - > serial . u64 , HEX ) + " _ " + uniq_id ;
doc [ " uniq_id " ] = String ( iv - > serial . u64 , HEX ) + " _ " + uniq_id ;
doc [ " dev " ] = deviceObj ;
doc [ " dev " ] = deviceObj ;
doc [ " exp_aft " ] = invertval + 5 ; // add 5 sec if connection is bad or ESP too slow @TODO: stimmt das wirklich als expire!?
doc [ " exp_aft " ] = MQTT_INTERVAL + 5 ; // add 5 sec if connection is bad or ESP too slow @TODO: stimmt das wirklich als expire!?
if ( devCls ! = NULL )
if ( devCls ! = NULL )
doc [ " dev_cla " ] = devCls ;
doc [ " dev_cla " ] = devCls ;
if ( stateCls ! = NULL )
if ( stateCls ! = NULL )
@ -148,7 +161,62 @@ class mqtt {
}
}
}
}
void sendIvData ( uint32_t mUtcTs , std : : queue < uint8_t > list ) {
private :
void reconnect ( void ) {
DPRINTLN ( DBG_DEBUG , F ( " mqtt.h:reconnect " ) ) ;
DPRINTLN ( DBG_DEBUG , F ( " MQTT mClient->_state " ) + String ( mClient - > state ( ) ) ) ;
# ifdef ESP8266
DPRINTLN ( DBG_DEBUG , F ( " WIFI mEspClient.status " ) + String ( mEspClient . status ( ) ) ) ;
# endif
boolean resub = false ;
if ( ! mClient - > connected ( ) & & ( millis ( ) - mLastReconnect ) > MQTT_RECONNECT_DELAY ) {
mLastReconnect = millis ( ) ;
if ( strlen ( mDevName ) > 0 ) {
// der Server und der Port müssen neu gesetzt werden,
// da ein MQTT_CONNECTION_LOST -3 die Werte zerstört hat.
mClient - > setServer ( mCfg - > broker , mCfg - > port ) ;
mClient - > setBufferSize ( MQTT_MAX_PACKET_SIZE ) ;
char lwt [ MQTT_TOPIC_LEN + 7 ] ; // "/uptime" --> + 7 byte
snprintf ( lwt , MQTT_TOPIC_LEN + 7 , " %s/uptime " , mCfg - > topic ) ;
if ( ( strlen ( mCfg - > user ) > 0 ) & & ( strlen ( mCfg - > pwd ) > 0 ) )
resub = mClient - > connect ( mDevName , mCfg - > user , mCfg - > pwd , lwt , 0 , false , " offline " ) ;
else
resub = mClient - > connect ( mDevName , lwt , 0 , false , " offline " ) ;
// ein Subscribe ist nur nach einem connect notwendig
if ( resub ) {
char topic [ MQTT_TOPIC_LEN + 13 ] ; // "/devcontrol/#" --> + 6 byte
// ToDo: "/devcontrol/#" is hardcoded
snprintf ( topic , MQTT_TOPIC_LEN + 13 , " %s/devcontrol/# " , mCfg - > topic ) ;
DPRINTLN ( DBG_INFO , F ( " subscribe to " ) + String ( topic ) ) ;
mClient - > subscribe ( topic ) ; // subscribe to mTopic + "/devcontrol/#"
}
}
}
}
const char * getFieldDeviceClass ( uint8_t fieldId ) {
uint8_t pos = 0 ;
for ( ; pos < DEVICE_CLS_ASSIGN_LIST_LEN ; pos + + ) {
if ( deviceFieldAssignment [ pos ] . fieldId = = fieldId )
break ;
}
return ( pos > = DEVICE_CLS_ASSIGN_LIST_LEN ) ? NULL : deviceClasses [ deviceFieldAssignment [ pos ] . deviceClsId ] ;
}
const char * getFieldStateClass ( uint8_t fieldId ) {
uint8_t pos = 0 ;
for ( ; pos < DEVICE_CLS_ASSIGN_LIST_LEN ; pos + + ) {
if ( deviceFieldAssignment [ pos ] . fieldId = = fieldId )
break ;
}
return ( pos > = DEVICE_CLS_ASSIGN_LIST_LEN ) ? NULL : stateClasses [ deviceFieldAssignment [ pos ] . stateClsId ] ;
}
void sendIvData ( ) {
isConnected ( true ) ; // really needed? See comment from HorstG-57 #176
isConnected ( true ) ; // really needed? See comment from HorstG-57 #176
char topic [ 32 + MAX_NAME_LENGTH ] , val [ 32 ] ;
char topic [ 32 + MAX_NAME_LENGTH ] , val [ 32 ] ;
float total [ 4 ] ;
float total [ 4 ] ;
@ -158,26 +226,26 @@ class mqtt {
sendMsg ( " uptime " , val ) ;
sendMsg ( " uptime " , val ) ;
if ( l ist. empty ( ) )
if ( mSendL ist. empty ( ) )
return ;
return ;
while ( ! l ist. empty ( ) ) {
while ( ! mSendL ist. empty ( ) ) {
memset ( total , 0 , sizeof ( float ) * 4 ) ;
memset ( total , 0 , sizeof ( float ) * 4 ) ;
for ( uint8_t id = 0 ; id < mSys - > getNumInverters ( ) ; id + + ) {
for ( uint8_t id = 0 ; id < mSys - > getNumInverters ( ) ; id + + ) {
Inverter < > * iv = mSys - > getInverterByPos ( id ) ;
Inverter < > * iv = mSys - > getInverterByPos ( id ) ;
if ( NULL = = iv )
if ( NULL = = iv )
continue ; // skip to next inverter
continue ; // skip to next inverter
record_t < > * rec = iv - > getRecordStruct ( l ist. front ( ) ) ;
record_t < > * rec = iv - > getRecordStruct ( mSendL ist. front ( ) ) ;
if ( l ist. front ( ) = = RealTimeRunData_Debug ) {
if ( mSendL ist. front ( ) = = RealTimeRunData_Debug ) {
// inverter status
// inverter status
uint8_t status = MQTT_STATUS_AVAIL_PROD ;
uint8_t status = MQTT_STATUS_AVAIL_PROD ;
if ( ! iv - > isAvailable ( mUtcTs , rec ) ) {
if ( ! iv - > isAvailable ( * mUtcTime stamp , rec ) ) {
status = MQTT_STATUS_NOT_AVAIL_NOT_PROD ;
status = MQTT_STATUS_NOT_AVAIL_NOT_PROD ;
totalIncomplete = true ;
totalIncomplete = true ;
}
}
else if ( ! iv - > isProducing ( mUtcTs , rec ) ) {
else if ( ! iv - > isProducing ( * mUtcTime stamp , rec ) ) {
if ( MQTT_STATUS_AVAIL_PROD = = status )
if ( MQTT_STATUS_AVAIL_PROD = = status )
status = MQTT_STATUS_AVAIL_NOT_PROD ;
status = MQTT_STATUS_AVAIL_NOT_PROD ;
}
}
@ -200,14 +268,14 @@ class mqtt {
}
}
// data
// data
if ( iv - > isAvailable ( mUtcTs , rec ) ) {
if ( iv - > isAvailable ( * mUtcTime stamp , rec ) ) {
for ( uint8_t i = 0 ; i < rec - > length ; i + + ) {
for ( uint8_t i = 0 ; i < rec - > length ; i + + ) {
snprintf ( topic , 32 + MAX_NAME_LENGTH , " %s/ch%d/%s " , iv - > name , rec - > assign [ i ] . ch , fields [ rec - > assign [ i ] . fieldId ] ) ;
snprintf ( topic , 32 + MAX_NAME_LENGTH , " %s/ch%d/%s " , iv - > name , rec - > assign [ i ] . ch , fields [ rec - > assign [ i ] . fieldId ] ) ;
snprintf ( val , 10 , " %.3f " , iv - > getValue ( i , rec ) ) ;
snprintf ( val , 10 , " %.3f " , iv - > getValue ( i , rec ) ) ;
sendMsg ( topic , val ) ;
sendMsg ( topic , val ) ;
// calculate total values for RealTimeRunData_Debug
// calculate total values for RealTimeRunData_Debug
if ( l ist. front ( ) = = RealTimeRunData_Debug ) {
if ( mSendL ist. front ( ) = = RealTimeRunData_Debug ) {
if ( CH0 = = rec - > assign [ i ] . ch ) {
if ( CH0 = = rec - > assign [ i ] . ch ) {
switch ( rec - > assign [ i ] . fieldId ) {
switch ( rec - > assign [ i ] . fieldId ) {
case FLD_PAC :
case FLD_PAC :
@ -231,7 +299,7 @@ class mqtt {
}
}
}
}
l ist. pop ( ) ; // remove from list once all inverters were processed
mSendL ist. pop ( ) ; // remove from list once all inverters were processed
if ( ( true = = sendTotal ) & & ( false = = totalIncomplete ) ) {
if ( ( true = = sendTotal ) & & ( false = = totalIncomplete ) ) {
uint8_t fieldId ;
uint8_t fieldId ;
@ -259,61 +327,6 @@ class mqtt {
}
}
}
}
private :
void reconnect ( void ) {
DPRINTLN ( DBG_DEBUG , F ( " mqtt.h:reconnect " ) ) ;
DPRINTLN ( DBG_DEBUG , F ( " MQTT mClient->_state " ) + String ( mClient - > state ( ) ) ) ;
# ifdef ESP8266
DPRINTLN ( DBG_DEBUG , F ( " WIFI mEspClient.status " ) + String ( mEspClient . status ( ) ) ) ;
# endif
boolean resub = false ;
if ( ! mClient - > connected ( ) & & ( millis ( ) - mLastReconnect ) > MQTT_RECONNECT_DELAY ) {
mLastReconnect = millis ( ) ;
if ( strlen ( mDevName ) > 0 ) {
// der Server und der Port müssen neu gesetzt werden,
// da ein MQTT_CONNECTION_LOST -3 die Werte zerstört hat.
mClient - > setServer ( mCfg - > broker , mCfg - > port ) ;
mClient - > setBufferSize ( MQTT_MAX_PACKET_SIZE ) ;
char lwt [ MQTT_TOPIC_LEN + 7 ] ; // "/uptime" --> + 7 byte
snprintf ( lwt , MQTT_TOPIC_LEN + 7 , " %s/uptime " , mCfg - > topic ) ;
if ( ( strlen ( mCfg - > user ) > 0 ) & & ( strlen ( mCfg - > pwd ) > 0 ) )
resub = mClient - > connect ( mDevName , mCfg - > user , mCfg - > pwd , lwt , 0 , false , " offline " ) ;
else
resub = mClient - > connect ( mDevName , lwt , 0 , false , " offline " ) ;
// ein Subscribe ist nur nach einem connect notwendig
if ( resub ) {
char topic [ MQTT_TOPIC_LEN + 13 ] ; // "/devcontrol/#" --> + 6 byte
// ToDo: "/devcontrol/#" is hardcoded
snprintf ( topic , MQTT_TOPIC_LEN + 13 , " %s/devcontrol/# " , mCfg - > topic ) ;
DPRINTLN ( DBG_INFO , F ( " subscribe to " ) + String ( topic ) ) ;
mClient - > subscribe ( topic ) ; // subscribe to mTopic + "/devcontrol/#"
}
}
}
}
const char * getFieldDeviceClass ( uint8_t fieldId ) {
uint8_t pos = 0 ;
for ( ; pos < DEVICE_CLS_ASSIGN_LIST_LEN ; pos + + ) {
if ( deviceFieldAssignment [ pos ] . fieldId = = fieldId )
break ;
}
return ( pos > = DEVICE_CLS_ASSIGN_LIST_LEN ) ? NULL : deviceClasses [ deviceFieldAssignment [ pos ] . deviceClsId ] ;
}
const char * getFieldStateClass ( uint8_t fieldId ) {
uint8_t pos = 0 ;
for ( ; pos < DEVICE_CLS_ASSIGN_LIST_LEN ; pos + + ) {
if ( deviceFieldAssignment [ pos ] . fieldId = = fieldId )
break ;
}
return ( pos > = DEVICE_CLS_ASSIGN_LIST_LEN ) ? NULL : stateClasses [ deviceFieldAssignment [ pos ] . stateClsId ] ;
}
void cbMqtt ( char * topic , byte * payload , unsigned int length ) {
void cbMqtt ( char * topic , byte * payload , unsigned int length ) {
// callback handling on subscribed devcontrol topic
// callback handling on subscribed devcontrol topic
DPRINTLN ( DBG_INFO , F ( " cbMqtt " ) ) ;
DPRINTLN ( DBG_INFO , F ( " cbMqtt " ) ) ;
@ -420,12 +433,14 @@ class mqtt {
WiFiClient mEspClient ;
WiFiClient mEspClient ;
PubSubClient * mClient ;
PubSubClient * mClient ;
HMSYSTEM * mSys ;
HMSYSTEM * mSys ;
uint32_t * mUtcTimestamp ;
bool mAddressSet ;
bool mAddressSet ;
mqttConfig_t * mCfg ;
mqttConfig_t * mCfg ;
char mDevName [ DEVNAME_LEN ] ;
char mDevName [ DEVNAME_LEN ] ;
uint32_t mLastReconnect ;
uint32_t mLastReconnect ;
uint32_t mTxCnt ;
uint32_t mTxCnt ;
std : : queue < uint8_t > mSendList ;
} ;
} ;
# endif /*__MQTT_H_*/
# endif /*__MQTT_H_*/