LoRaWAN Sensor standard formatter (#23761)
* LoRaWAN decoder standard formatter implemented * New icons added * Milesight WS202/WS301/WS522 standard formatter implementation * Milesight WS202/WS301 bug fix * little code optimization * bug fix * formatter sample usage * Removed Icons handling from standard formatted, adjust standard sensor format, added links to standard formatter * altitude formatter, add_link formatter, SendDownLink function * WS522 missing attribute, Downlink refactoring, UI Standardization
This commit is contained in:
parent
324ac9b158
commit
efb5ab1ef8
@ -7,6 +7,88 @@ var LwDeco
|
|||||||
import mqtt
|
import mqtt
|
||||||
import string
|
import string
|
||||||
|
|
||||||
|
class LwSensorFormatter_cls
|
||||||
|
var Msg
|
||||||
|
|
||||||
|
static Formatter = {
|
||||||
|
"string": { "u": nil, "f": " %s", "i": nil },
|
||||||
|
"volt": { "u": "V", "f": " %.1f", "i": "⚡" }, # High Voltage ⚡
|
||||||
|
"milliamp": { "u": "mA", "f": " %.0f", "i": "🔌" }, # Electric Plug 🔌
|
||||||
|
"power_factor%": { "u": "%", "f": " %.0f", "i": "📊" }, # Bar Chart 📊
|
||||||
|
"power": { "u": "W", "f": " %.0f", "i": "💡" }, # Light Bulb 💡
|
||||||
|
"energy": { "u": "Wh", "f": " %.0f", "i": "🧮" }, # Abacus 🧮
|
||||||
|
"altitude": { "u": "mt", "f": " %d", "i": "⛰" }, # Moutain ⛰
|
||||||
|
"empty": { "u": nil, "f": nil, "i": nil }
|
||||||
|
}
|
||||||
|
|
||||||
|
def init()
|
||||||
|
self.Msg = ""
|
||||||
|
end
|
||||||
|
|
||||||
|
def start_line()
|
||||||
|
self.Msg += format("<tr class='htr'><td colspan='4'>┆" ) # | <sensor1><sensor2>...
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
def end_line()
|
||||||
|
self.Msg += "{e}" # End of line
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
def next_line()
|
||||||
|
return self.end_line().start_line() # End of current line and start new line
|
||||||
|
end
|
||||||
|
|
||||||
|
def begin_tooltip(ttip)
|
||||||
|
self.Msg += format(" <div title='%s' class='si'>", ttip)
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
def end_tooltip()
|
||||||
|
self.Msg += "</div>"
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
def add_link(title, url, target)
|
||||||
|
self.Msg += format( " <a target=%s href='%s'>%s</a>", ( target ? target : "_blank" ), url, title )
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
def add_sensor(formatter, value, tooltip, alt_icon)
|
||||||
|
if tooltip
|
||||||
|
self.begin_tooltip(tooltip)
|
||||||
|
end
|
||||||
|
|
||||||
|
var fmt = self.Formatter.find(formatter)
|
||||||
|
|
||||||
|
if alt_icon
|
||||||
|
self.Msg += format(" %s", alt_icon ) # Use alternative icon
|
||||||
|
elif fmt && fmt.find("i") && fmt["i"]
|
||||||
|
self.Msg += format(" %s", fmt["i"] ) # Use icon from formatter
|
||||||
|
end
|
||||||
|
|
||||||
|
if fmt && fmt.find("f") && fmt["f"]
|
||||||
|
self.Msg += format(fmt["f"], value)
|
||||||
|
else
|
||||||
|
self.Msg += str(value) # Default to string representation
|
||||||
|
end
|
||||||
|
|
||||||
|
if fmt && fmt.find("u") && fmt["u"]
|
||||||
|
self.Msg += format("%s", fmt["u"]) # Append unit if defined
|
||||||
|
end
|
||||||
|
|
||||||
|
if tooltip
|
||||||
|
return self.end_tooltip()
|
||||||
|
end
|
||||||
|
|
||||||
|
return self
|
||||||
|
end
|
||||||
|
|
||||||
|
def get_msg()
|
||||||
|
return self.Msg
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
class lwdecode_cls
|
class lwdecode_cls
|
||||||
var LwDecoders
|
var LwDecoders
|
||||||
var topic
|
var topic
|
||||||
@ -26,6 +108,17 @@ class lwdecode_cls
|
|||||||
tasmota.add_rule("LwReceived", /value, trigger, payload -> self.LwDecode(payload))
|
tasmota.add_rule("LwReceived", /value, trigger, payload -> self.LwDecode(payload))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def SendDownlink(nodes, cmd, idx, payload, ok_result)
|
||||||
|
if nodes.find(idx)
|
||||||
|
var sendcmd = 'LoRaWanSend'
|
||||||
|
var output = tasmota.cmd( f'{sendcmd}{idx} {payload}', true)
|
||||||
|
if output.find(sendcmd) == 'Done'
|
||||||
|
return tasmota.resp_cmnd(f'{{"{cmd}{idx}":"{ok_result}"}}')
|
||||||
|
end
|
||||||
|
return output
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def LwDecode(data)
|
def LwDecode(data)
|
||||||
import json
|
import json
|
||||||
|
|
||||||
@ -85,6 +178,10 @@ class lwdecode_cls
|
|||||||
return format("%02d%s", since, unit)
|
return format("%02d%s", since, unit)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def dhm_tt(last_time)
|
||||||
|
return format( "Received %s ago", self.dhm(last_time) )
|
||||||
|
end
|
||||||
|
|
||||||
def header(name, name_tooltip, battery, battery_last_seen, rssi, last_seen)
|
def header(name, name_tooltip, battery, battery_last_seen, rssi, last_seen)
|
||||||
var color_text = f'{tasmota.webcolor(0 #-COL_TEXT-#)}' # '#eaeaea'
|
var color_text = f'{tasmota.webcolor(0 #-COL_TEXT-#)}' # '#eaeaea'
|
||||||
var msg = "<tr class='ltd htr'>" # ==== Start first table row
|
var msg = "<tr class='ltd htr'>" # ==== Start first table row
|
||||||
|
|||||||
@ -94,20 +94,20 @@ class LwDecoWS202
|
|||||||
msg += lwdecode.header(name, name_tooltip, battery + 100000, battery_last_seen, rssi, last_seen)
|
msg += lwdecode.header(name, name_tooltip, battery + 100000, battery_last_seen, rssi, last_seen)
|
||||||
|
|
||||||
# Sensors
|
# Sensors
|
||||||
var pir = sensor[6]
|
var pir = lwdecode.dhm(sensor[7])
|
||||||
var pir_last_seen = sensor[7]
|
var pir_ls = lwdecode.dhm_tt(sensor[7])
|
||||||
var light = sensor[8]
|
var pir_alt = (sensor[6] == true ? "🚫" : "🆓") # No Entry 🚫 / Free 🆓
|
||||||
var light_last_seen = sensor[9]
|
|
||||||
|
var light = lwdecode.dhm(sensor[9])
|
||||||
|
var light_ls = lwdecode.dhm_tt(sensor[9])
|
||||||
|
var light_alt = (sensor[8] == 0) ? "🌕" : "🌞" # Moon 🌕 / Sun 🌞
|
||||||
|
|
||||||
msg += "<tr class='htr'><td colspan='4'>┆" # |
|
var fmt = LwSensorFormatter_cls()
|
||||||
|
msg += fmt.start_line()
|
||||||
msg += string.format(" %s %s", pir == true ? "🚫" : "🆓", # PIR Free or Busy
|
.add_sensor( "string", pir, pir_ls, pir_alt )
|
||||||
lwdecode.dhm(pir_last_seen))
|
.add_sensor( "string", light, light_ls, light_alt )
|
||||||
|
.end_line()
|
||||||
msg += string.format(" %s %s", light == 0 ? "🌕" : "🌞", # Light
|
.get_msg()
|
||||||
lwdecode.dhm(light_last_seen))
|
|
||||||
|
|
||||||
msg += "{e}" # = </td></tr>
|
|
||||||
end
|
end
|
||||||
return msg
|
return msg
|
||||||
end #add_web_sensor()
|
end #add_web_sensor()
|
||||||
|
|||||||
@ -94,19 +94,20 @@ class LwDecoWS301
|
|||||||
msg += lwdecode.header(name, name_tooltip, battery + 100000, battery_last_seen, rssi, last_seen)
|
msg += lwdecode.header(name, name_tooltip, battery + 100000, battery_last_seen, rssi, last_seen)
|
||||||
|
|
||||||
# Sensors
|
# Sensors
|
||||||
var door_open = sensor[6]
|
var dopen = lwdecode.dhm(sensor[7])
|
||||||
var door_open_last_seen = sensor[7]
|
var dopen_tt = nil
|
||||||
var installed = sensor[8]
|
var dopen_alt = (sensor[6] == true) ? "🔓" : "🔒" # Open Lock 🔓 / Lock 🔒
|
||||||
var installed_last_seen = sensor[9]
|
|
||||||
|
|
||||||
msg += "<tr class='htr'><td colspan='4'>┆" # |
|
var inst = lwdecode.dhm(sensor[9])
|
||||||
msg += string.format(" %s %s", (door_open == true) ? "🔓" : "🔒", # Open or Closed lock - Door
|
var inst_tt = nil
|
||||||
lwdecode.dhm(door_open_last_seen))
|
var inst_alt = (sensor[8] == true) ? "✅" : "❌" # Heavy Check Mark ✅ / Cross Mark ❌
|
||||||
|
|
||||||
msg += string.format(" %s %s", (installed == true) ? "✅" : "❌", # Installed
|
var fmt = LwSensorFormatter_cls()
|
||||||
lwdecode.dhm(installed_last_seen))
|
msg += fmt.start_line()
|
||||||
|
.add_sensor( "string", dopen, dopen_tt, dopen_alt )
|
||||||
msg += "{e}" # = </td></tr>
|
.add_sensor( "string", inst, inst_tt, inst_alt )
|
||||||
|
.end_line()
|
||||||
|
.get_msg()
|
||||||
end
|
end
|
||||||
return msg
|
return msg
|
||||||
end #add_web_sensor()
|
end #add_web_sensor()
|
||||||
|
|||||||
@ -119,6 +119,38 @@ class LwDecoWS522
|
|||||||
data.insert("Button_State", button_state ? "Open" : "Close" )
|
data.insert("Button_State", button_state ? "Open" : "Close" )
|
||||||
i += 1
|
i += 1
|
||||||
|
|
||||||
|
# FE02(ReportInterval) 3C00=>06
|
||||||
|
elif channel_id == 0xFE && channel_type == 0x02
|
||||||
|
data.insert("Period", ((Bytes[i+1] << 8) | Bytes[i]) )
|
||||||
|
i += 2
|
||||||
|
|
||||||
|
# FF01(ProtocolVersion) 01=>V1
|
||||||
|
elif channel_id == 0xFF && channel_type == 0x01
|
||||||
|
data.insert("Protocol Version", Bytes[i] )
|
||||||
|
i += 1
|
||||||
|
|
||||||
|
# FF09(HardwareVersion) 0140=>V1.4
|
||||||
|
elif channel_id == 0xFF && channel_type == 0x09
|
||||||
|
data.insert("Hardware Version", format("v%02x.%02x", Bytes[i], Bytes[i+1]) )
|
||||||
|
i += 2
|
||||||
|
|
||||||
|
# FF0a(SoftwareVersion) 0114=>V1.14
|
||||||
|
elif channel_id == 0xFF && channel_type == 0x0A
|
||||||
|
data.insert("Software Version", format("v%02x.%02x", Bytes[i], Bytes[i+1]) )
|
||||||
|
i += 2
|
||||||
|
|
||||||
|
# FF0b(PowerOn) Deviceison
|
||||||
|
elif channel_id == 0xFF && channel_type == 0x0B
|
||||||
|
i += 1
|
||||||
|
|
||||||
|
# FF16(DeviceSN) 16digits
|
||||||
|
elif channel_id == 0xFF && channel_type == 0x16
|
||||||
|
i += 8
|
||||||
|
|
||||||
|
# FF0f(DeviceType) 00:ClassA,01:ClassB,02:ClassC
|
||||||
|
elif channel_id == 0xFF && channel_type == 0x0F
|
||||||
|
i += 1
|
||||||
|
|
||||||
else
|
else
|
||||||
log( string.format("WS522: something missing? id={%s} type={%s}", channel_id, channel_type), 1)
|
log( string.format("WS522: something missing? id={%s} type={%s}", channel_id, channel_type), 1)
|
||||||
|
|
||||||
@ -130,58 +162,42 @@ class LwDecoWS522
|
|||||||
|
|
||||||
if valid_values
|
if valid_values
|
||||||
if !command_init
|
if !command_init
|
||||||
tasmota.add_cmd( "LoraWS522Power",
|
tasmota.add_cmd( "LwWS522Power",
|
||||||
def (cmd, idx, payload)
|
def (cmd, idx, payload)
|
||||||
if global.ws522Nodes.find(idx)
|
if payload == "1" || string.toupper(payload) == "ON"
|
||||||
if payload == "1" || string.toupper(payload) == "ON"
|
return lwdecode.SendDownlink(global.ws522Nodes, cmd, idx, '080100FF', 'ON')
|
||||||
tasmota.cmd(string.format("LoraWanSend%d 080100FF",idx))
|
elif payload == "0" || string.toupper(payload) == "OFF"
|
||||||
elif payload == "0" || string.toupper(payload) == "OFF"
|
return lwdecode.SendDownlink(global.ws522Nodes, cmd, idx, '080000FF', 'OFF')
|
||||||
tasmota.cmd(string.format("LoraWanSend%d 080000FF",idx))
|
|
||||||
else
|
|
||||||
# nothing else
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
end
|
||||||
)
|
)
|
||||||
|
|
||||||
tasmota.add_cmd( "LoraWS522Period",
|
tasmota.add_cmd( "LwWS522Period",
|
||||||
def (cmd, idx, payload)
|
def (cmd, idx, payload)
|
||||||
if global.ws522Nodes.find(idx)
|
return lwdecode.SendDownlink(global.ws522Nodes, cmd, idx, format('FF02%s',uint16le(number(payload))), number(payload))
|
||||||
if number(payload) > 30
|
end
|
||||||
tasmota.cmd( string.format("LoraWanSend%d FF02%s", idx, uint16le(number(payload))) )
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
)
|
)
|
||||||
|
|
||||||
tasmota.add_cmd( "LoraWS522Reboot",
|
tasmota.add_cmd( "LwWS522Reboot",
|
||||||
def (cmd, idx, payload)
|
def (cmd, idx, payload)
|
||||||
if global.ws522Nodes.find(idx)
|
return lwdecode.SendDownlink(global.ws522Nodes, cmd, idx, 'FF10FF', 'Done')
|
||||||
tasmota.cmd( string.format("LoraWanSend%d FF10FF", idx) )
|
end
|
||||||
end
|
|
||||||
end
|
|
||||||
)
|
)
|
||||||
|
|
||||||
tasmota.add_cmd( "LoraWS522ResetPowerUsage",
|
tasmota.add_cmd( "LwWS522ResetPowerUsage",
|
||||||
def (cmd, idx, payload)
|
def (cmd, idx, payload)
|
||||||
if global.ws522Nodes.find(idx)
|
return lwdecode.SendDownlink(global.ws522Nodes, cmd, idx, 'FF27FF', 'Done')
|
||||||
tasmota.cmd( string.format("LoraWanSend%d FF27FF", idx) )
|
end
|
||||||
end
|
|
||||||
end
|
|
||||||
)
|
)
|
||||||
|
|
||||||
tasmota.add_cmd( "LoraWS522PowerLock",
|
tasmota.add_cmd( "LwWS522PowerLock",
|
||||||
def (cmd, idx, payload)
|
def (cmd, idx, payload)
|
||||||
if global.ws522Nodes.find(idx)
|
if payload == "1" || string.toupper(payload) == "ON"
|
||||||
if payload == "1" || string.toupper(payload) == "ON"
|
return lwdecode.SendDownlink(global.ws522Nodes, cmd, idx, 'FF250080', 'ON')
|
||||||
tasmota.cmd(string.format("LoraWanSend%d FF250080",idx))
|
elif payload == "0" || string.toupper(payload) == "OFF"
|
||||||
elif payload == "0" || string.toupper(payload) == "OFF"
|
return lwdecode.SendDownlink(global.ws522Nodes, cmd, idx, 'FF250080', 'OFF')
|
||||||
tasmota.cmd(string.format("LoraWanSend%d FF250000",idx))
|
|
||||||
else
|
|
||||||
# nothing else
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
end
|
||||||
)
|
)
|
||||||
command_init = true
|
command_init = true
|
||||||
end
|
end
|
||||||
@ -233,37 +249,40 @@ class LwDecoWS522
|
|||||||
|
|
||||||
msg += lwdecode.header(name, name_tooltip, 1000, last_seen, rssi, last_seen)
|
msg += lwdecode.header(name, name_tooltip, 1000, last_seen, rssi, last_seen)
|
||||||
|
|
||||||
# IEC Power Symbols
|
|
||||||
# Power ⏻ ⏻
|
|
||||||
# Toggle Power ⏼ ⏼
|
|
||||||
# Power On ⏽ ⏽
|
|
||||||
# Power Off ⭘ ⭘
|
|
||||||
# Sleep Mode ⏾ ⏾
|
|
||||||
|
|
||||||
# Sensors
|
# Sensors
|
||||||
var voltage = sensor[4]
|
var voltage = sensor[4]
|
||||||
var active_power = sensor[5]
|
var voltage_tt = lwdecode.dhm(sensor[10])
|
||||||
var power_factor = sensor[6]
|
|
||||||
var energy_sum = sensor[7]
|
|
||||||
var current = sensor[8]
|
|
||||||
var button_state = sensor[9] ? "⏽" : "⭘"
|
|
||||||
var voltage_ls = sensor[10]
|
|
||||||
var active_power_ls = sensor[11]
|
|
||||||
var power_factor_ls = sensor[12]
|
|
||||||
var energy_sum_ls = sensor[13]
|
|
||||||
var current_ls = sensor[14]
|
|
||||||
var button_state_ls = sensor[15]
|
|
||||||
|
|
||||||
msg += "<tr class='htr'><td colspan='4'>┆" # |
|
var active_power = sensor[5]
|
||||||
msg += string.format(" %s %.1fV", "⚡", voltage) # High Voltage Icon
|
var active_power_tt = lwdecode.dhm(sensor[11])
|
||||||
msg += string.format(" %s %dmA", "🔌", current) # Electric Plug Icon
|
|
||||||
msg += string.format(" %s %d%%", "📊", power_factor) # Bar Chart Icon
|
var power_factor = sensor[6]
|
||||||
msg += string.format(" %s %dw", "💡", active_power) # Light Bulb Icon
|
var power_factor_tt = lwdecode.dhm(sensor[12])
|
||||||
msg += "{e}<tr class='htr'><td colspan='4'>┆" # |
|
|
||||||
msg += string.format(" %s", button_state) # Button Sate ON | OFF icon
|
var current = sensor[8]
|
||||||
msg += string.format(" %s %s", "⏱", lwdecode.dhm(button_state_ls)) # Stopwatch icon
|
var current_tt = lwdecode.dhm(sensor[14])
|
||||||
msg += string.format(" %s %dWh", "🧮", energy_sum) # Abacus Icon
|
|
||||||
msg += "{e}" # = </td></tr>
|
var button_state = lwdecode.dhm(sensor[15])
|
||||||
|
var button_state_tt = lwdecode.dhm(sensor[15])
|
||||||
|
var button_state_icon = (sensor[9] ? " 🟢 " : " ⚫ ") # Large Green Circle 🟢 | Medium Black Circle ⚫
|
||||||
|
|
||||||
|
var energy_sum = sensor[7]
|
||||||
|
var energy_sum_tt = lwdecode.dhm(sensor[13] )
|
||||||
|
|
||||||
|
var fmt = LwSensorFormatter_cls()
|
||||||
|
|
||||||
|
# Formatter Value Tooltip alternative icon
|
||||||
|
# ================ ============ ================== ================
|
||||||
|
msg += fmt.start_line()
|
||||||
|
.add_sensor("volt", voltage, voltage_tt )
|
||||||
|
.add_sensor("milliamp", current, current_tt )
|
||||||
|
.add_sensor("power_factor%", power_factor, power_factor_tt )
|
||||||
|
.add_sensor("power", active_power, active_power_tt )
|
||||||
|
.next_line()
|
||||||
|
.add_sensor("string", button_state, button_state_tt, button_state_icon )
|
||||||
|
.add_sensor("energy", energy_sum, energy_sum_tt )
|
||||||
|
.end_line()
|
||||||
|
.get_msg()
|
||||||
end
|
end
|
||||||
return msg
|
return msg
|
||||||
end #add_web_sensor()
|
end #add_web_sensor()
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user