mirror of https://github.com/arendst/Tasmota.git
Merge pull request #8191 from pcdiem/device-groups-7
Add DevGroupStatus command, Include member add/remove logging
This commit is contained in:
commit
908b1113b9
|
@ -297,6 +297,7 @@
|
||||||
#define D_CMND_DEVGROUP_NAME "DevGroupName"
|
#define D_CMND_DEVGROUP_NAME "DevGroupName"
|
||||||
#define D_CMND_DEVGROUP_SEND "DevGroupSend"
|
#define D_CMND_DEVGROUP_SEND "DevGroupSend"
|
||||||
#define D_CMND_DEVGROUP_SHARE "DevGroupShare"
|
#define D_CMND_DEVGROUP_SHARE "DevGroupShare"
|
||||||
|
#define D_CMND_DEVGROUPSTATUS "DevGroupStatus"
|
||||||
#define D_CMND_SERIALSEND "SerialSend"
|
#define D_CMND_SERIALSEND "SerialSend"
|
||||||
#define D_CMND_SERIALDELIMITER "SerialDelimiter"
|
#define D_CMND_SERIALDELIMITER "SerialDelimiter"
|
||||||
#define D_CMND_BAUDRATE "Baudrate"
|
#define D_CMND_BAUDRATE "Baudrate"
|
||||||
|
|
|
@ -36,7 +36,7 @@ const char kTasmotaCommands[] PROGMEM = "|" // No prefix
|
||||||
#ifdef USE_DEVICE_GROUPS_SEND
|
#ifdef USE_DEVICE_GROUPS_SEND
|
||||||
D_CMND_DEVGROUP_SEND "|"
|
D_CMND_DEVGROUP_SEND "|"
|
||||||
#endif // USE_DEVICE_GROUPS_SEND
|
#endif // USE_DEVICE_GROUPS_SEND
|
||||||
D_CMND_DEVGROUP_SHARE "|"
|
D_CMND_DEVGROUP_SHARE "|" D_CMND_DEVGROUPSTATUS "|"
|
||||||
#endif // USE_DEVICE_GROUPS
|
#endif // USE_DEVICE_GROUPS
|
||||||
D_CMND_SENSOR "|" D_CMND_DRIVER;
|
D_CMND_SENSOR "|" D_CMND_DRIVER;
|
||||||
|
|
||||||
|
@ -59,7 +59,7 @@ void (* const TasmotaCommand[])(void) PROGMEM = {
|
||||||
#ifdef USE_DEVICE_GROUPS_SEND
|
#ifdef USE_DEVICE_GROUPS_SEND
|
||||||
&CmndDevGroupSend,
|
&CmndDevGroupSend,
|
||||||
#endif // USE_DEVICE_GROUPS_SEND
|
#endif // USE_DEVICE_GROUPS_SEND
|
||||||
&CmndDevGroupShare,
|
&CmndDevGroupShare, &CmndDevGroupStatus,
|
||||||
#endif // USE_DEVICE_GROUPS
|
#endif // USE_DEVICE_GROUPS
|
||||||
&CmndSensor, &CmndDriver };
|
&CmndSensor, &CmndDriver };
|
||||||
|
|
||||||
|
@ -1793,6 +1793,11 @@ void CmndDevGroupShare(void)
|
||||||
Settings.device_group_share_out = parm[1];
|
Settings.device_group_share_out = parm[1];
|
||||||
Response_P(PSTR("{\"" D_CMND_DEVGROUP_SHARE "\":{\"In\":\"%X\",\"Out\":\"%X\"}}"), Settings.device_group_share_in, Settings.device_group_share_out);
|
Response_P(PSTR("{\"" D_CMND_DEVGROUP_SHARE "\":{\"In\":\"%X\",\"Out\":\"%X\"}}"), Settings.device_group_share_in, Settings.device_group_share_out);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void CmndDevGroupStatus(void)
|
||||||
|
{
|
||||||
|
DeviceGroupStatus((XdrvMailbox.usridx ? XdrvMailbox.index - 1 : 0));
|
||||||
|
}
|
||||||
#endif // USE_DEVICE_GROUPS
|
#endif // USE_DEVICE_GROUPS
|
||||||
|
|
||||||
void CmndSensor(void)
|
void CmndSensor(void)
|
||||||
|
|
|
@ -34,6 +34,7 @@ struct device_group_member {
|
||||||
IPAddress ip_address;
|
IPAddress ip_address;
|
||||||
uint16_t received_sequence;
|
uint16_t received_sequence;
|
||||||
uint16_t acked_sequence;
|
uint16_t acked_sequence;
|
||||||
|
uint32_t unicast_count;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct device_group {
|
struct device_group {
|
||||||
|
@ -48,7 +49,6 @@ struct device_group {
|
||||||
bool local;
|
bool local;
|
||||||
char group_name[TOPSZ];
|
char group_name[TOPSZ];
|
||||||
char message[128];
|
char message[128];
|
||||||
uint8_t group_member_count;
|
|
||||||
struct device_group_member * device_group_members;
|
struct device_group_member * device_group_members;
|
||||||
#ifdef USE_DEVICE_GROUPS_SEND
|
#ifdef USE_DEVICE_GROUPS_SEND
|
||||||
uint8_t values_8bit[DGR_ITEM_LAST_8BIT];
|
uint8_t values_8bit[DGR_ITEM_LAST_8BIT];
|
||||||
|
@ -77,7 +77,7 @@ void DeviceGroupsInit(void)
|
||||||
// Initialize the device information for each device group.
|
// Initialize the device information for each device group.
|
||||||
device_groups = (struct device_group *)calloc(device_group_count, sizeof(struct device_group));
|
device_groups = (struct device_group *)calloc(device_group_count, sizeof(struct device_group));
|
||||||
if (device_groups == nullptr) {
|
if (device_groups == nullptr) {
|
||||||
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: error allocating %u-element device group array"), device_group_count);
|
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: Error allocating %u-element device group array"), device_group_count);
|
||||||
device_groups_initialization_failed = true;
|
device_groups_initialization_failed = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -169,7 +169,7 @@ void SendDeviceGroupPacket(IPAddress ip, char * packet, int len, const char * la
|
||||||
}
|
}
|
||||||
delay(10);
|
delay(10);
|
||||||
}
|
}
|
||||||
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: error sending %s packet"), label);
|
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: Error sending %s packet"), label);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _SendDeviceGroupMessage(uint8_t device_group_index, DevGroupMessageType message_type, ...)
|
void _SendDeviceGroupMessage(uint8_t device_group_index, DevGroupMessageType message_type, ...)
|
||||||
|
@ -184,7 +184,7 @@ void _SendDeviceGroupMessage(uint8_t device_group_index, DevGroupMessageType mes
|
||||||
if (processing_remote_device_message && message_type != DGR_MSGTYPE_UPDATE_COMMAND) return;
|
if (processing_remote_device_message && message_type != DGR_MSGTYPE_UPDATE_COMMAND) return;
|
||||||
|
|
||||||
// Get a pointer to the device information for this device.
|
// Get a pointer to the device information for this device.
|
||||||
device_group * device_group = &device_groups[device_group_index];
|
struct device_group * device_group = &device_groups[device_group_index];
|
||||||
|
|
||||||
// If we're still sending initial status requests, ignore this request.
|
// If we're still sending initial status requests, ignore this request.
|
||||||
if (device_group->initial_status_requests_remaining) return;
|
if (device_group->initial_status_requests_remaining) return;
|
||||||
|
@ -551,7 +551,7 @@ void _SendDeviceGroupMessage(uint8_t device_group_index, DevGroupMessageType mes
|
||||||
|
|
||||||
// Multicast the packet.
|
// Multicast the packet.
|
||||||
#ifdef DEVICE_GROUPS_DEBUG
|
#ifdef DEVICE_GROUPS_DEBUG
|
||||||
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: sending %u-byte device group %s packet via multicast, sequence=%u"), device_group->message_length, device_group->group_name, device_group->message[device_group->message_header_length] | device_group->message[device_group->message_header_length + 1] << 8);
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Sending %u-byte device group %s packet via multicast, sequence=%u"), device_group->message_length, device_group->group_name, device_group->message[device_group->message_header_length] | device_group->message[device_group->message_header_length + 1] << 8);
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
#endif // DEVICE_GROUPS_DEBUG
|
||||||
SendDeviceGroupPacket(IPAddress(0,0,0,0), device_group->message, device_group->message_length, PSTR("Multicast"));
|
SendDeviceGroupPacket(IPAddress(0,0,0,0), device_group->message, device_group->message_length, PSTR("Multicast"));
|
||||||
|
|
||||||
|
@ -598,12 +598,10 @@ void ProcessDeviceGroupMessage(char * packet, int packet_length)
|
||||||
if (!device_group_member) {
|
if (!device_group_member) {
|
||||||
device_group_member = (struct device_group_member *)calloc(1, sizeof(struct device_group_member));
|
device_group_member = (struct device_group_member *)calloc(1, sizeof(struct device_group_member));
|
||||||
if (device_group_member == nullptr) {
|
if (device_group_member == nullptr) {
|
||||||
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: error allocating device group member block"));
|
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: Error allocating device group member block"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
#ifdef DEVICE_GROUPS_DEBUG
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Adding member %s"), IPAddressToString(remote_ip));
|
||||||
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: adding member %s (%p)"), IPAddressToString(remote_ip), device_group_member);
|
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
|
||||||
device_group_member->ip_address = remote_ip;
|
device_group_member->ip_address = remote_ip;
|
||||||
*flink = device_group_member;
|
*flink = device_group_member;
|
||||||
break;
|
break;
|
||||||
|
@ -720,7 +718,7 @@ void ProcessDeviceGroupMessage(char * packet, int packet_length)
|
||||||
case DGR_ITEM_LIGHT_CHANNELS:
|
case DGR_ITEM_LIGHT_CHANNELS:
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: ********** invalid item=%u received from device group %s member %s"), item, device_group->group_name, IPAddressToString(remote_ip));
|
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: ********** Invalid item=%u received from device group %s member %s"), item, device_group->group_name, IPAddressToString(remote_ip));
|
||||||
}
|
}
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
#endif // DEVICE_GROUPS_DEBUG
|
||||||
|
|
||||||
|
@ -802,13 +800,28 @@ void ProcessDeviceGroupMessage(char * packet, int packet_length)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
badmsg:
|
badmsg:
|
||||||
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: malformed message received from %s"), IPAddressToString(remote_ip));
|
AddLog_P2(LOG_LEVEL_ERROR, PSTR("DGR: Malformed message received from %s"), IPAddressToString(remote_ip));
|
||||||
#ifdef DEVICE_GROUPS_DEBUG
|
#ifdef DEVICE_GROUPS_DEBUG
|
||||||
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("packet_length=%u, offset=%u"), packet_length, message_ptr - packet);
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("packet_length=%u, offset=%u"), packet_length, message_ptr - packet);
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
#endif // DEVICE_GROUPS_DEBUG
|
||||||
processing_remote_device_message = false;
|
processing_remote_device_message = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void DeviceGroupStatus(uint8_t device_group_index)
|
||||||
|
{
|
||||||
|
if (Settings.flag4.device_groups_enabled && device_group_index < device_group_count) {
|
||||||
|
char buffer[1024];
|
||||||
|
int member_count = 0;
|
||||||
|
struct device_group * device_group = &device_groups[device_group_index];
|
||||||
|
buffer[0] = buffer[1] = 0;
|
||||||
|
for (struct device_group_member * device_group_member = device_group->device_group_members; device_group_member; device_group_member = device_group_member->flink) {
|
||||||
|
snprintf(buffer, sizeof(buffer), PSTR("%s,{\"IPAddress\":\"%s\",\"ResendCount\":%u,\"LastRcvdSeq\":%u,\"LastAckedSeq\":%u}"), buffer, IPAddressToString(device_group_member->ip_address), device_group_member->unicast_count, device_group_member->received_sequence, device_group_member->acked_sequence);
|
||||||
|
member_count++;
|
||||||
|
}
|
||||||
|
Response_P(PSTR("{\"" D_CMND_DEVGROUPSTATUS "\":{\"Index\":%u,\"GroupName\":\"%s\",\"MessageSeq\":%u,\"MemberCount\":%d,\"Members\":[%s]}"), device_group_index, device_group->group_name, outgoing_sequence, member_count, &buffer[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void DeviceGroupsLoop(void)
|
void DeviceGroupsLoop(void)
|
||||||
{
|
{
|
||||||
if (!Settings.flag4.device_groups_enabled) return;
|
if (!Settings.flag4.device_groups_enabled) return;
|
||||||
|
@ -822,16 +835,16 @@ void DeviceGroupsLoop(void)
|
||||||
if (!device_groups_initialized) DeviceGroupsInit();
|
if (!device_groups_initialized) DeviceGroupsInit();
|
||||||
if (device_groups_initialization_failed) return;
|
if (device_groups_initialization_failed) return;
|
||||||
|
|
||||||
// Load the status request message for all device groups. This message will be multicast 5
|
// Load the status request message for all device groups. This message will be multicast 10
|
||||||
// times.
|
// times at 200ms intervals.
|
||||||
next_check_time = now + 3000;
|
next_check_time = now + 3000;
|
||||||
for (uint32_t device_group_index = 0; device_group_index < device_group_count; device_group_index++) {
|
for (uint32_t device_group_index = 0; device_group_index < device_group_count; device_group_index++) {
|
||||||
device_group * device_group = &device_groups[device_group_index];
|
device_group * device_group = &device_groups[device_group_index];
|
||||||
device_group->message_length = BeginDeviceGroupMessage(device_group, DGR_FLAG_RESET | DGR_FLAG_STATUS_REQUEST) - device_group->message;
|
device_group->message_length = BeginDeviceGroupMessage(device_group, DGR_FLAG_RESET | DGR_FLAG_STATUS_REQUEST) - device_group->message;
|
||||||
device_group->initial_status_requests_remaining = 5;
|
device_group->initial_status_requests_remaining = 10;
|
||||||
device_group->next_ack_check_time = next_check_time;
|
device_group->next_ack_check_time = next_check_time;
|
||||||
}
|
}
|
||||||
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: (Re)discovering device groups"));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (device_groups_initialization_failed) return;
|
if (device_groups_initialization_failed) return;
|
||||||
|
@ -856,15 +869,15 @@ AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Ckecking next_check_time=%u, now=%u"), nex
|
||||||
if (device_group->initial_status_requests_remaining) {
|
if (device_group->initial_status_requests_remaining) {
|
||||||
if (--device_group->initial_status_requests_remaining) {
|
if (--device_group->initial_status_requests_remaining) {
|
||||||
#ifdef DEVICE_GROUPS_DEBUG
|
#ifdef DEVICE_GROUPS_DEBUG
|
||||||
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: sending initial status request for group %s"), device_group->group_name);
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Sending initial status request for group %s"), device_group->group_name);
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
#endif // DEVICE_GROUPS_DEBUG
|
||||||
SendDeviceGroupPacket(IPAddress(0,0,0,0), device_group->message, device_group->message_length, PSTR("Initial"));
|
SendDeviceGroupPacket(IPAddress(0,0,0,0), device_group->message, device_group->message_length, PSTR("Initial"));
|
||||||
device_group->message[device_group->message_header_length + 2] = DGR_FLAG_STATUS_REQUEST; // The reset flag is on only for the first packet - turn it off now
|
device_group->message[device_group->message_header_length + 2] = DGR_FLAG_STATUS_REQUEST; // The reset flag is on only for the first packet - turn it off now
|
||||||
device_group->next_ack_check_time = now + 200;
|
device_group->next_ack_check_time = now + 200;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we've sent the initial status request message 5 times, send our status to all
|
// If we've sent the initial status request message the set number of times, send our
|
||||||
// the members.
|
// status to all the members.
|
||||||
else {
|
else {
|
||||||
device_group->next_ack_check_time = 0;
|
device_group->next_ack_check_time = 0;
|
||||||
_SendDeviceGroupMessage(device_group_index, DGR_MSGTYP_FULL_STATUS);
|
_SendDeviceGroupMessage(device_group_index, DGR_MSGTYP_FULL_STATUS);
|
||||||
|
@ -874,7 +887,7 @@ AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Ckecking next_check_time=%u, now=%u"), nex
|
||||||
// If we're done initializing, iterate through the group memebers, ...
|
// If we're done initializing, iterate through the group memebers, ...
|
||||||
else {
|
else {
|
||||||
#ifdef DEVICE_GROUPS_DEBUG
|
#ifdef DEVICE_GROUPS_DEBUG
|
||||||
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: checking for ack's"));
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Checking for ack's"));
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
#endif // DEVICE_GROUPS_DEBUG
|
||||||
bool acked = true;
|
bool acked = true;
|
||||||
struct device_group_member ** flink = &device_group->device_group_members;
|
struct device_group_member ** flink = &device_group->device_group_members;
|
||||||
|
@ -887,9 +900,7 @@ AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Ckecking next_check_time=%u, now=%u"), nex
|
||||||
// If we haven't receive an ack from this member in DGR_MEMBER_TIMEOUT ms, assume
|
// If we haven't receive an ack from this member in DGR_MEMBER_TIMEOUT ms, assume
|
||||||
// they're offline and remove them from the group.
|
// they're offline and remove them from the group.
|
||||||
if (device_group->member_timeout_time < now) {
|
if (device_group->member_timeout_time < now) {
|
||||||
#ifdef DEVICE_GROUPS_DEBUG
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Removing member %s"), IPAddressToString(device_group_member->ip_address));
|
||||||
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: removing member %s (%p)"), IPAddressToString(device_group_member->ip_address), device_group_member);
|
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
|
||||||
*flink = device_group_member->flink;
|
*flink = device_group_member->flink;
|
||||||
free(device_group_member);
|
free(device_group_member);
|
||||||
}
|
}
|
||||||
|
@ -897,9 +908,10 @@ AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Ckecking next_check_time=%u, now=%u"), nex
|
||||||
// Otherwise, unicast the last message directly to this member.
|
// Otherwise, unicast the last message directly to this member.
|
||||||
else {
|
else {
|
||||||
#ifdef DEVICE_GROUPS_DEBUG
|
#ifdef DEVICE_GROUPS_DEBUG
|
||||||
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: sending %u-byte device group %s packet via unicast to %s, sequence %u, last message acked=%u"), device_group->message_length, device_group->group_name, IPAddressToString(device_group_member->ip_address), outgoing_sequence, device_group_member->acked_sequence);
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Sending %u-byte device group %s packet via unicast to %s, sequence %u, last message acked=%u"), device_group->message_length, device_group->group_name, IPAddressToString(device_group_member->ip_address), outgoing_sequence, device_group_member->acked_sequence);
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
#endif // DEVICE_GROUPS_DEBUG
|
||||||
SendDeviceGroupPacket(device_group_member->ip_address, device_group->message, device_group->message_length, PSTR("Unicast"));
|
SendDeviceGroupPacket(device_group_member->ip_address, device_group->message, device_group->message_length, PSTR("Unicast"));
|
||||||
|
device_group_member->unicast_count++;
|
||||||
acked = false;
|
acked = false;
|
||||||
flink = &device_group_member->flink;
|
flink = &device_group_member->flink;
|
||||||
}
|
}
|
||||||
|
@ -940,7 +952,7 @@ AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Ckecking next_check_time=%u, now=%u"), nex
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
#endif // DEVICE_GROUPS_DEBUG
|
||||||
if (device_group->next_announcement_time <= now) {
|
if (device_group->next_announcement_time <= now) {
|
||||||
#ifdef DEVICE_GROUPS_DEBUG
|
#ifdef DEVICE_GROUPS_DEBUG
|
||||||
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: sending device group %s announcement"), device_group->group_name);
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("DGR: Sending device group %s announcement"), device_group->group_name);
|
||||||
#endif // DEVICE_GROUPS_DEBUG
|
#endif // DEVICE_GROUPS_DEBUG
|
||||||
SendDeviceGroupPacket(IPAddress(0,0,0,0), device_group->message, BeginDeviceGroupMessage(device_group, DGR_FLAG_ANNOUNCEMENT, true) - device_group->message, PSTR("Announcement"));
|
SendDeviceGroupPacket(IPAddress(0,0,0,0), device_group->message, BeginDeviceGroupMessage(device_group, DGR_FLAG_ANNOUNCEMENT, true) - device_group->message, PSTR("Announcement"));
|
||||||
device_group->next_announcement_time = now + DGR_ANNOUNCEMENT_INTERVAL + random(10000);
|
device_group->next_announcement_time = now + DGR_ANNOUNCEMENT_INTERVAL + random(10000);
|
||||||
|
|
Loading…
Reference in New Issue