mirror of
https://github.com/meshtastic/firmware.git
synced 2025-12-28 05:30:30 +00:00
* Update Favorite Node Message Options to unify against other screens * Rebuild Horizontal Battery, Resolve overlap concerns * Update positioning on Message frame and fix drawCommonHeader overlay * Beginnings of creating isHighResolution bool * Fixup determineResolution() * Implement isHighResolution in place of SCREEN_WIDTH > 128 checks * Line Spacing bound to isHighResolution * Analog Clock for all * Add AM/PM to Analog Clock if isHighResolution and not TWatch * Simple Menu Queue, and add time menu * Fix prompt string for 12/24 hour picker * More menu banners into functions * Fix Action Menu on Home frame * Correct pop-up calculation size and continue to leverage isHighResolution * Move menu bits to MenuHandler * Plumb in the digital/analog picker * Correct Clock Face Picker title * Clock picker fixes * Migrate the rest of the menus to MenuHandler.* * Add compass menu and needle point option * Minor fix for compass point menu * Correct Home menu into typical format * Fix emoji bounce, overlap, and missing commonHeader * Sanitize long_names and removed unused variables * Slightly better sanitizeString variation * Resolved apostrophe being shown as upside down question mark * Gotta keep height and width in expected order * Remove Second Hand for Analog Clock on EInk displays * Fix Clock menu option decision tree * Improvements to Eink Navigation * Pause Banner for Eink moved to bottom * Updated working for 12-/24-hour menu and Added US/Arizona to timezone picker * Add Adhoc Ping and resolve error with std::string sanitized * Hide quick toggle as option is available within Action Menu, commented out for the moment * Remove old battery icon and option, use drawCommonHeader throughout, re-add battery to Clock frames * fix misc build warnings. NFC * Update Analog Clock on EInk to show more digits * Establish Action Menu on all node list screens, add NodeDB reset (with confirmation) option * Add Toggle Backlight for EInk Displays * Suppress action screen Full refresh for Eink * Adjust drawBluetoothConnectedIcon on TWatch * Maintain clock frame when switching between Clock Faces * Move modules beyond the clock in navigation * addressed the conflicts, and changed target branch to 2.7-MiscFixes-Week1 * cleanup, cheers * Add AM/PM to low resolution clock also * Small adjustments to AM/PM replacement across various devices * Resolve dangling pointer issues with sanitize code * Update comments for Screen.cpp related to module load change * Trunk runs * Update message caching to correct aged timestamp * Menu wording adjustments * Time Format wording * Use all the rows on EInk since with autohide the navigation bar * Finalize Time Format picker word change * Retired drawFunctionOverlay code No longer being used * Actually honor the points-north setting * Trunk * Compressed action list * Update no-op showOverlayBanner function * trunk * Correct T_Watch_S3 specific line * Autosized Action menu per screen * Finalize Autosized Action menu per screen * Unify Message Titles * Reorder Timezones to match expectations * Adjust text location for pop-ups * Revert "Actually honor the points-north setting" This reverts commit20988aa4fa. * Make NodeDB sort its internal vector when lastheard is updated. Don't sort in NodeListRenderer * Update src/graphics/draw/NodeListRenderer.cpp Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update src/mesh/NodeDB.cpp Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Pass by reference -- Thanks Copilot! * Throttle sorting just a touch * Check more carefully for own node * Eliminate some now-unneeded sorting * Move function after include * Putting Modules back to position 0 and some trunk checks found * Add Scrollbar for Action menus * Second attempt to move modules down the navigation bar * Continue effort of moving modules in the navigation * Canned Messages tweak * Replicate Function + Space through the Menu System * Move init button parameters into config struct (#7145) * Remove bundling of web-ui from ESP32 devices (#7143) * Fixed triple click GPS toggle bungle * Move init button parameters into config struct * Reapply "Actually honor the points-north setting" This reverts commit42c1967e7b. * Actually do compass pointings correctly * Tweak to node bearings * Menu wording tweaks * Get the compass_north_top logic right * Don't jump frames after setting Compass * Get rid of the extra bearingTo functions * Don't blink Mail on EInk Clock Screens * Actually set lat and long * Calibrate * Convert Radians to Degrees * More degree vs radians fixes * De-duplicate draw arrow function * Don't advertise compass calibration without an accell thread. --------- Co-authored-by: Ben Meadors <benmmeadors@gmail.com> Co-authored-by: Jonathan Bennett <jbennett@incomsystems.biz> Co-authored-by: HarukiToreda <116696711+HarukiToreda@users.noreply.github.com> Co-authored-by: Thomas Göttgens <tgoettgens@gmail.com> Co-authored-by: csrutil <keming.cao@gmail.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
444 lines
24 KiB
C++
444 lines
24 KiB
C++
#ifndef NRF52_USE_JSON
|
|
#include "MeshPacketSerializer.h"
|
|
#include "JSON.h"
|
|
#include "NodeDB.h"
|
|
#include "mesh/generated/meshtastic/mqtt.pb.h"
|
|
#include "mesh/generated/meshtastic/telemetry.pb.h"
|
|
#include "modules/RoutingModule.h"
|
|
#include <DebugConfiguration.h>
|
|
#include <mesh-pb-constants.h>
|
|
#if defined(ARCH_ESP32)
|
|
#include "../mesh/generated/meshtastic/paxcount.pb.h"
|
|
#endif
|
|
#include "mesh/generated/meshtastic/remote_hardware.pb.h"
|
|
#include <sys/types.h>
|
|
|
|
static const char *errStr = "Error decoding proto for %s message!";
|
|
|
|
std::string MeshPacketSerializer::JsonSerialize(const meshtastic_MeshPacket *mp, bool shouldLog)
|
|
{
|
|
// the created jsonObj is immutable after creation, so
|
|
// we need to do the heavy lifting before assembling it.
|
|
std::string msgType;
|
|
JSONObject jsonObj;
|
|
|
|
if (mp->which_payload_variant == meshtastic_MeshPacket_decoded_tag) {
|
|
JSONObject msgPayload;
|
|
switch (mp->decoded.portnum) {
|
|
case meshtastic_PortNum_TEXT_MESSAGE_APP: {
|
|
msgType = "text";
|
|
// convert bytes to string
|
|
if (shouldLog)
|
|
LOG_DEBUG("got text message of size %u", mp->decoded.payload.size);
|
|
|
|
char payloadStr[(mp->decoded.payload.size) + 1];
|
|
memcpy(payloadStr, mp->decoded.payload.bytes, mp->decoded.payload.size);
|
|
payloadStr[mp->decoded.payload.size] = 0; // null terminated string
|
|
// check if this is a JSON payload
|
|
JSONValue *json_value = JSON::Parse(payloadStr);
|
|
if (json_value != NULL) {
|
|
if (shouldLog)
|
|
LOG_INFO("text message payload is of type json");
|
|
|
|
// if it is, then we can just use the json object
|
|
jsonObj["payload"] = json_value;
|
|
} else {
|
|
// if it isn't, then we need to create a json object
|
|
// with the string as the value
|
|
if (shouldLog)
|
|
LOG_INFO("text message payload is of type plaintext");
|
|
|
|
msgPayload["text"] = new JSONValue(payloadStr);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
}
|
|
break;
|
|
}
|
|
case meshtastic_PortNum_TELEMETRY_APP: {
|
|
msgType = "telemetry";
|
|
meshtastic_Telemetry scratch;
|
|
meshtastic_Telemetry *decoded = NULL;
|
|
memset(&scratch, 0, sizeof(scratch));
|
|
if (pb_decode_from_bytes(mp->decoded.payload.bytes, mp->decoded.payload.size, &meshtastic_Telemetry_msg, &scratch)) {
|
|
decoded = &scratch;
|
|
if (decoded->which_variant == meshtastic_Telemetry_device_metrics_tag) {
|
|
// If battery is present, encode the battery level value
|
|
// TODO - Add a condition to send a code for a non-present value
|
|
if (decoded->variant.device_metrics.has_battery_level) {
|
|
msgPayload["battery_level"] = new JSONValue((int)decoded->variant.device_metrics.battery_level);
|
|
}
|
|
msgPayload["voltage"] = new JSONValue(decoded->variant.device_metrics.voltage);
|
|
msgPayload["channel_utilization"] = new JSONValue(decoded->variant.device_metrics.channel_utilization);
|
|
msgPayload["air_util_tx"] = new JSONValue(decoded->variant.device_metrics.air_util_tx);
|
|
msgPayload["uptime_seconds"] = new JSONValue((unsigned int)decoded->variant.device_metrics.uptime_seconds);
|
|
} else if (decoded->which_variant == meshtastic_Telemetry_environment_metrics_tag) {
|
|
// Avoid sending 0s for sensors that could be 0
|
|
if (decoded->variant.environment_metrics.has_temperature) {
|
|
msgPayload["temperature"] = new JSONValue(decoded->variant.environment_metrics.temperature);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_relative_humidity) {
|
|
msgPayload["relative_humidity"] = new JSONValue(decoded->variant.environment_metrics.relative_humidity);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_barometric_pressure) {
|
|
msgPayload["barometric_pressure"] =
|
|
new JSONValue(decoded->variant.environment_metrics.barometric_pressure);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_gas_resistance) {
|
|
msgPayload["gas_resistance"] = new JSONValue(decoded->variant.environment_metrics.gas_resistance);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_voltage) {
|
|
msgPayload["voltage"] = new JSONValue(decoded->variant.environment_metrics.voltage);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_current) {
|
|
msgPayload["current"] = new JSONValue(decoded->variant.environment_metrics.current);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_lux) {
|
|
msgPayload["lux"] = new JSONValue(decoded->variant.environment_metrics.lux);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_white_lux) {
|
|
msgPayload["white_lux"] = new JSONValue(decoded->variant.environment_metrics.white_lux);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_iaq) {
|
|
msgPayload["iaq"] = new JSONValue((uint)decoded->variant.environment_metrics.iaq);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_wind_speed) {
|
|
msgPayload["wind_speed"] = new JSONValue(decoded->variant.environment_metrics.wind_speed);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_wind_direction) {
|
|
msgPayload["wind_direction"] = new JSONValue((uint)decoded->variant.environment_metrics.wind_direction);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_wind_gust) {
|
|
msgPayload["wind_gust"] = new JSONValue(decoded->variant.environment_metrics.wind_gust);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_wind_lull) {
|
|
msgPayload["wind_lull"] = new JSONValue(decoded->variant.environment_metrics.wind_lull);
|
|
}
|
|
if (decoded->variant.environment_metrics.has_radiation) {
|
|
msgPayload["radiation"] = new JSONValue(decoded->variant.environment_metrics.radiation);
|
|
}
|
|
} else if (decoded->which_variant == meshtastic_Telemetry_air_quality_metrics_tag) {
|
|
if (decoded->variant.air_quality_metrics.has_pm10_standard) {
|
|
msgPayload["pm10"] = new JSONValue((unsigned int)decoded->variant.air_quality_metrics.pm10_standard);
|
|
}
|
|
if (decoded->variant.air_quality_metrics.has_pm25_standard) {
|
|
msgPayload["pm25"] = new JSONValue((unsigned int)decoded->variant.air_quality_metrics.pm25_standard);
|
|
}
|
|
if (decoded->variant.air_quality_metrics.has_pm100_standard) {
|
|
msgPayload["pm100"] = new JSONValue((unsigned int)decoded->variant.air_quality_metrics.pm100_standard);
|
|
}
|
|
if (decoded->variant.air_quality_metrics.has_pm10_environmental) {
|
|
msgPayload["pm10_e"] =
|
|
new JSONValue((unsigned int)decoded->variant.air_quality_metrics.pm10_environmental);
|
|
}
|
|
if (decoded->variant.air_quality_metrics.has_pm25_environmental) {
|
|
msgPayload["pm25_e"] =
|
|
new JSONValue((unsigned int)decoded->variant.air_quality_metrics.pm25_environmental);
|
|
}
|
|
if (decoded->variant.air_quality_metrics.has_pm100_environmental) {
|
|
msgPayload["pm100_e"] =
|
|
new JSONValue((unsigned int)decoded->variant.air_quality_metrics.pm100_environmental);
|
|
}
|
|
} else if (decoded->which_variant == meshtastic_Telemetry_power_metrics_tag) {
|
|
if (decoded->variant.power_metrics.has_ch1_voltage) {
|
|
msgPayload["voltage_ch1"] = new JSONValue(decoded->variant.power_metrics.ch1_voltage);
|
|
}
|
|
if (decoded->variant.power_metrics.has_ch1_current) {
|
|
msgPayload["current_ch1"] = new JSONValue(decoded->variant.power_metrics.ch1_current);
|
|
}
|
|
if (decoded->variant.power_metrics.has_ch2_voltage) {
|
|
msgPayload["voltage_ch2"] = new JSONValue(decoded->variant.power_metrics.ch2_voltage);
|
|
}
|
|
if (decoded->variant.power_metrics.has_ch2_current) {
|
|
msgPayload["current_ch2"] = new JSONValue(decoded->variant.power_metrics.ch2_current);
|
|
}
|
|
if (decoded->variant.power_metrics.has_ch3_voltage) {
|
|
msgPayload["voltage_ch3"] = new JSONValue(decoded->variant.power_metrics.ch3_voltage);
|
|
}
|
|
if (decoded->variant.power_metrics.has_ch3_current) {
|
|
msgPayload["current_ch3"] = new JSONValue(decoded->variant.power_metrics.ch3_current);
|
|
}
|
|
}
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
} else if (shouldLog) {
|
|
LOG_ERROR(errStr, msgType.c_str());
|
|
}
|
|
break;
|
|
}
|
|
case meshtastic_PortNum_NODEINFO_APP: {
|
|
msgType = "nodeinfo";
|
|
meshtastic_User scratch;
|
|
meshtastic_User *decoded = NULL;
|
|
memset(&scratch, 0, sizeof(scratch));
|
|
if (pb_decode_from_bytes(mp->decoded.payload.bytes, mp->decoded.payload.size, &meshtastic_User_msg, &scratch)) {
|
|
decoded = &scratch;
|
|
msgPayload["id"] = new JSONValue(decoded->id);
|
|
msgPayload["longname"] = new JSONValue(decoded->long_name);
|
|
msgPayload["shortname"] = new JSONValue(decoded->short_name);
|
|
msgPayload["hardware"] = new JSONValue(decoded->hw_model);
|
|
msgPayload["role"] = new JSONValue((int)decoded->role);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
} else if (shouldLog) {
|
|
LOG_ERROR(errStr, msgType.c_str());
|
|
}
|
|
break;
|
|
}
|
|
case meshtastic_PortNum_POSITION_APP: {
|
|
msgType = "position";
|
|
meshtastic_Position scratch;
|
|
meshtastic_Position *decoded = NULL;
|
|
memset(&scratch, 0, sizeof(scratch));
|
|
if (pb_decode_from_bytes(mp->decoded.payload.bytes, mp->decoded.payload.size, &meshtastic_Position_msg, &scratch)) {
|
|
decoded = &scratch;
|
|
if ((int)decoded->time) {
|
|
msgPayload["time"] = new JSONValue((unsigned int)decoded->time);
|
|
}
|
|
if ((int)decoded->timestamp) {
|
|
msgPayload["timestamp"] = new JSONValue((unsigned int)decoded->timestamp);
|
|
}
|
|
msgPayload["latitude_i"] = new JSONValue((int)decoded->latitude_i);
|
|
msgPayload["longitude_i"] = new JSONValue((int)decoded->longitude_i);
|
|
if ((int)decoded->altitude) {
|
|
msgPayload["altitude"] = new JSONValue((int)decoded->altitude);
|
|
}
|
|
if ((int)decoded->ground_speed) {
|
|
msgPayload["ground_speed"] = new JSONValue((unsigned int)decoded->ground_speed);
|
|
}
|
|
if (int(decoded->ground_track)) {
|
|
msgPayload["ground_track"] = new JSONValue((unsigned int)decoded->ground_track);
|
|
}
|
|
if (int(decoded->sats_in_view)) {
|
|
msgPayload["sats_in_view"] = new JSONValue((unsigned int)decoded->sats_in_view);
|
|
}
|
|
if ((int)decoded->PDOP) {
|
|
msgPayload["PDOP"] = new JSONValue((int)decoded->PDOP);
|
|
}
|
|
if ((int)decoded->HDOP) {
|
|
msgPayload["HDOP"] = new JSONValue((int)decoded->HDOP);
|
|
}
|
|
if ((int)decoded->VDOP) {
|
|
msgPayload["VDOP"] = new JSONValue((int)decoded->VDOP);
|
|
}
|
|
if ((int)decoded->precision_bits) {
|
|
msgPayload["precision_bits"] = new JSONValue((int)decoded->precision_bits);
|
|
}
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
} else if (shouldLog) {
|
|
LOG_ERROR(errStr, msgType.c_str());
|
|
}
|
|
break;
|
|
}
|
|
case meshtastic_PortNum_WAYPOINT_APP: {
|
|
msgType = "waypoint";
|
|
meshtastic_Waypoint scratch;
|
|
meshtastic_Waypoint *decoded = NULL;
|
|
memset(&scratch, 0, sizeof(scratch));
|
|
if (pb_decode_from_bytes(mp->decoded.payload.bytes, mp->decoded.payload.size, &meshtastic_Waypoint_msg, &scratch)) {
|
|
decoded = &scratch;
|
|
msgPayload["id"] = new JSONValue((unsigned int)decoded->id);
|
|
msgPayload["name"] = new JSONValue(decoded->name);
|
|
msgPayload["description"] = new JSONValue(decoded->description);
|
|
msgPayload["expire"] = new JSONValue((unsigned int)decoded->expire);
|
|
msgPayload["locked_to"] = new JSONValue((unsigned int)decoded->locked_to);
|
|
msgPayload["latitude_i"] = new JSONValue((int)decoded->latitude_i);
|
|
msgPayload["longitude_i"] = new JSONValue((int)decoded->longitude_i);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
} else if (shouldLog) {
|
|
LOG_ERROR(errStr, msgType.c_str());
|
|
}
|
|
break;
|
|
}
|
|
case meshtastic_PortNum_NEIGHBORINFO_APP: {
|
|
msgType = "neighborinfo";
|
|
meshtastic_NeighborInfo scratch;
|
|
meshtastic_NeighborInfo *decoded = NULL;
|
|
memset(&scratch, 0, sizeof(scratch));
|
|
if (pb_decode_from_bytes(mp->decoded.payload.bytes, mp->decoded.payload.size, &meshtastic_NeighborInfo_msg,
|
|
&scratch)) {
|
|
decoded = &scratch;
|
|
msgPayload["node_id"] = new JSONValue((unsigned int)decoded->node_id);
|
|
msgPayload["node_broadcast_interval_secs"] = new JSONValue((unsigned int)decoded->node_broadcast_interval_secs);
|
|
msgPayload["last_sent_by_id"] = new JSONValue((unsigned int)decoded->last_sent_by_id);
|
|
msgPayload["neighbors_count"] = new JSONValue(decoded->neighbors_count);
|
|
JSONArray neighbors;
|
|
for (uint8_t i = 0; i < decoded->neighbors_count; i++) {
|
|
JSONObject neighborObj;
|
|
neighborObj["node_id"] = new JSONValue((unsigned int)decoded->neighbors[i].node_id);
|
|
neighborObj["snr"] = new JSONValue((int)decoded->neighbors[i].snr);
|
|
neighbors.push_back(new JSONValue(neighborObj));
|
|
}
|
|
msgPayload["neighbors"] = new JSONValue(neighbors);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
} else if (shouldLog) {
|
|
LOG_ERROR(errStr, msgType.c_str());
|
|
}
|
|
break;
|
|
}
|
|
case meshtastic_PortNum_TRACEROUTE_APP: {
|
|
if (mp->decoded.request_id) { // Only report the traceroute response
|
|
msgType = "traceroute";
|
|
meshtastic_RouteDiscovery scratch;
|
|
meshtastic_RouteDiscovery *decoded = NULL;
|
|
memset(&scratch, 0, sizeof(scratch));
|
|
if (pb_decode_from_bytes(mp->decoded.payload.bytes, mp->decoded.payload.size, &meshtastic_RouteDiscovery_msg,
|
|
&scratch)) {
|
|
decoded = &scratch;
|
|
JSONArray route; // Route this message took
|
|
JSONArray routeBack; // Route this message took back
|
|
JSONArray snrTowards; // Snr for forward route
|
|
JSONArray snrBack; // Snr for reverse route
|
|
|
|
// Lambda function for adding a long name to the route
|
|
auto addToRoute = [](JSONArray *route, NodeNum num) {
|
|
char long_name[40] = "Unknown";
|
|
meshtastic_NodeInfoLite *node = nodeDB->getMeshNode(num);
|
|
bool name_known = node ? node->has_user : false;
|
|
if (name_known)
|
|
memcpy(long_name, node->user.long_name, sizeof(long_name));
|
|
route->push_back(new JSONValue(long_name));
|
|
};
|
|
addToRoute(&route, mp->to); // Started at the original transmitter (destination of response)
|
|
for (uint8_t i = 0; i < decoded->route_count; i++) {
|
|
addToRoute(&route, decoded->route[i]);
|
|
}
|
|
addToRoute(&route, mp->from); // Ended at the original destination (source of response)
|
|
|
|
addToRoute(&routeBack, mp->from); // Started at the original destination (source of response)
|
|
for (uint8_t i = 0; i < decoded->route_back_count; i++) {
|
|
addToRoute(&routeBack, decoded->route_back[i]);
|
|
}
|
|
addToRoute(&routeBack, mp->to); // Ended at the original transmitter (destination of response)
|
|
|
|
for (uint8_t i = 0; i < decoded->snr_back_count; i++) {
|
|
snrBack.push_back(new JSONValue((float)decoded->snr_back[i] / 4));
|
|
}
|
|
|
|
for (uint8_t i = 0; i < decoded->snr_towards_count; i++) {
|
|
snrTowards.push_back(new JSONValue((float)decoded->snr_towards[i] / 4));
|
|
}
|
|
|
|
msgPayload["route"] = new JSONValue(route);
|
|
msgPayload["route_back"] = new JSONValue(routeBack);
|
|
msgPayload["snr_back"] = new JSONValue(snrBack);
|
|
msgPayload["snr_towards"] = new JSONValue(snrTowards);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
} else if (shouldLog) {
|
|
LOG_ERROR(errStr, msgType.c_str());
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case meshtastic_PortNum_DETECTION_SENSOR_APP: {
|
|
msgType = "detection";
|
|
char payloadStr[(mp->decoded.payload.size) + 1];
|
|
memcpy(payloadStr, mp->decoded.payload.bytes, mp->decoded.payload.size);
|
|
payloadStr[mp->decoded.payload.size] = 0; // null terminated string
|
|
msgPayload["text"] = new JSONValue(payloadStr);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
break;
|
|
}
|
|
#ifdef ARCH_ESP32
|
|
case meshtastic_PortNum_PAXCOUNTER_APP: {
|
|
msgType = "paxcounter";
|
|
meshtastic_Paxcount scratch;
|
|
meshtastic_Paxcount *decoded = NULL;
|
|
memset(&scratch, 0, sizeof(scratch));
|
|
if (pb_decode_from_bytes(mp->decoded.payload.bytes, mp->decoded.payload.size, &meshtastic_Paxcount_msg, &scratch)) {
|
|
decoded = &scratch;
|
|
msgPayload["wifi_count"] = new JSONValue((unsigned int)decoded->wifi);
|
|
msgPayload["ble_count"] = new JSONValue((unsigned int)decoded->ble);
|
|
msgPayload["uptime"] = new JSONValue((unsigned int)decoded->uptime);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
} else if (shouldLog) {
|
|
LOG_ERROR(errStr, msgType.c_str());
|
|
}
|
|
break;
|
|
}
|
|
#endif
|
|
case meshtastic_PortNum_REMOTE_HARDWARE_APP: {
|
|
meshtastic_HardwareMessage scratch;
|
|
meshtastic_HardwareMessage *decoded = NULL;
|
|
memset(&scratch, 0, sizeof(scratch));
|
|
if (pb_decode_from_bytes(mp->decoded.payload.bytes, mp->decoded.payload.size, &meshtastic_HardwareMessage_msg,
|
|
&scratch)) {
|
|
decoded = &scratch;
|
|
if (decoded->type == meshtastic_HardwareMessage_Type_GPIOS_CHANGED) {
|
|
msgType = "gpios_changed";
|
|
msgPayload["gpio_value"] = new JSONValue((unsigned int)decoded->gpio_value);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
} else if (decoded->type == meshtastic_HardwareMessage_Type_READ_GPIOS_REPLY) {
|
|
msgType = "gpios_read_reply";
|
|
msgPayload["gpio_value"] = new JSONValue((unsigned int)decoded->gpio_value);
|
|
msgPayload["gpio_mask"] = new JSONValue((unsigned int)decoded->gpio_mask);
|
|
jsonObj["payload"] = new JSONValue(msgPayload);
|
|
}
|
|
} else if (shouldLog) {
|
|
LOG_ERROR(errStr, "RemoteHardware");
|
|
}
|
|
break;
|
|
}
|
|
// add more packet types here if needed
|
|
default:
|
|
break;
|
|
}
|
|
} else if (shouldLog) {
|
|
LOG_WARN("Couldn't convert encrypted payload of MeshPacket to JSON");
|
|
}
|
|
|
|
jsonObj["id"] = new JSONValue((unsigned int)mp->id);
|
|
jsonObj["timestamp"] = new JSONValue((unsigned int)mp->rx_time);
|
|
jsonObj["to"] = new JSONValue((unsigned int)mp->to);
|
|
jsonObj["from"] = new JSONValue((unsigned int)mp->from);
|
|
jsonObj["channel"] = new JSONValue((unsigned int)mp->channel);
|
|
jsonObj["type"] = new JSONValue(msgType.c_str());
|
|
jsonObj["sender"] = new JSONValue(owner.id);
|
|
if (mp->rx_rssi != 0)
|
|
jsonObj["rssi"] = new JSONValue((int)mp->rx_rssi);
|
|
if (mp->rx_snr != 0)
|
|
jsonObj["snr"] = new JSONValue((float)mp->rx_snr);
|
|
if (mp->hop_start != 0 && mp->hop_limit <= mp->hop_start) {
|
|
jsonObj["hops_away"] = new JSONValue((unsigned int)(mp->hop_start - mp->hop_limit));
|
|
jsonObj["hop_start"] = new JSONValue((unsigned int)(mp->hop_start));
|
|
}
|
|
|
|
// serialize and write it to the stream
|
|
JSONValue *value = new JSONValue(jsonObj);
|
|
std::string jsonStr = value->Stringify();
|
|
|
|
if (shouldLog)
|
|
LOG_INFO("serialized json message: %s", jsonStr.c_str());
|
|
|
|
delete value;
|
|
return jsonStr;
|
|
}
|
|
|
|
std::string MeshPacketSerializer::JsonSerializeEncrypted(const meshtastic_MeshPacket *mp)
|
|
{
|
|
JSONObject jsonObj;
|
|
|
|
jsonObj["id"] = new JSONValue((unsigned int)mp->id);
|
|
jsonObj["time_ms"] = new JSONValue((double)millis());
|
|
jsonObj["timestamp"] = new JSONValue((unsigned int)mp->rx_time);
|
|
jsonObj["to"] = new JSONValue((unsigned int)mp->to);
|
|
jsonObj["from"] = new JSONValue((unsigned int)mp->from);
|
|
jsonObj["channel"] = new JSONValue((unsigned int)mp->channel);
|
|
jsonObj["want_ack"] = new JSONValue(mp->want_ack);
|
|
|
|
if (mp->rx_rssi != 0)
|
|
jsonObj["rssi"] = new JSONValue((int)mp->rx_rssi);
|
|
if (mp->rx_snr != 0)
|
|
jsonObj["snr"] = new JSONValue((float)mp->rx_snr);
|
|
if (mp->hop_start != 0 && mp->hop_limit <= mp->hop_start) {
|
|
jsonObj["hops_away"] = new JSONValue((unsigned int)(mp->hop_start - mp->hop_limit));
|
|
jsonObj["hop_start"] = new JSONValue((unsigned int)(mp->hop_start));
|
|
}
|
|
jsonObj["size"] = new JSONValue((unsigned int)mp->encrypted.size);
|
|
auto encryptedStr = bytesToHex(mp->encrypted.bytes, mp->encrypted.size);
|
|
jsonObj["bytes"] = new JSONValue(encryptedStr.c_str());
|
|
|
|
// serialize and write it to the stream
|
|
JSONValue *value = new JSONValue(jsonObj);
|
|
std::string jsonStr = value->Stringify();
|
|
|
|
delete value;
|
|
return jsonStr;
|
|
}
|
|
#endif |