diff --git a/dingtek-do201/plugin.json b/dingtek-do201/plugin.json
index fa33715b0..895c44a5d 100644
--- a/dingtek-do201/plugin.json
+++ b/dingtek-do201/plugin.json
@@ -1,134 +1,870 @@
{
- "name": "dingtek_do201",
- "version": "1.0.0",
- "description": "The CNDingtek do201 is parking occupancy sensor with combination of ultrasonic and magnetic detection, temperature and humidity measurement.",
- "author": "Thinger.io",
- "license": "MIT",
- "repository": {
- "type": "git",
- "url": "https://github.com/thinger-io/plugins.git",
- "directory": "dingtek-do201"
- },
- "metadata": {
- "name": "Dingtek DO201",
+ "name": "dingtek-do201",
+ "version": "1.0.0",
"description": "The CNDingtek do201 is parking occupancy sensor with combination of ultrasonic and magnetic detection, temperature and humidity measurement.",
- "image": "assets/do201.png",
- "category": "devices",
- "vendor": "dingtek"
- },
- "resources": {
- "products": [
- {
- "description": "The CNDingtek do201 is parking occupancy sensor with combination of ultrasonic and magnetic detection, temperature and humidity measurement.",
- "enabled": true,
+ "author": "Thinger.io",
+ "license": "MIT",
+ "repository": {
+ "type": "git",
+ "url": "https://github.com/thinger-io/plugins.git",
+ "directory": "dingtek-do201"
+ },
+ "metadata": {
"name": "Dingtek DO201",
- "product": "dingtek_do201",
- "profile": {
- "api": {
- "downlink": {
- "enabled": true,
- "handle_connectivity": false,
- "request": {
- "data": {
- "path": "/downlink",
- "payload": "{\n \"data\" : \"{{payload.data=\"\"}}\",\n \"port\" : {{payload.port=3}},\n \"priority\": {{payload.priority=3}},\n \"confirmed\" : {{payload.confirmed=false}},\n \"uplink\" : {{property.uplink}} \n}",
- "payload_function": "",
- "payload_type": "",
- "plugin": "{{property.uplink.source}}",
- "target": "plugin_endpoint"
- }
- }
- },
- "uplink": {
- "device_id_resolver": "getId",
- "enabled": true,
- "handle_connectivity": true,
- "request": {
- "data": {
- "payload": "{{payload}}",
- "payload_function": "",
- "payload_type": "source_payload",
- "resource_stream": "uplink",
- "target": "resource_stream"
+ "description": "The CNDingtek do201 is parking occupancy sensor with combination of ultrasonic and magnetic detection, temperature and humidity measurement.",
+ "image": "assets/do201.png",
+ "category": "devices",
+ "vendor": "dingtek"
+ },
+ "resources": {
+ "products": [
+ {
+ "description": "The CNDingtek do201 is parking occupancy sensor with combination of ultrasonic and magnetic detection, temperature and humidity measurement.",
+ "enabled": true,
+ "name": "Dingtek DO201",
+ "product": "dingtek_do201",
+ "profile": {
+ "api": {
+ "downlink": {
+ "enabled": true,
+ "handle_connectivity": false,
+ "request": {
+ "data": {
+ "path": "/downlink",
+ "payload": "{\n \"data\" : \"{{payload.data=\"\"}}\",\n \"port\" : {{payload.port=3}},\n \"priority\": {{payload.priority=3}},\n \"confirmed\" : {{payload.confirmed=false}},\n \"uplink\" : {{property.uplink}} \n}",
+ "payload_function": "",
+ "payload_type": "",
+ "plugin": "{{property.uplink.source}}",
+ "target": "plugin_endpoint"
+ }
+ }
+ },
+ "uplink": {
+ "device_id_resolver": "getId",
+ "enabled": true,
+ "handle_connectivity": true,
+ "request": {
+ "data": {
+ "payload": "{{payload}}",
+ "payload_function": "",
+ "payload_type": "source_payload",
+ "resource_stream": "uplink",
+ "target": "resource_stream"
+ }
+ }
+ }
+ },
+ "autoprovisions": {
+ "device_autoprovisioning": {
+ "config": {
+ "mode": "pattern",
+ "pattern": "do201_.*"
+ },
+ "enabled": true
+ }
+ },
+ "buckets": {
+ "dingtek_do201_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\nvar units = [' â', ' hours', ' minutes', ' mm', ' °', ' cm'];\n//IEEE754 hex to float convert\nfunction hex2float(num) {\n var sign = num & 0x80000000 ? -1 : 1;\n var exponent = ((num >> 23) & 0xff) - 127;\n var mantissa = 1 + (num & 0x7fffff) / 0x7fffff;\n return sign * mantissa * Math.pow(2, exponent);\n}\n\nfunction decodeUplink(input) {\n if (input.fPort != 3) {\n return {\n errors: ['unknown FPort'],\n };\n }\n\n switch (input.bytes.length) {\n case 23:\n var mag_x = (input.bytes[11] << 8) + input.bytes[12];\n var mag_y = (input.bytes[13] << 8) + input.bytes[14];\n var mag_z = (input.bytes[15] << 8) + input.bytes[16];\n var rawTemperature = input.bytes[17];\n return {\n // Decoded data\n data: {\n level: (input.bytes[5] << 8) + input.bytes[6],\n volt: ((input.bytes[9] << 8) + input.bytes[10]) / 100,\n alarmPark: Boolean(input.bytes[7] >> 4),\n alarmLevel: Boolean(input.bytes[7] & 0x0f),\n alarmMagnet: Boolean(input.bytes[8] >> 4),\n alarmBattery: Boolean(input.bytes[8] & 0x0f),\n xMagnet: mag_x > 32767 ? mag_x - 65536 :\n mag_x,\n yMagnet: mag_y > 32767 ? mag_y - 65536 :\n mag_y,\n zMagnet: mag_z > 32767 ? mag_z - 65536 :\n mag_z,\n temperature: rawTemperature > 127 ? rawTemperature - 256 : rawTemperature,\n humidity: input.bytes[18],\n frameCounter: (input.bytes[19] << 8) + input.bytes[20],\n },\n };\n\n case 16:\n var data_type = input.bytes[3];\n if (data_type === 0x03) {\n return {\n // Decoded parameter\n data: {\n firmware: input.bytes[5] + \".\" + input.bytes[6],\n uploadInterval: input.bytes[7],\n detectInterval: input.bytes[8],\n levelThreshold: input.bytes[9],\n magnetThreshold: (input.bytes[10] << 8) + input.bytes[11],\n batteryThreshold: input.bytes[12],\n },\n };\n }\n default:\n return {\n errors: ['wrong length'],\n };\n }\n}\n\nfunction encodeDownlink(input) {\n if (input.data.uploadInterval !== null && !isNaN(input.data.uploadInterval)) {\n var periodic_interval = input.data.uploadInterval;\n var periodic_interval_high = periodic_interval.toString(16).padStart(2, '0').toUpperCase()[0].charCodeAt(0);\n var periodic_interval_low = periodic_interval.toString(16).padStart(2, '0').toUpperCase()[1].charCodeAt(0);\n if (periodic_interval > 168 || periodic_interval < 1) {\n return {\n errors: ['periodic upload interval range 1-168 hours.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x31, periodic_interval_high, periodic_interval_low, 0x38, 0x31],\n };\n }\n }\n if (input.data.detectInterval !== null && !isNaN(input.data.detectInterval)) {\n var detection_interval = input.data.detectInterval;\n var detection_interval_high = detection_interval.toString(16).padStart(2, '0').toUpperCase()[0].charCodeAt(0);\n var detection_interval_low = detection_interval.toString(16).padStart(2, '0').toUpperCase()[1].charCodeAt(0);\n if (detection_interval > 60 || detection_interval < 1) {\n return {\n errors: ['ultra detection interval range 1-60 minutes.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x38, detection_interval_high, detection_interval_low, 0x38, 0x31],\n };\n }\n }\n if (input.data.levelThreshold !== null && !isNaN(input.data.levelThreshold)) {\n var levelThreshold = input.data.levelThreshold;\n var levelThreshold_high = levelThreshold.toString(16).padStart(2, '0').toUpperCase()[0].charCodeAt(0);\n var levelThreshold_low = levelThreshold.toString(16).padStart(2, '0').toUpperCase()[1].charCodeAt(0);\n if (levelThreshold > 255 || levelThreshold < 15) {\n return {\n errors: ['battery alarm threshold range 15-255 %.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x32, levelThreshold_high, levelThreshold_low, 0x38, 0x31],\n };\n }\n }\n if (input.data.magnetThreshold !== null && !isNaN(input.data.magnetThreshold)) {\n var magnetThreshold = input.data.magnetThreshold;\n var magnetThreshold_1 = magnetThreshold.toString(16).padStart(4, '0').toUpperCase()[0].charCodeAt(0);\n var magnetThreshold_2 = magnetThreshold.toString(16).padStart(4, '0').toUpperCase()[1].charCodeAt(0);\n var magnetThreshold_3 = magnetThreshold.toString(16).padStart(4, '0').toUpperCase()[2].charCodeAt(0);\n var magnetThreshold_4 = magnetThreshold.toString(16).padStart(4, '0').toUpperCase()[3].charCodeAt(0);\n if (magnetThreshold > 65535 || magnetThreshold < 1) {\n return {\n errors: ['magnet threshold range 1-65535.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x46, magnetThreshold_1, magnetThreshold_2, magnetThreshold_3, magnetThreshold_4, 0x38, 0x31],\n };\n }\n\n\n }\n\n if (input.data.batteryThreshold !== null && !isNaN(input.data.batteryThreshold)) {\n var batteryThreshold = input.data.batteryThreshold;\n var batteryThreshold_high = batteryThreshold.toString(16).padStart(2, '0').toUpperCase()[0].charCodeAt(0);\n var batteryThreshold_low = batteryThreshold.toString(16).padStart(2, '0').toUpperCase()[1].charCodeAt(0);\n if (batteryThreshold > 99 || batteryThreshold < 5) {\n return {\n errors: ['battery alarm threshold range 5-99 %.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x35, batteryThreshold_high, batteryThreshold_low, 0x38, 0x31],\n };\n }\n }\n\n\n return {\n errors: ['invalid downlink parameter.'],\n };\n}\n\nfunction decodeDownlink(input) {\n var input_length = input.bytes.length;\n if (input.fPort != 3) {\n return {\n errors: ['invalid FPort.'],\n };\n }\n\n if (\n input_length < 12 ||\n input.bytes[0] != 0x38 ||\n input.bytes[1] != 0x30 ||\n input.bytes[2] != 0x30 ||\n input.bytes[3] != 0x32 ||\n input.bytes[4] != 0x39 ||\n input.bytes[5] != 0x39 ||\n input.bytes[6] != 0x39 ||\n input.bytes[7] != 0x39 ||\n input.bytes[input_length - 2] != 0x38 ||\n input.bytes[input_length - 1] != 0x31\n ) {\n return {\n errors: ['invalid format.'],\n };\n }\n var option = parseInt(String.fromCharCode(input.bytes[8]) + String.fromCharCode(input.bytes[9]), 16);\n var value = parseInt(String.fromCharCode(input.bytes[10]) + String.fromCharCode(input.bytes[11]), 16);\n var value2Bytes = parseInt(String.fromCharCode(input.bytes[10]) + String.fromCharCode(input.bytes[11]) + String.fromCharCode(input.bytes[12]) + String.fromCharCode(input.bytes[13]), 16);\n switch (option) {\n case 1:\n return {\n data: {\n uploadInterval: value,\n },\n };\n case 2:\n return {\n data: {\n levelThreshold: value,\n },\n };\n case 5:\n return {\n data: {\n batteryThreshold: value,\n },\n };\n case 8:\n return {\n data: {\n detectInterval: value,\n },\n };\n case 15:\n return {\n data: {\n magnetThreshold: value2Bytes,\n },\n };\n\n default:\n return {\n errors: ['invalid parameter key.'],\n };\n }\n}",
+ "environment": "javascript",
+ "storage": "",
+ "version": "1.0"
+ },
+ "flows": {
+ "dingtek_do201_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": "Overview",
+ "widgets": [
+ {
+ "layout": {
+ "col": 0,
+ "row": 0,
+ "sizeX": 3,
+ "sizeY": 6
+ },
+ "panel": {
+ "color": "#ffffff",
+ "currentColor": "#ffffff",
+ "showOffline": {
+ "type": "none"
+ },
+ "title": "Parking Status"
+ },
+ "properties": {
+ "source": "code",
+ "template": "
\n
{{ (value && value[0] && value[0].alarmPark) ? 'đ' : 'đĸ' }}
\n
\n {{ (value && value[0] && value[0].alarmPark) ? 'OCCUPIED' : 'FREE' }}\n
\n
\n \n {{ (value && value[0] && value[0].alarmLevel) ? 'â ī¸ Level Alarm' : 'Level OK' }}\n \n \n {{ (value && value[0] && value[0].alarmMagnet) ? '𧲠Magnet Alarm' : 'Magnet OK' }}\n \n
\n
Last update: {{ (value && value[0] && value[0].ts) | date:'medium' }}
\n
\n"
+ },
+ "sources": [
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "alarmPark",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e74c3c",
+ "name": "alarmPark",
+ "source": "bucket",
+ "timespan": {
+ "mode": "latest"
+ }
+ },
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "alarmLevel",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e67e22",
+ "name": "alarmLevel",
+ "source": "bucket",
+ "timespan": {
+ "mode": "latest"
+ }
+ },
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "alarmMagnet",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#9b59b6",
+ "name": "alarmMagnet",
+ "source": "bucket",
+ "timespan": {
+ "mode": "latest"
+ }
+ },
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "ts",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#95a5a6",
+ "name": "ts",
+ "source": "bucket",
+ "timespan": {
+ "mode": "latest"
+ }
+ }
+ ],
+ "type": "html_time"
+ },
+ {
+ "layout": {
+ "col": 3,
+ "row": 0,
+ "sizeX": 1,
+ "sizeY": 6
+ },
+ "panel": {
+ "color": "#ffffff",
+ "currentColor": "#ffffff",
+ "showOffline": {
+ "type": "none"
+ },
+ "title": "Temperature"
+ },
+ "properties": {
+ "color": "#e74c3c",
+ "max": 60,
+ "min": -20,
+ "unit": "°C"
+ },
+ "sources": [
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "temperature",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e74c3c",
+ "name": "Temperature",
+ "source": "bucket",
+ "timespan": {
+ "mode": "latest"
+ }
+ }
+ ],
+ "type": "donutchart"
+ },
+ {
+ "layout": {
+ "col": 4,
+ "row": 0,
+ "sizeX": 1,
+ "sizeY": 6
+ },
+ "panel": {
+ "color": "#ffffff",
+ "currentColor": "#ffffff",
+ "showOffline": {
+ "type": "none"
+ },
+ "title": "Humidity"
+ },
+ "properties": {
+ "color": "#3498db",
+ "max": 100,
+ "min": 0,
+ "unit": "%RH"
+ },
+ "sources": [
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "humidity",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#3498db",
+ "name": "Humidity",
+ "source": "bucket",
+ "timespan": {
+ "mode": "latest"
+ }
+ }
+ ],
+ "type": "donutchart"
+ },
+ {
+ "layout": {
+ "col": 5,
+ "row": 0,
+ "sizeX": 1,
+ "sizeY": 6
+ },
+ "panel": {
+ "color": "#ffffff",
+ "currentColor": "#ffffff",
+ "showOffline": {
+ "type": "none"
+ },
+ "title": "Device Status"
+ },
+ "properties": {
+ "source": "code",
+ "template": "\n
\n
{{ (value && value[0] && value[0].alarmBattery) ? 'đĒĢ' : 'đ' }}
\n
Battery
\n
{{ (value && value[0] && value[0].alarmBattery) ? 'Low' : 'OK' }}
\n
\n
\n
{{ (value && value[0] && value[0].volt !== undefined) ? (value[0].volt | number:2) + ' V' : 'â' }}
\n
Voltage
\n
\n
\n"
+ },
+ "sources": [
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "alarmBattery",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e67e22",
+ "name": "alarmBattery",
+ "source": "bucket",
+ "timespan": {
+ "mode": "latest"
+ }
+ },
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "volt",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#f1c40f",
+ "name": "volt",
+ "source": "bucket",
+ "timespan": {
+ "mode": "latest"
+ }
+ }
+ ],
+ "type": "html_time"
+ },
+ {
+ "layout": {
+ "col": 0,
+ "row": 6,
+ "sizeX": 3,
+ "sizeY": 7
+ },
+ "panel": {
+ "color": "#ffffff",
+ "currentColor": "#ffffff",
+ "showOffline": {
+ "type": "none"
+ },
+ "title": "Ultrasonic Level History"
+ },
+ "properties": {
+ "axis": true,
+ "fill": true,
+ "legend": true,
+ "multiple_axes": false
+ },
+ "sources": [
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "level",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#3498db",
+ "name": "Level (mm)",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ }
+ ],
+ "type": "chart"
+ },
+ {
+ "layout": {
+ "col": 3,
+ "row": 6,
+ "sizeX": 2,
+ "sizeY": 7
+ },
+ "panel": {
+ "color": "#ffffff",
+ "currentColor": "#ffffff",
+ "showOffline": {
+ "type": "none"
+ },
+ "title": "Temperature & Humidity History"
+ },
+ "properties": {
+ "axis": true,
+ "fill": false,
+ "legend": true,
+ "multiple_axes": true
+ },
+ "sources": [
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "temperature",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e74c3c",
+ "name": "Temperature (°C)",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "humidity",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#3498db",
+ "name": "Humidity (%RH)",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ }
+ ],
+ "type": "chart"
+ },
+ {
+ "layout": {
+ "col": 5,
+ "row": 6,
+ "sizeX": 1,
+ "sizeY": 7
+ },
+ "panel": {
+ "color": "#ffffff",
+ "currentColor": "#ffffff",
+ "showOffline": {
+ "type": "none"
+ },
+ "title": "Magnetic Field (X/Y/Z)"
+ },
+ "properties": {
+ "axis": true,
+ "fill": false,
+ "legend": true,
+ "multiple_axes": false
+ },
+ "sources": [
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "xMagnet",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e74c3c",
+ "name": "X Magnet",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "yMagnet",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#2ecc71",
+ "name": "Y Magnet",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "zMagnet",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#3498db",
+ "name": "Z Magnet",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ }
+ ],
+ "type": "chart"
+ },
+ {
+ "layout": {
+ "col": 0,
+ "row": 13,
+ "sizeX": 6,
+ "sizeY": 8
+ },
+ "panel": {
+ "color": "#ffffff",
+ "currentColor": "#ffffff",
+ "showOffline": {
+ "type": "none"
+ },
+ "title": "Last 24h Records"
+ },
+ "properties": {
+ "source": "code",
+ "template": "\n
\n \n \n | Date | \n Parking | \n Level (mm) | \n Temp (°C) | \n Hum (%RH) | \n Volt (V) | \n X Mag | \n Y Mag | \n Z Mag | \n Level Alarm | \n Magnet Alarm | \n Battery | \n Frame # | \n
\n \n \n \n | {{ entry.ts | date:'dd/MM/yy HH:mm' }} | \n {{ entry.alarmPark ? 'đ OCC' : 'đĸ FREE' }} | \n {{ entry.level !== undefined ? entry.level : 'â' }} | \n {{ entry.temperature !== undefined ? entry.temperature : 'â' }} | \n {{ entry.humidity !== undefined ? entry.humidity : 'â' }} | \n {{ entry.volt !== undefined ? (entry.volt | number:2) : 'â' }} | \n {{ entry.xMagnet !== undefined ? entry.xMagnet : 'â' }} | \n {{ entry.yMagnet !== undefined ? entry.yMagnet : 'â' }} | \n {{ entry.zMagnet !== undefined ? entry.zMagnet : 'â' }} | \n {{ entry.alarmLevel ? 'â ī¸ YES' : 'â
NO' }} | \n {{ entry.alarmMagnet ? '𧲠YES' : 'â
NO' }} | \n {{ entry.alarmBattery ? 'đĒĢ Low' : 'đ OK' }} | \n {{ entry.frameCounter !== undefined ? entry.frameCounter : 'â' }} | \n
\n \n
\n
\n"
+ },
+ "sources": [
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "ts",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#95a5a6",
+ "name": "ts",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "alarmPark",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e74c3c",
+ "name": "alarmPark",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "level",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#3498db",
+ "name": "level",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "temperature",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e74c3c",
+ "name": "temperature",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "humidity",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#3498db",
+ "name": "humidity",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "volt",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#f1c40f",
+ "name": "volt",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "xMagnet",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e74c3c",
+ "name": "xMagnet",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "yMagnet",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#2ecc71",
+ "name": "yMagnet",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "zMagnet",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#3498db",
+ "name": "zMagnet",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "alarmLevel",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#e67e22",
+ "name": "alarmLevel",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "alarmMagnet",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#9b59b6",
+ "name": "alarmMagnet",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "alarmBattery",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#f39c12",
+ "name": "alarmBattery",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ },
+ {
+ "aggregation": {},
+ "bucket": {
+ "backend": "mongodb",
+ "id": "dingtek_do201_data",
+ "mapping": "frameCounter",
+ "tags": {
+ "device": [],
+ "group": []
+ }
+ },
+ "color": "#1abc9c",
+ "name": "frameCounter",
+ "source": "bucket",
+ "timespan": {
+ "magnitude": "hour",
+ "mode": "relative",
+ "period": "latest",
+ "value": 24
+ }
+ }
+ ],
+ "type": "html_time"
+ }
+ ]
+ }
+ ]
+ }
+ }
+ ]
}
- }
- }
- },
- "autoprovisions": {
- "device_autoprovisioning": {
- "config": {
- "mode": "pattern",
- "pattern": "do201_.*"
- },
- "enabled": true
- }
- },
- "buckets": {
- "dingtek_do201_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\nvar units = [' â', ' hours', ' minutes', ' mm', ' °', ' cm'];\n//IEEE754 hex to float convert\nfunction hex2float(num) {\n var sign = num & 0x80000000 ? -1 : 1;\n var exponent = ((num >> 23) & 0xff) - 127;\n var mantissa = 1 + (num & 0x7fffff) / 0x7fffff;\n return sign * mantissa * Math.pow(2, exponent);\n}\n\nfunction decodeUplink(input) {\n if (input.fPort != 3) {\n return {\n errors: ['unknown FPort'],\n };\n }\n\n switch (input.bytes.length) {\n case 23:\n var mag_x = (input.bytes[11] << 8) + input.bytes[12];\n var mag_y = (input.bytes[13] << 8) + input.bytes[14];\n var mag_z = (input.bytes[15] << 8) + input.bytes[16];\n var rawTemperature = input.bytes[17];\n return {\n // Decoded data\n data: {\n level: (input.bytes[5] << 8) + input.bytes[6],\n volt: ((input.bytes[9] << 8) + input.bytes[10]) / 100,\n alarmPark: Boolean(input.bytes[7] >> 4),\n alarmLevel: Boolean(input.bytes[7] & 0x0f),\n alarmMagnet: Boolean(input.bytes[8] >> 4),\n alarmBattery: Boolean(input.bytes[8] & 0x0f),\n xMagnet: mag_x > 32767 ? mag_x - 65536 :\n mag_x,\n yMagnet: mag_y > 32767 ? mag_y - 65536 :\n mag_y,\n zMagnet: mag_z > 32767 ? mag_z - 65536 :\n mag_z,\n temperature: rawTemperature > 127 ? rawTemperature - 256 : rawTemperature,\n humidity: input.bytes[18],\n frameCounter: (input.bytes[19] << 8) + input.bytes[20],\n },\n };\n\n case 16:\n var data_type = input.bytes[3];\n if (data_type === 0x03) {\n return {\n // Decoded parameter\n data: {\n firmware: input.bytes[5] + \".\" + input.bytes[6],\n uploadInterval: input.bytes[7],\n detectInterval: input.bytes[8],\n levelThreshold: input.bytes[9],\n magnetThreshold: (input.bytes[10] << 8) + input.bytes[11],\n batteryThreshold: input.bytes[12],\n },\n };\n }\n default:\n return {\n errors: ['wrong length'],\n };\n }\n}\n\nfunction encodeDownlink(input) {\n if (input.data.uploadInterval !== null && !isNaN(input.data.uploadInterval)) {\n var periodic_interval = input.data.uploadInterval;\n var periodic_interval_high = periodic_interval.toString(16).padStart(2, '0').toUpperCase()[0].charCodeAt(0);\n var periodic_interval_low = periodic_interval.toString(16).padStart(2, '0').toUpperCase()[1].charCodeAt(0);\n if (periodic_interval > 168 || periodic_interval < 1) {\n return {\n errors: ['periodic upload interval range 1-168 hours.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x31, periodic_interval_high, periodic_interval_low, 0x38, 0x31],\n };\n }\n }\n if (input.data.detectInterval !== null && !isNaN(input.data.detectInterval)) {\n var detection_interval = input.data.detectInterval;\n var detection_interval_high = detection_interval.toString(16).padStart(2, '0').toUpperCase()[0].charCodeAt(0);\n var detection_interval_low = detection_interval.toString(16).padStart(2, '0').toUpperCase()[1].charCodeAt(0);\n if (detection_interval > 60 || detection_interval < 1) {\n return {\n errors: ['ultra detection interval range 1-60 minutes.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x38, detection_interval_high, detection_interval_low, 0x38, 0x31],\n };\n }\n }\n if (input.data.levelThreshold !== null && !isNaN(input.data.levelThreshold)) {\n var levelThreshold = input.data.levelThreshold;\n var levelThreshold_high = levelThreshold.toString(16).padStart(2, '0').toUpperCase()[0].charCodeAt(0);\n var levelThreshold_low = levelThreshold.toString(16).padStart(2, '0').toUpperCase()[1].charCodeAt(0);\n if (levelThreshold > 255 || levelThreshold < 15) {\n return {\n errors: ['battery alarm threshold range 15-255 %.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x32, levelThreshold_high, levelThreshold_low, 0x38, 0x31],\n };\n }\n }\n if (input.data.magnetThreshold !== null && !isNaN(input.data.magnetThreshold)) {\n var magnetThreshold = input.data.magnetThreshold;\n var magnetThreshold_1 = magnetThreshold.toString(16).padStart(4, '0').toUpperCase()[0].charCodeAt(0);\n var magnetThreshold_2 = magnetThreshold.toString(16).padStart(4, '0').toUpperCase()[1].charCodeAt(0);\n var magnetThreshold_3 = magnetThreshold.toString(16).padStart(4, '0').toUpperCase()[2].charCodeAt(0);\n var magnetThreshold_4 = magnetThreshold.toString(16).padStart(4, '0').toUpperCase()[3].charCodeAt(0);\n if (magnetThreshold > 65535 || magnetThreshold < 1) {\n return {\n errors: ['magnet threshold range 1-65535.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x46, magnetThreshold_1, magnetThreshold_2, magnetThreshold_3, magnetThreshold_4, 0x38, 0x31],\n };\n }\n\n\n }\n\n if (input.data.batteryThreshold !== null && !isNaN(input.data.batteryThreshold)) {\n var batteryThreshold = input.data.batteryThreshold;\n var batteryThreshold_high = batteryThreshold.toString(16).padStart(2, '0').toUpperCase()[0].charCodeAt(0);\n var batteryThreshold_low = batteryThreshold.toString(16).padStart(2, '0').toUpperCase()[1].charCodeAt(0);\n if (batteryThreshold > 99 || batteryThreshold < 5) {\n return {\n errors: ['battery alarm threshold range 5-99 %.'],\n };\n } else {\n return {\n // LoRaWAN FPort used for the downlink message\n fPort: 3,\n // Encoded bytes\n bytes: [0x38, 0x30, 0x30, 0x32, 0x39, 0x39, 0x39, 0x39, 0x30, 0x35, batteryThreshold_high, batteryThreshold_low, 0x38, 0x31],\n };\n }\n }\n\n\n return {\n errors: ['invalid downlink parameter.'],\n };\n}\n\nfunction decodeDownlink(input) {\n var input_length = input.bytes.length;\n if (input.fPort != 3) {\n return {\n errors: ['invalid FPort.'],\n };\n }\n\n if (\n input_length < 12 ||\n input.bytes[0] != 0x38 ||\n input.bytes[1] != 0x30 ||\n input.bytes[2] != 0x30 ||\n input.bytes[3] != 0x32 ||\n input.bytes[4] != 0x39 ||\n input.bytes[5] != 0x39 ||\n input.bytes[6] != 0x39 ||\n input.bytes[7] != 0x39 ||\n input.bytes[input_length - 2] != 0x38 ||\n input.bytes[input_length - 1] != 0x31\n ) {\n return {\n errors: ['invalid format.'],\n };\n }\n var option = parseInt(String.fromCharCode(input.bytes[8]) + String.fromCharCode(input.bytes[9]), 16);\n var value = parseInt(String.fromCharCode(input.bytes[10]) + String.fromCharCode(input.bytes[11]), 16);\n var value2Bytes = parseInt(String.fromCharCode(input.bytes[10]) + String.fromCharCode(input.bytes[11]) + String.fromCharCode(input.bytes[12]) + String.fromCharCode(input.bytes[13]), 16);\n switch (option) {\n case 1:\n return {\n data: {\n uploadInterval: value,\n },\n };\n case 2:\n return {\n data: {\n levelThreshold: value,\n },\n };\n case 5:\n return {\n data: {\n batteryThreshold: value,\n },\n };\n case 8:\n return {\n data: {\n detectInterval: value,\n },\n };\n case 15:\n return {\n data: {\n magnetThreshold: value2Bytes,\n },\n };\n\n default:\n return {\n errors: ['invalid parameter key.'],\n };\n }\n}",
- "environment": "javascript",
- "storage": "",
- "version": "1.0"
- },
- "flows": {
- "dingtek_do201_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": []
- }
- }
- ]
- }
-}
\ No newline at end of file
+ ]
+ }
+}