From 52d11e74e7cc45be9a8588a4aa92b6cbe1a950c9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:06:11 +0200 Subject: [PATCH 1/5] Create Changelog for milesight-iot-ct310 --- milesight-iot-ct310/CHANGELOG.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 milesight-iot-ct310/CHANGELOG.md diff --git a/milesight-iot-ct310/CHANGELOG.md b/milesight-iot-ct310/CHANGELOG.md new file mode 100644 index 000000000..57140b7b5 --- /dev/null +++ b/milesight-iot-ct310/CHANGELOG.md @@ -0,0 +1,5 @@ +# Changelog + +## 1.0.0 - 2026-04-16 + +- First version of plugin \ No newline at end of file From 7a8946225a13cb92b6e09961de8147347223e2aa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:06:11 +0200 Subject: [PATCH 2/5] Create LICENSE for milesight-iot-ct310 --- milesight-iot-ct310/LICENSE.md | 7 +++++++ 1 file changed, 7 insertions(+) create mode 100644 milesight-iot-ct310/LICENSE.md diff --git a/milesight-iot-ct310/LICENSE.md b/milesight-iot-ct310/LICENSE.md new file mode 100644 index 000000000..bffeef341 --- /dev/null +++ b/milesight-iot-ct310/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 71a980f9ace95e7264aad9b1b2ca05435627235b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:06:13 +0200 Subject: [PATCH 3/5] Upload image milesight-iot-ct310 --- milesight-iot-ct310/assets/ct310.png | Bin 0 -> 35838 bytes 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 milesight-iot-ct310/assets/ct310.png diff --git a/milesight-iot-ct310/assets/ct310.png b/milesight-iot-ct310/assets/ct310.png new file mode 100644 index 0000000000000000000000000000000000000000..b9e7c9f3dee23f639a85e81302a5a4c61087a64c GIT binary patch literal 35838 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 Xk6inGe&gb`3_#%N>gTe~DWM4fL2}`h literal 0 HcmV?d00001 From b3bc328f9f20602346ed18d7fbd78a14992eaf94 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:07:31 +0200 Subject: [PATCH 4/5] Create README for milesight-iot-ct310 --- milesight-iot-ct310/README.md | 62 +++++++++++++++++++++++++++++++++++ 1 file changed, 62 insertions(+) create mode 100644 milesight-iot-ct310/README.md diff --git a/milesight-iot-ct310/README.md b/milesight-iot-ct310/README.md new file mode 100644 index 000000000..6f39fa54d --- /dev/null +++ b/milesight-iot-ct310/README.md @@ -0,0 +1,62 @@ +# Smart Current Transformer - CT310 + +The CT310 is a LoRaWAN® Smart Current Transformer designed for monitoring energy consumption and analyzing usage remotely. Part of the CT30x series, it provides high-capacity current measurement options to suit energy monitoring applications and supports sending threshold alarms for proactive energy management. + +## Features + +- **Current Measurement Range**: 0 – 1000 A RMS +- **High Accuracy**: ±1% (typical) +- **High Resolution**: 1 mA +- **Sampling Frequency**: 3.3 kHz +- **Data Reporting**: Real-time RMS (1s interval) and cumulative energy data +- **Temperature Sensor**: Integrated NTC sensor (-20°C to 100°C, ±1%) +- **Working Frequency**: 50–60 Hz +- **Compact Design**: 56.5 x 36 x 68.5 mm +- **Enclosure Rating**: IP30 +- **Operating Temperature**: -20°C to 70°C +- **LoRaWAN Class A**: Support for multiple frequency plans (EU868, US915, AS923, AU915, CN470) + +## Use Cases + +- Industrial energy monitoring +- Commercial building energy management +- Data center power consumption tracking +- Remote energy consumption analysis +- Threshold-based alarm notifications +- Electrical load profiling + +## Thinger.io Integration + +The CT310 seamlessly integrates with Thinger.io through LoRaWAN connectivity, enabling real-time monitoring of current consumption, cumulative energy data, and temperature readings via customizable dashboards. + +## Requirements + +A LoRaWAN server is required to communicate the CT310 into Thinger.io. Compatible options include: + +- [The Things Stack](https://www.thethingsindustries.com/stack/) +- [LORIOT](https://loriot.io/) +- [ChirpStack](https://www.chirpstack.io/) + +The corresponding plugin for the selected LoRaWAN server must 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 preference. + +### Usage + +Start sending uplinks for autoprovisioning devices and buckets. This product also provides a predefined dashboard and downlinks for configuration and alarm management. + +## Additional Resources + +Milesight resources can be found at: + +- [CT3xx Datasheet](https://resource.milesight.com/milesight/iot/document/ct3xx-datasheet-en.pdf) +- [Device Repository](https://www.thethingsnetwork.org/device-repository/devices/milesight-iot/ct310/) +- [Thinger.io Documentation](https://docs.thinger.io) \ No newline at end of file From 2347a277c1e8f7bff85f6f44bb6231e1e7631aee Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Penas=20D=C3=ADaz?= Date: Thu, 16 Apr 2026 10:10:11 +0200 Subject: [PATCH 5/5] Create plugin.json for milesight-iot-ct310 --- milesight-iot-ct310/plugin.json | 829 ++++++++++++++++++++++++++++++++ 1 file changed, 829 insertions(+) create mode 100644 milesight-iot-ct310/plugin.json diff --git a/milesight-iot-ct310/plugin.json b/milesight-iot-ct310/plugin.json new file mode 100644 index 000000000..4a52e4830 --- /dev/null +++ b/milesight-iot-ct310/plugin.json @@ -0,0 +1,829 @@ +{ + "name": "milesight_iot_ct310", + "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-ct310" + }, + "metadata": { + "name": "Milesight-Iot CT310", + "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/ct310.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 CT310", + "product": "milesight_iot_ct310", + "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_type": "source_payload", + "resource_stream": "uplink", + "target": "resource_stream" + } + }, + "response": { + "data": { + "source": "request_response" + } + } + } + }, + "autoprovisions": { + "device_autoprovisioning": { + "config": { + "mode": "pattern", + "pattern": "ct310_.*" + }, + "enabled": true + } + }, + "buckets": { + "milesight_ct310_data": { + "backend": "mongodb", + "data": { + "payload": "{{payload}}", + "payload_type": "source_payload", + "resource_stream": "uplink_decoded", + "target": "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_ct310_decoder": { + "data": { + "payload": "{{payload}}", + "payload_function": "decodeThingerUplink", + "payload_type": "source_payload", + "resource": "uplink", + "source": "resource", + "update": "events" + }, + "enabled": true, + "sink": { + "payload": "{{payload}}", + "payload_type": "source_payload", + "resource_stream": "uplink_decoded", + "target": "resource_stream" + }, + "split_data": false + } + }, + "properties": { + "uplink": { + "data": { + "payload": "{{payload}}", + "payload_type": "source_payload", + "resource": "uplink", + "source": "resource", + "update": "events" + }, + "default": { + "source": "value" + }, + "enabled": true + } + } + }, + "_resources": { + "properties": [ + { + "property": "dashboard", + "value": { + "tabs": [ + { + "name": "Overview", + "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": "75px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "65px", + "textWeight": "font-light", + "unit": "A", + "unitSize": "18px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_data", + "mapping": "current_chn1", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Current Ch1", + "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": "75px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "65px", + "textWeight": "font-light", + "unit": "A", + "unitSize": "18px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_data", + "mapping": "current_chn2", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#3498db", + "name": "Current Ch2", + "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": "75px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "65px", + "textWeight": "font-light", + "unit": "A", + "unitSize": "18px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_data", + "mapping": "current_chn3", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#e74c3c", + "name": "Current Ch3", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 0, + "row": 5, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Total Ch1" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "Cumulative", + "extraTextColor": "#7f8c8d", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "14px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "75px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "50px", + "textWeight": "font-light", + "unit": "Ah", + "unitSize": "16px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_data", + "mapping": "current_chn1_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Total Ch1", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 2, + "row": 5, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Total Ch2" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "Cumulative", + "extraTextColor": "#7f8c8d", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "14px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "75px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "50px", + "textWeight": "font-light", + "unit": "Ah", + "unitSize": "16px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_data", + "mapping": "current_chn2_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#3498db", + "name": "Total Ch2", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 4, + "row": 5, + "sizeX": 2, + "sizeY": 5 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Total Ch3" + }, + "properties": { + "decimalPlaces": 2, + "enableExtraTextColor": false, + "enableIconColor": false, + "enableIconSize": false, + "extraText": "Cumulative", + "extraTextColor": "#7f8c8d", + "extraTextColorConditions": [], + "extraTextConditions": [], + "extraTextPosition": "above-value", + "extraTextSize": "14px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "75px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "50px", + "textWeight": "font-light", + "unit": "Ah", + "unitSize": "16px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_data", + "mapping": "current_chn3_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#e74c3c", + "name": "Total Ch3", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 0, + "row": 10, + "sizeX": 6, + "sizeY": 10 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Current Monitoring (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 xaxis: {\n type: 'datetime',\n labels: {\n datetimeUTC: false\n },\n tooltip: {\n enabled: false\n }\n },\n yaxis: {\n labels: {\n formatter: function (val) {\n if (val !== null && typeof val !== 'undefined')\n return val.toFixed(2) + ' A';\n }\n },\n title: {\n text: 'Current (A)'\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_ct310_data", + "mapping": "current_chn1", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Channel 1", + "source": "bucket", + "timespan": { + "magnitude": "hour", + "mode": "relative", + "period": "latest", + "value": 24 + } + }, + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_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_ct310_data", + "mapping": "current_chn3", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#e74c3c", + "name": "Channel 3", + "source": "bucket", + "timespan": { + "magnitude": "hour", + "mode": "relative", + "period": "latest", + "value": 24 + } + } + ], + "type": "apex_charts" + }, + { + "layout": { + "col": 0, + "row": 20, + "sizeX": 3, + "sizeY": 5 + }, + "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": "18px", + "extraTextWeight": "font-light", + "icon": "", + "iconColor": "#1E313E", + "iconColorConditions": [], + "iconConditions": [], + "iconGap": "8px", + "iconPosition": "before-value", + "iconSize": "75px", + "iconVerticalOffset": "0px", + "link": "", + "textAlign": "center", + "textColor": "#1E313E", + "textColorConditions": [], + "textSize": "65px", + "textWeight": "font-light", + "unit": "°C", + "unitSize": "18px" + }, + "sources": [ + { + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_data", + "mapping": "temperature", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#f39c12", + "name": "Temperature", + "source": "bucket", + "timespan": { + "mode": "latest" + } + } + ], + "type": "text" + }, + { + "layout": { + "col": 3, + "row": 20, + "sizeX": 3, + "sizeY": 10 + }, + "panel": { + "color": "#ffffff", + "currentColor": "#ffffff", + "showOffline": { + "type": "none" + }, + "title": "Temperature Trend (24h)" + }, + "properties": { + "alignTimeSeries": false, + "dataAppend": false, + "options": "var options = {\n chart: {\n type: 'line'\n },\n dataLabels: {\n enabled: false\n },\n stroke: {\n curve: 'smooth',\n width: 3\n },\n xaxis: {\n type: 'datetime',\n labels: {\n datetimeUTC: false\n },\n tooltip: {\n enabled: false\n }\n },\n yaxis: {\n labels: {\n formatter: function (val) {\n if (val !== null && typeof val !== 'undefined')\n return val.toFixed(1) + ' °C';\n }\n },\n title: {\n text: 'Temperature (°C)'\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_ct310_data", + "mapping": "temperature", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#f39c12", + "name": "Temperature", + "source": "bucket", + "timespan": { + "magnitude": "hour", + "mode": "relative", + "period": "latest", + "value": 24 + } + } + ], + "type": "apex_charts" + }, + { + "layout": { + "col": 0, + "row": 25, + "sizeX": 3, + "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 },\n dataLabels: {\n enabled: false\n },\n stroke: {\n curve: 'stepline',\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 labels: {\n formatter: function (val) {\n if (val !== null && typeof val !== 'undefined')\n return val.toFixed(2) + ' Ah';\n }\n },\n title: {\n text: 'Cumulative Current (Ah)'\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_ct310_data", + "mapping": "current_chn1_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#1abc9c", + "name": "Channel 1", + "source": "bucket", + "timespan": { + "magnitude": "day", + "mode": "relative", + "period": "latest", + "value": 7 + } + }, + { + "aggregation": {}, + "bucket": { + "backend": "mongodb", + "id": "milesight_ct310_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_ct310_data", + "mapping": "current_chn3_total", + "tags": { + "device": [], + "group": [] + } + }, + "color": "#e74c3c", + "name": "Channel 3", + "source": "bucket", + "timespan": { + "magnitude": "day", + "mode": "relative", + "period": "latest", + "value": 7 + } + } + ], + "type": "apex_charts" + } + ] + } + ] + } + } + ] + } + } + ] + } +} \ No newline at end of file