From 3a1567d41537317eb276d4c23463d4c63b771a04 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:00:02 +0200 Subject: [PATCH 1/5] Create Changelog for milesight-iot-ct305 --- milesight-iot-ct305/CHANGELOG.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 milesight-iot-ct305/CHANGELOG.md diff --git a/milesight-iot-ct305/CHANGELOG.md b/milesight-iot-ct305/CHANGELOG.md new file mode 100644 index 000000000..57140b7b5 --- /dev/null +++ b/milesight-iot-ct305/CHANGELOG.md @@ -0,0 +1,5 @@ +# Changelog + +## 1.0.0 - 2026-04-16 + +- First version of plugin \ No newline at end of file From c4043b10f00759e4afe054050e69e38350cf1271 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:00:02 +0200 Subject: [PATCH 2/5] Create LICENSE for milesight-iot-ct305 --- milesight-iot-ct305/LICENSE.md | 7 +++++++ 1 file changed, 7 insertions(+) create mode 100644 milesight-iot-ct305/LICENSE.md diff --git a/milesight-iot-ct305/LICENSE.md b/milesight-iot-ct305/LICENSE.md new file mode 100644 index 000000000..bffeef341 --- /dev/null +++ b/milesight-iot-ct305/LICENSE.md @@ -0,0 +1,7 @@ +Copyright 2026 Thinger.io + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. \ No newline at end of file From cd589069d029ee74543bf37f4e648552acbb43b4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:00:04 +0200 Subject: [PATCH 3/5] Upload image milesight-iot-ct305 --- milesight-iot-ct305/assets/ct305.png | Bin 0 -> 30102 bytes 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 milesight-iot-ct305/assets/ct305.png diff --git a/milesight-iot-ct305/assets/ct305.png b/milesight-iot-ct305/assets/ct305.png new file mode 100644 index 0000000000000000000000000000000000000000..2772d179dd3fc992d10de03dacee878aa99f56e4 GIT binary patch literal 30102 zcmce-^LyS+_XQf;wr$(CZL=}ksIk>pjh%*#Z98d_#GOWi`4i4{CBG#1J+o)F z_gWLFq9lz7hX)4&0)i+jBcTQY0%`^PCcr=ef5|x?LjwWvFqD-L)$q)5S*Xa))buny!UJSsiatXD8t5XX!!1TxOjYVVqQ^E zDRl4GHT_5eA&T<<>rc6wO{+GDs1y&CB<>QpC`!>u{$N-T%1(G}3B$%0MR&fb1H zi_1Zensg8U^v|zg?gAEaO;GwrIFiNaLvgI!&$mN+g5l@!iwnn%nVGGL40dT4((I29 zf5*&&tjx^Jsy~0U;>ZPx^h8Xh#=@*2lXMug(k87+VMx<$GZp4J>~NsQtr{jg)^QJi z`ZyA7P($EN$asA3i%BLNb|K4lOLAk0x@8$ELQ$6^&n(FwM=YmP0fbf8*N5+`eD#|m z86#OpR)b3RAw;!c>jfvG<&R;FMKSwMW1mU&%d!QUra1Bo7e3=7AK8dSkhnzAS$fGz z?x;oF-NVC_J3+X+mse6SH1dy^mv3|P^KK+rlSuztN)!-R%HF;*bMo{so0t#>heSG_ zh%!<_EQ(!cGAAbonvahU>dg@e0+#=1Ok^x9s(ve*eX12{x;Hl8@ycMHgUj#5UBjZN zLL?fO7N5?RB1cI%cF@eNL^)T<&d#m~-u!Z-v#72vaUz{{_pYm}i%(4EZ}c3^qaFE3 zNbtYoE6&@&lB$y@!esEeQ9+6-6{~Iegy9_8#?;|-AUAv_*nt}2%8-255U!0dTncTm z`g+;QM=uSjc;#0Au8}h7|?Mlf6aqw>XkT2sm~$o;S0YKkRC;c`W3C zdCP2C)3at$6{!DxiqYH~0r6u&eHuEJL=znNEKCpEgZ}VfmlZ@g6|jrQ)}@ge#G(h` zBJO|669J(V62u;ILDbKWmR-?85FG;{x{}E=z�zq>|wT4i6D$raEQSU};DUk@4l< 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 literal 0 HcmV?d00001 From 4ee6fc3377fb557058c2c76120867f6b8d3dc04a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:01:38 +0200 Subject: [PATCH 4/5] Create README for milesight-iot-ct305 --- milesight-iot-ct305/README.md | 66 +++++++++++++++++++++++++++++++++++ 1 file changed, 66 insertions(+) create mode 100644 milesight-iot-ct305/README.md diff --git a/milesight-iot-ct305/README.md b/milesight-iot-ct305/README.md new file mode 100644 index 000000000..0e6ad362b --- /dev/null +++ b/milesight-iot-ct305/README.md @@ -0,0 +1,66 @@ +# Smart Current Transformer - CT305 + +The CT30x is a LoRaWAN® Smart Current Transformer for monitoring energy consumption and analyzing usage remotely. CT305 provides multiple current options to suit energy monitoring and supports sending threshold alarms. With a 500A current capacity, the CT305 features self-powered operation, eliminating the need for batteries or external wiring, and uses a non-invasive clamp-on design for easy and safe installation. + +## Key Features + +- **Self-Powered Operation**: Energy harvesting from measured cables, maintenance-free with no batteries or external wires required +- **High Accuracy Monitoring**: RMS current and accumulated current reporting with sampling frequency up to 3.3 kHz +- **500A Current Capacity**: Rated primary current of 500 Arms per CT with 150 mArms rated secondary current +- **Real-Time Monitoring**: Sampling rate up to 1s for fast response and alarm capabilities +- **Non-Invasive Design**: Clamp-on installation without power de-energizing +- **Temperature Monitoring**: Support for external NTC sensor for cable temperature measurement (-20°C to 100°C) +- **Three-Phase Detection**: Simultaneous monitoring of three phases +- **Threshold Alarms**: Configurable threshold and overrange alarm notifications +- **LED Indicator**: Visual indication of working status and alarms +- **FUOTA Support**: Firmware Update Over the Air capability + +## Technical Specifications + +- **Protocol**: LoRaWAN® (Class A) +- **Frequency**: CN470/IN865/RU864/EU868/US915/AU915/KR920/AS923-1&2&3&4 +- **Tx Power**: 16 dBm (868 MHz) / 20 dBm (915 MHz) / 19 dBm (470 MHz) +- **Sensitivity**: -137 dBm +- **Activation**: OTAA/ABP +- **Measurement Accuracy**: ±1% +- **Resolution**: 1 mA +- **Working Frequency**: 50~60 Hz +- **Antenna**: 50Ω SMA Connector +- **Configuration**: USB Type-C or Downlink + +## Thinger.io Integration + +The CT305 seamlessly integrates with Thinger.io through LoRaWAN network servers, enabling remote energy monitoring, real-time current tracking, and alarm management through cloud-based dashboards. + +## Requirements + +A LoRaWAN server is required to communicate the CT305 into Thinger.io, some options are: + +- [The Things Stack](https://www.thethingsindustries.com/stack/) +- [LORIOT](https://loriot.io/) +- [ChirpStack](https://www.chirpstack.io/) + +Alongside, the corresponding plugin for the selected LoRaWAN server needs to be installed in your Thinger.io instance. + +## Get Started + +### Installation + +Look for the plugin in the [Thinger.io Plugin Store](https://marketplace.thinger.io/) and install it in your Thinger.io instance. Once the plugin is installed a new Product will be created for this device. + +### Configuration + +The Product is already preconfigured, check that the auto provision prefix matches the one selected in your LoRaWAN server plugin in Thinger.io, or change it to your desire. + +### Usage + +Start sending uplinks for autoprovisioning devices and buckets. +This product also provides a predefined dashboard and downlinks. + +## Additional Resources + +Milesight CT305 resources can be found at: + +- [Datasheet](https://resource.milesight.com/milesight/iot/document/ct3xx-datasheet-en.pdf) +- [User Guide](https://resource.milesight.com/milesight/iot/document/ct3xx-user-guide-en.pdf) +- [Thinger docs](https://docs.thinger.io) \ No newline at end of file From 656db69db7fb00f5034e3c7723581b7ed77d1d12 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:04:06 +0200 Subject: [PATCH 5/5] Create plugin.json for milesight-iot-ct305 --- milesight-iot-ct305/plugin.json | 838 ++++++++++++++++++++++++++++++++ 1 file changed, 838 insertions(+) create mode 100644 milesight-iot-ct305/plugin.json diff --git a/milesight-iot-ct305/plugin.json b/milesight-iot-ct305/plugin.json new file mode 100644 index 000000000..bb2adc909 --- /dev/null +++ b/milesight-iot-ct305/plugin.json @@ -0,0 +1,838 @@ +{ + "name": "milesight_iot_ct305", + "version": "1.0.0", + "description": "CT30x is a LoRaWAN® Smart Current Transformer for monitoring the energy andanalyzingconsumption remotely. CT10x provides multiple current options to suit energy monitoringandsupport sending threshold alarms", + "author": "Thinger.io", + "license": "MIT", + "repository": { + "type": "git", + "url": "https://github.com/thinger-io/plugins.git", + "directory": "milesight-iot-ct305" + }, + "metadata": { + "name": "Milesight-Iot CT305", + "description": "CT30x is a LoRaWAN® Smart Current Transformer for monitoring the energy andanalyzingconsumption remotely. CT10x provides multiple current options to suit energy monitoringandsupport sending threshold alarms", + "image": "assets/ct305.png", + "category": "devices", + "vendor": "milesight-iot" + }, + "resources": { + "products": [ + { + "description": "CT30x is a LoRaWAN® Smart Current Transformer for monitoring the energy andanalyzingconsumption remotely. CT10x provides multiple current options to suit energy monitoringandsupport sending threshold alarms", + "enabled": true, + "name": "Milesight-Iot CT305", + "product": "milesight_iot_ct305", + "profile": { + "api": { + "downlink": { + "enabled": true, + "handle_connectivity": false, + "request": { + "data": { + "path": "/downlink", + "payload": "{\r\n \"data\" : \"{{payload.data=\"\"}}\",\r\n \"port\" : {{payload.port=2}},\r\n \"priority\": {{payload.priority=3}},\r\n \"confirmed\" : {{payload.confirmed=false}},\r\n \"uplink\" : {{property.uplink}} \r\n}", + "payload_function": "", + "payload_type": "", + "plugin": "{{property.uplink.source}}", + "target": "plugin_endpoint" + } + }, + "response": { + "data": { + "source": "request_response" + } + } + }, + "uplink": { + "enabled": true, + "handle_connectivity": true, + "request": { + "data": { + "payload": "{{payload}}", + "payload_function": "", + "payload_type": "source_payload", + "resource_stream": "uplink", + "target": "resource_stream" + } + }, + "response": { + "data": { + "source": "request_response" + } + } + } + }, + "autoprovisions": { + "device_autoprovisioning": { + "config": { + "mode": "pattern", + "pattern": "ct305_.*" + }, + "enabled": true + } + }, + "buckets": { + "milesight_ct305_data": { + "backend": "mongodb", + "data": { + "payload": "{{payload}}", + "payload_function": "", + "payload_type": "source_payload", + "resource_stream": "uplink_decoded", + "source": "resource_stream" + }, + "enabled": true, + "retention": { + "period": 3, + "unit": "months" + }, + "tags": [] + } + }, + "code": { + "code": "function decodeThingerUplink(thingerData) {\n // 0. If data has already been decoded, we will return it\n if (thingerData.decodedPayload) return thingerData.decodedPayload;\n \n // 1. Extract and Validate Input\n // We need 'payload' (hex string) and 'fPort' (integer)\n const hexPayload = thingerData.payload || \"\";\n const port = thingerData.fPort || 1;\n\n // 2. Convert Hex String to Byte Array\n const bytes = [];\n for (let i = 0; i < hexPayload.length; i += 2) {\n bytes.push(parseInt(hexPayload.substr(i, 2), 16));\n }\n\n // 3. Dynamic Function Detection and Execution\n \n // CASE A: (The Things Stack v3)\n if (typeof decodeUplink === 'function') {\n try {\n const input = {\n bytes: bytes,\n fPort: port\n };\n var result = decodeUplink(input);\n \n if (result.data) return result.data;\n\n return result; \n } catch (e) {\n console.error(\"Error inside decodeUplink:\", e);\n throw e;\n }\n }\n\n // CASE B: Legacy TTN (v2)\n else if (typeof Decoder === 'function') {\n try {\n return Decoder(bytes, port);\n } catch (e) {\n console.error(\"Error inside Decoder:\", e);\n throw e;\n }\n }\n\n // CASE C: No decoder found\n else {\n throw new Error(\"No compatible TTN decoder function (decodeUplink or Decoder) found in scope.\");\n }\n}\n\n\n// TTN decoder\nfunction decodeUplink(input) {\n var res = Decoder(input.bytes, input.fPort);\n if (res.error) {\n return {\n errors: [res.error],\n };\n }\n return {\n data: res,\n };\n}\n/**\n * Payload Decoder\n *\n * Copyright 2025 Milesight IoT\n *\n * @product CT303 / CT305 / CT310\n */\nvar RAW_VALUE = 0x00;\n\n// The Things Network\nfunction Decoder(bytes, port) {\n return milesightDeviceDecode(bytes);\n}\n/* eslint-enable */\n\nvar current_total_chns = [0x03, 0x05, 0x07];\nvar current_chns = [0x04, 0x06, 0x08];\nvar current_alarm_chns = [0x84, 0x86, 0x88];\n\nfunction milesightDeviceDecode(bytes) {\n var decoded = {};\n for (var i = 0; i < bytes.length;) {\n var channel_id = bytes[i++];\n var channel_type = bytes[i++];\n\n // IPSO VERSION\n if (channel_id === 0xff && channel_type === 0x01) {\n decoded.ipso_version = readProtocolVersion(bytes[i]);\n i += 1;\n }\n // HARDWARE VERSION\n else if (channel_id === 0xff && channel_type === 0x09) {\n decoded.hardware_version = readHardwareVersion(bytes.slice(i, i + 2));\n i += 2;\n }\n // FIRMWARE VERSION\n else if (channel_id === 0xff && channel_type === 0x0a) {\n decoded.firmware_version = readFirmwareVersion(bytes.slice(i, i + 2));\n i += 2;\n }\n // TSL VERSION\n else if (channel_id === 0xff && channel_type === 0xff) {\n decoded.tsl_version = readTslVersion(bytes.slice(i, i + 2));\n i += 2;\n }\n // SERIAL NUMBER\n else if (channel_id === 0xff && channel_type === 0x16) {\n decoded.sn = readSerialNumber(bytes.slice(i, i + 8));\n i += 8;\n }\n // LORAWAN CLASS TYPE\n else if (channel_id === 0xff && channel_type === 0x0f) {\n decoded.lorawan_class = readLoRaWANClass(bytes[i]);\n i += 1;\n }\n // RESET EVENT\n else if (channel_id === 0xff && channel_type === 0xfe) {\n decoded.reset_event = readResetEvent(1);\n i += 1;\n }\n // DEVICE STATUS\n else if (channel_id === 0xff && channel_type === 0x0b) {\n decoded.device_status = readDeviceStatus(1);\n i += 1;\n }\n // TOTAL CURRENT\n else if (includes(current_total_chns, channel_id) && channel_type === 0x97) {\n var current_total_chn_name = \"current_chn\" + (current_total_chns.indexOf(channel_id) + 1) + \"_total\";\n decoded[current_total_chn_name] = readUInt32LE(bytes.slice(i, i + 4)) / 100;\n i += 4;\n }\n // CURRENT\n else if (includes(current_chns, channel_id) && channel_type === 0x99) {\n var current_alarm_chn_name = \"current_chn\" + (current_chns.indexOf(channel_id) + 1);\n var current_value = readUInt16LE(bytes.slice(i, i + 2));\n if (current_value === 0xffff) {\n decoded[current_alarm_chn_name + \"_sensor_status\"] = readSensorStatus(2);\n } else {\n decoded[current_alarm_chn_name] = current_value / 10;\n }\n i += 2;\n }\n // TEMPERATURE\n else if (channel_id === 0x09 && channel_type === 0x67) {\n var temperature_value = readUInt16LE(bytes.slice(i, i + 2));\n if (temperature_value === 0xfffd) {\n decoded.temperature_sensor_status = readSensorStatus(1);\n } else if (temperature_value === 0xffff) {\n decoded.temperature_sensor_status = readSensorStatus(2);\n } else {\n decoded.temperature = readInt16LE(bytes.slice(i, i + 2)) / 10;\n }\n i += 2;\n }\n // CURRENT ALARM\n else if (includes(current_alarm_chns, channel_id) && channel_type === 0x99) {\n var current_alarm_chn_name = \"current_chn\" + (current_alarm_chns.indexOf(channel_id) + 1);\n decoded[current_alarm_chn_name + \"_max\"] = readUInt16LE(bytes.slice(i, i + 2)) / 10;\n decoded[current_alarm_chn_name + \"_min\"] = readUInt16LE(bytes.slice(i + 2, i + 4)) / 10;\n decoded[current_alarm_chn_name] = readUInt16LE(bytes.slice(i + 4, i + 6)) / 10;\n decoded[current_alarm_chn_name + \"_alarm\"] = readCurrentAlarm(bytes[i + 6]);\n i += 7;\n }\n // TEMPERATURE ALARM\n else if (channel_id === 0x89 && channel_type === 0x67) {\n decoded.temperature = readInt16LE(bytes.slice(i, i + 2)) / 10;\n decoded.temperature_alarm = readTemperatureAlarm(bytes[i + 2]);\n i += 3;\n }\n // DOWNLINK RESPONSE\n else if (channel_id === 0xfe || channel_id === 0xff) {\n var result = handle_downlink_response(channel_type, bytes, i);\n decoded = Object.assign(decoded, result.data);\n i = result.offset;\n }\n else {\n break;\n }\n }\n\n return decoded;\n}\n\nfunction handle_downlink_response(channel_type, bytes, offset) {\n var decoded = {};\n\n switch (channel_type) {\n case 0x02:\n decoded.alarm_report_interval = readUInt16LE(bytes.slice(offset, offset + 2));\n offset += 2;\n break;\n case 0x06:\n var value = readUInt8(bytes[offset]);\n var channel_value = (value >>> 3) & 0x07;\n if (channel_value === 0x01 || channel_value === 0x02 || channel_value === 0x03) {\n var current_alarm_config_name = \"current_chn\" + (channel_value) + \"_alarm_config\";\n decoded[current_alarm_config_name] = {};\n decoded[current_alarm_config_name].condition = readConditionType(value & 0x07);\n decoded[current_alarm_config_name].threshold_min = readUInt16LE(bytes.slice(offset + 1, offset + 3));\n decoded[current_alarm_config_name].threshold_max = readUInt16LE(bytes.slice(offset + 3, offset + 5));\n decoded[current_alarm_config_name].alarm_interval = readUInt16LE(bytes.slice(offset + 5, offset + 7));\n decoded[current_alarm_config_name].alarm_counts = readUInt16LE(bytes.slice(offset + 7, offset + 9));\n } else if (channel_value === 0x04) {\n decoded.temperature_alarm_config = {};\n decoded.temperature_alarm_config.condition = readConditionType(value & 0x07);\n decoded.temperature_alarm_config.threshold_min = readInt16LE(bytes.slice(offset + 1, offset + 3)) / 10;\n decoded.temperature_alarm_config.threshold_max = readInt16LE(bytes.slice(offset + 3, offset + 5)) / 10;\n }\n offset += 9;\n break;\n case 0x10:\n decoded.reboot = readYesNoStatus(1);\n offset += 1;\n break;\n case 0x27:\n var index = readUInt8(bytes[offset]);\n var clear_current_cumulative_name = \"clear_current_chn\" + index + \"_cumulative\";\n decoded[clear_current_cumulative_name] = readYesNoStatus(1);\n offset += 1;\n break;\n case 0x28:\n decoded.report_status = readYesNoStatus(1);\n offset += 1;\n break;\n case 0x8e:\n // ignore first byte\n decoded.report_interval = readUInt16LE(bytes.slice(offset + 1, offset + 3));\n offset += 3;\n break;\n case 0xf2:\n decoded.alarm_report_counts = readUInt16LE(bytes.slice(offset, offset + 2));\n offset += 2;\n break;\n default:\n throw new Error(\"unknown downlink response\");\n }\n\n return { data: decoded, offset: offset };\n}\n\nfunction readProtocolVersion(bytes) {\n var major = (bytes & 0xf0) >> 4;\n var minor = bytes & 0x0f;\n return \"v\" + major + \".\" + minor;\n}\n\nfunction readHardwareVersion(bytes) {\n var major = (bytes[0] & 0xff).toString(16);\n var minor = (bytes[1] & 0xff) >> 4;\n return \"v\" + major + \".\" + minor;\n}\n\nfunction readFirmwareVersion(bytes) {\n var major = (bytes[0] & 0xff).toString(16);\n var minor = (bytes[1] & 0xff).toString(16);\n return \"v\" + major + \".\" + minor;\n}\n\nfunction readTslVersion(bytes) {\n var major = bytes[0] & 0xff;\n var minor = bytes[1] & 0xff;\n return \"v\" + major + \".\" + minor;\n}\n\nfunction readSerialNumber(bytes) {\n var temp = [];\n for (var idx = 0; idx < bytes.length; idx++) {\n temp.push((\"0\" + (bytes[idx] & 0xff).toString(16)).slice(-2));\n }\n return temp.join(\"\");\n}\n\nfunction readLoRaWANClass(type) {\n var class_map = {\n 0: \"Class A\",\n 1: \"Class B\",\n 2: \"Class C\",\n 3: \"Class CtoB\",\n };\n return getValue(class_map, type);\n}\n\nfunction readResetEvent(status) {\n var status_map = { 0: \"normal\", 1: \"reset\" };\n return getValue(status_map, status);\n}\n\nfunction readDeviceStatus(status) {\n var status_map = { 0: \"off\", 1: \"on\" };\n return getValue(status_map, status);\n}\n\nfunction readYesNoStatus(status) {\n var status_map = { 0: \"no\", 1: \"yes\" };\n return getValue(status_map, status);\n}\n\nfunction readSensorStatus(status) {\n var status_map = { 0: \"normal\", 1: \"over range alarm\", 2: \"read failed\" };\n return getValue(status_map, status);\n}\n\nfunction readCurrentAlarm(value) {\n var alarm_bit_offset = { \"current_threshold_alarm\": 0, \"current_threshold_alarm_release\": 1, \"current_over_range_alarm\": 2, \"current_over_range_alarm_release\": 3 };\n\n var event = {};\n for (var key in alarm_bit_offset) {\n event[key] = readYesNoStatus((value >> alarm_bit_offset[key]) & 0x01);\n }\n return event;\n}\n\nfunction readConditionType(value) {\n var condition_map = { 0: \"disable\", 1: \"below\", 2: \"above\", 3: \"between\", 4: \"outside\" };\n return getValue(condition_map, value);\n}\n\nfunction readTemperatureAlarm(type) {\n var alarm_map = { 0: \"temperature threshold alarm release\", 1: \"temperature threshold alarm\" };\n return getValue(alarm_map, type);\n}\n\n/* eslint-disable */\nfunction readUInt8(bytes) {\n return bytes & 0xff;\n}\n\nfunction readInt8(bytes) {\n var ref = readUInt8(bytes);\n return ref > 0x7f ? ref - 0x100 : ref;\n}\n\nfunction readUInt16LE(bytes) {\n var value = (bytes[1] << 8) + bytes[0];\n return value & 0xffff;\n}\n\nfunction readInt16LE(bytes) {\n var ref = readUInt16LE(bytes);\n return ref > 0x7fff ? ref - 0x10000 : ref;\n}\n\nfunction readUInt32LE(bytes) {\n var value = (bytes[3] << 24) + (bytes[2] << 16) + (bytes[1] << 8) + bytes[0];\n return (value & 0xffffffff) >>> 0;\n}\n\nfunction readInt32LE(bytes) {\n var ref = readUInt32LE(bytes);\n return ref > 0x7fffffff ? ref - 0x100000000 : ref;\n}\n\nfunction readFloatLE(bytes) {\n var bits = (bytes[3] << 24) | (bytes[2] << 16) | (bytes[1] << 8) | bytes[0];\n var sign = bits >>> 31 === 0 ? 1.0 : -1.0;\n var e = (bits >>> 23) & 0xff;\n var m = e === 0 ? (bits & 0x7fffff) << 1 : (bits & 0x7fffff) | 0x800000;\n var f = sign * m * Math.pow(2, e - 150);\n return f;\n}\n\nfunction includes(items, value) {\n var size = items.length;\n for (var i = 0; i < size; i++) {\n if (items[i] == value) {\n return true;\n }\n }\n return false;\n}\n\nfunction getValue(map, key) {\n if (RAW_VALUE) return key;\n\n var value = map[key];\n if (!value) value = \"unknown\";\n return value;\n}\n\nif (!Object.assign) {\n Object.defineProperty(Object, \"assign\", {\n enumerable: false,\n configurable: true,\n writable: true,\n value: function (target) {\n \"use strict\";\n if (target == null) {\n throw new TypeError(\"Cannot convert first argument to object\");\n }\n\n var to = Object(target);\n for (var i = 1; i < arguments.length; i++) {\n var nextSource = arguments[i];\n if (nextSource == null) {\n continue;\n }\n nextSource = Object(nextSource);\n\n var keysArray = Object.keys(Object(nextSource));\n for (var nextIndex = 0, len = keysArray.length; nextIndex < len; nextIndex++) {\n var nextKey = keysArray[nextIndex];\n var desc = Object.getOwnPropertyDescriptor(nextSource, nextKey);\n if (desc !== undefined && desc.enumerable) {\n // concat array\n if (Array.isArray(to[nextKey]) && Array.isArray(nextSource[nextKey])) {\n to[nextKey] = to[nextKey].concat(nextSource[nextKey]);\n } else {\n to[nextKey] = nextSource[nextKey];\n }\n }\n }\n }\n return to;\n },\n });\n}", + "environment": "javascript", + "storage": "", + "version": "1.0" + }, + "flows": { + "milesight_ct305_decoder": { + "data": { + "payload": "{{payload}}", + "payload_function": "decodeThingerUplink", + "payload_type": "source_payload", + "resource": "uplink", + "source": "resource", + "update": "events" + }, + "enabled": true, + "sink": { + "payload": "{{payload}}", + "payload_function": "", + "payload_type": "source_payload", + "resource_stream": "uplink_decoded", + "target": "resource_stream" + }, + "split_data": false + } + }, + "properties": { + "uplink": { + "data": { + "payload": "{{payload}}", + "payload_function": "", + "payload_type": "source_payload", + "resource": "uplink", + "source": "resource", + "update": "events" + }, + "default": { + "source": "value" + }, + "enabled": true + } + } + }, + "_resources": { + "properties": [ + { + "property": "dashboard", + "value": { + "tabs": [ + { + "name": "Energy Monitoring", + "widgets": [ + { + "layout": { + "col": 0, + "row": 0, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Current CH1" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "", + "extraTextColor": "#1E313E", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "18px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "60px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "60px", + "textWeight": "font-light", + "unit": "A", + "unitSize": "18px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn1", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Source 1", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 2, + "row": 0, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Current CH2" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "", + "extraTextColor": "#1E313E", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "18px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "60px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "60px", + "textWeight": "font-light", + "unit": "A", + "unitSize": "18px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn2", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Source 1", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 4, + "row": 0, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Current CH3" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "", + "extraTextColor": "#1E313E", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "18px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "60px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "60px", + "textWeight": "font-light", + "unit": "A", + "unitSize": "18px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn3", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Source 1", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 0, + "row": 5, + "sizeX": 6, + "sizeY": 10 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Current Channels (24h)" + }, + "properties": { + "alignTimeSeries": false, + "dataAppend": false, + "options": "var options = {\n chart: {\n type: 'line',\n stacked: false\n },\n dataLabels: {\n enabled: false\n },\n stroke: {\n curve: 'smooth',\n width: 2\n },\n xaxis: {\n type: 'datetime',\n labels: {\n datetimeUTC: false\n },\n tooltip: {\n enabled: false\n }\n },\n yaxis: {\n title: {\n text: 'Current (A)'\n },\n labels: {\n formatter: function (val) {\n if (val !== null && typeof val !== 'undefined')\n return val.toFixed(2);\n }\n }\n },\n tooltip: {\n x: {\n format: 'dd/MM/yyyy HH:mm:ss'\n }\n },\n legend: {\n position: 'top'\n }\n};\n", + "realTimeUpdate": true + }, + "sources": [ + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn1", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#e74c3c", + "name": "Channel 1", + "source": "bucket", + "timespan": { + "magnitude": "hour", + "mode": "relative", + "period": "latest", + "value": 24 + } + }, + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn2", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#3498db", + "name": "Channel 2", + "source": "bucket", + "timespan": { + "magnitude": "hour", + "mode": "relative", + "period": "latest", + "value": 24 + } + }, + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn3", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#2ecc71", + "name": "Channel 3", + "source": "bucket", + "timespan": { + "magnitude": "hour", + "mode": "relative", + "period": "latest", + "value": 24 + } + } + ], + "type": "apex_charts" + }, + { + "layout": { + "col": 0, + "row": 15, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Cumulative CH1" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "Total Energy", + "extraTextColor": "#666666", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "14px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "50px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "50px", + "textWeight": "font-light", + "unit": "Ah", + "unitSize": "16px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn1_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Source 1", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 2, + "row": 15, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Cumulative CH2" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "Total Energy", + "extraTextColor": "#666666", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "14px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "50px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "50px", + "textWeight": "font-light", + "unit": "Ah", + "unitSize": "16px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn2_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Source 1", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 4, + "row": 15, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Cumulative CH3" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "Total Energy", + "extraTextColor": "#666666", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "14px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "50px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "50px", + "textWeight": "font-light", + "unit": "Ah", + "unitSize": "16px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn3_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Source 1", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 0, + "row": 20, + "sizeX": 6, + "sizeY": 10 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Cumulative Energy (7 days)" + }, + "properties": { + "alignTimeSeries": false, + "dataAppend": false, + "options": "var options = {\n chart: {\n type: 'line',\n stacked: false\n },\n dataLabels: {\n enabled: false\n },\n stroke: {\n curve: 'smooth',\n width: 2\n },\n xaxis: {\n type: 'datetime',\n labels: {\n datetimeUTC: false\n },\n tooltip: {\n enabled: false\n }\n },\n yaxis: {\n title: {\n text: 'Energy (Ah)'\n },\n labels: {\n formatter: function (val) {\n if (val !== null && typeof val !== 'undefined')\n return val.toFixed(2);\n }\n }\n },\n tooltip: {\n x: {\n format: 'dd/MM/yyyy HH:mm:ss'\n }\n },\n legend: {\n position: 'top'\n }\n};\n", + "realTimeUpdate": true + }, + "sources": [ + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn1_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#e74c3c", + "name": "Channel 1", + "source": "bucket", + "timespan": { + "magnitude": "day", + "mode": "relative", + "period": "latest", + "value": 7 + } + }, + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn2_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#3498db", + "name": "Channel 2", + "source": "bucket", + "timespan": { + "magnitude": "day", + "mode": "relative", + "period": "latest", + "value": 7 + } + }, + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "current_chn3_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#2ecc71", + "name": "Channel 3", + "source": "bucket", + "timespan": { + "magnitude": "day", + "mode": "relative", + "period": "latest", + "value": 7 + } + } + ], + "type": "apex_charts" + } + ] + }, + { + "name": "Temperature", + "widgets": [ + { + "layout": { + "col": 0, + "row": 0, + "sizeX": 3, + "sizeY": 8 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Temperature" + }, + "properties": { + "decimalPlaces": 1, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "", + "extraTextColor": "#1E313E", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "22px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "75px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "75px", + "textWeight": "font-light", + "unit": "ºC", + "unitSize": "22px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "temperature", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Source 1", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 3, + "row": 0, + "sizeX": 3, + "sizeY": 8 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Temperature (24h)" + }, + "properties": { + "alignTimeSeries": false, + "dataAppend": false, + "options": "var options = {\n chart: {\n type: 'area',\n stacked: false\n },\n dataLabels: {\n enabled: false\n },\n stroke: {\n curve: 'smooth',\n width: 2\n },\n fill: {\n type: 'gradient',\n gradient: {\n shadeIntensity: 1,\n opacityFrom: 0.7,\n opacityTo: 0.3\n }\n },\n xaxis: {\n type: 'datetime',\n labels: {\n datetimeUTC: false\n },\n tooltip: {\n enabled: false\n }\n },\n yaxis: {\n title: {\n text: 'Temperature (ºC)'\n },\n labels: {\n formatter: function (val) {\n if (val !== null && typeof val !== 'undefined')\n return val.toFixed(1);\n }\n }\n },\n tooltip: {\n x: {\n format: 'dd/MM/yyyy HH:mm:ss'\n }\n }\n};\n", + "realTimeUpdate": true + }, + "sources": [ + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct305_data", + "mapping": "temperature", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#e67e22", + "name": "Temperature", + "source": "bucket", + "timespan": { + "magnitude": "hour", + "mode": "relative", + "period": "latest", + "value": 24 + } + } + ], + "type": "apex_charts" + } + ] + } + ] + } + } + ] + } + } + ] + } +} \ No newline at end of file