Multitech
Creation procedure
Import steps
Multitech BACnet Codec file
// DataCake
function Decoder(bytes, port){
var decoded = decodeUplink({ bytes: bytes, fPort: port }).data;
return decoded;
}
// Milesight
function Decode(port, bytes){
var decoded = decodeUplink({ bytes: bytes, fPort: port }).data;
return decoded;
}
// The Things Industries / Main
function decodeUplink(input) {
try {
var bytes = input.bytes;
var data = {};
function handleKeepalive(bytes, data) {
// Byte 1 bit 2: Occupied flag
var occupiedValue = (bytes[1] & 0x04) >> 2;
data.occupied = occupiedValue === 1;
// Byte 1 (bits 1:0) and Byte 2: Internal temperature sensor data
// Formula: t[°C] = (T[9:0] - 400) / 10
// Extract bits 1:0 from byte 1 for the higher bits (bits 9:8)
var tempHighBits = (bytes[1] & 0x03) << 8;
// Use all bits from byte 2 for the lower bits (bits 7:0)
var tempLowBits = bytes[2];
// Combine to get the full 10-bit temperature value
var tempValue = tempHighBits | tempLowBits;
data.sensorTemperature = Number(((tempValue - 400) / 10).toFixed(2));
// Byte 3: Relative Humidity data
// Formula: RH[%] = (XX * 100) / 256
data.relativeHumidity = Number(((bytes[3] * 100) / 256).toFixed(2));
// Byte 4: Device battery voltage data
// Battery voltage [mV] = ((XX * 2200) / 255) + 1600
data.batteryVoltage = Number(((((bytes[4] * 2200) / 255) + 1600) / 1000).toFixed(2));
// Bytes 5-6: CO2 value in ppm
// Byte 5: CO2 value lower bits [7:0]
// Byte 6 bits 7:3: CO2 value higher bits [12:8]
var co2LowBits = bytes[5];
var co2HighBits = ((bytes[6] & 0xF8) >> 3) << 8; // Mask upper 5 bits, shift right by 3 to get bits in position, then shift left by 8
data.CO2 = co2HighBits | co2LowBits;
// Byte 7: PIR trigger count
data.pirTriggerCount = bytes[7];
// For backward compatibility
var pirValue = (bytes[1] & 0x04) >> 2;
data.pirSensorStatus = pirValue === 1 ? "Motion detected" : "No motion detected";
data.pirSensorValue = pirValue;
return data;
}
function handleResponse(bytes, data){
var commands = bytes.map(function(byte){
return ("0" + byte.toString(16)).substr(-2);
});
commands = commands.slice(0,-8); // Adjust based on CO2-PIR-Lite keepalive message length (8 bytes)
var command_len = 0;
commands.map(function (command, i) {
switch (command) {
case '04':
{
command_len = 2;
var hardwareVersion = commands[i + 1];
var softwareVersion = commands[i + 2];
data.deviceVersions = { hardware: Number(hardwareVersion), software: Number(softwareVersion) };
}
break;
case '12':
{
command_len = 1;
data.keepAliveTime = parseInt(commands[i + 1], 16);
}
break;
case '19':
{
command_len = 1;
var commandResponse = parseInt(commands[i + 1], 16);
var periodInMinutes = commandResponse * 5 / 60;
data.joinRetryPeriod = periodInMinutes;
}
break;
case '1b':
{
command_len = 1;
data.uplinkType = parseInt(commands[i + 1], 16);
}
break;
case '1d':
{
command_len = 2;
var wdpC = commands[i + 1] == '00' ? false : parseInt(commands[i + 1], 16);
var wdpUc = commands[i + 2] == '00' ? false : parseInt(commands[i + 2], 16);
data.watchDogParams= { wdpC: wdpC, wdpUc: wdpUc } ;
}
break;
case '1f':
{
command_len = 4;
var good_medium = (parseInt(commands[i + 1], 16) << 8) |
parseInt(commands[i + 2], 16);
var medium_bad = (parseInt(commands[i + 3], 16) << 8) |
parseInt(commands[i + 4], 16);
data.boundaryLevels = { good_medium: Number(good_medium), medium_bad: Number(medium_bad) };
}
break;
case '21':
{
command_len = 2;
data.autoZeroValue = (parseInt(commands[i + 1], 16) << 8) |
parseInt(commands[i + 2], 16);
}
break;
case '25':
{
command_len = 3;
var good_zone = parseInt(commands[i + 1], 16);
var medium_zone = parseInt(commands[i + 2], 16);
var bad_zone = parseInt(commands[i + 3], 16);
data.measurementPeriod = { good_zone: Number(good_zone), medium_zone: Number(medium_zone), bad_zone: Number(bad_zone) };
}
break;
case '2b':
{
command_len = 1;
data.autoZeroPeriod = parseInt(commands[i + 1], 16);
}
break;
case '2f':
{
command_len = 1;
data.uplinkSendingOnButtonPress = parseInt(commands[i + 1], 16) ;
}
break;
case '3d':
{
command_len = 1;
data.pirSensorStatus = parseInt(commands[i + 1], 16);
}
break;
case '3f':
{
command_len = 1;
data.pirSensorSensitivity = parseInt(commands[i + 1], 16);
}
break;
case '49':
{
command_len = 1;
data.pirMeasurementPeriod = parseInt(commands[i + 1], 16);
}
break;
case '4b':
{
command_len = 1;
data.pirCheckPeriod = parseInt(commands[i + 1], 16);
}
break;
case '37':
{
command_len = 1;
data.pirSensorState = parseInt(commands[i + 1], 16);
}
break;
case '39':
{
command_len = 2;
data.occupancyTimeout = (parseInt(commands[i + 1], 16) << 8) | parseInt(commands[i + 2], 16);
}
break;
case '4d':
{
command_len = 1;
data.pirBlindPeriod = parseInt(commands[i + 1], 16);
}
break;
case 'a4': {
command_len = 1;
data.region = parseInt(commands[i + 1], 16);
break;
}
default:
break;
}
commands.splice(i,command_len);
});
return data;
}
// Route the message based on the command byte
if (bytes[0] == 81) {
// This is a keepalive message
data = handleKeepalive(bytes, data);
} else {
// This is a response message
data = handleResponse(bytes, data);
// Handle the remaining keepalive data if present
if (bytes.length >= 8) { // Check if there's enough bytes for a keepalive message (8 bytes for CO2-PIR-Lite)
bytes = bytes.slice(-8); // Extract the last 8 bytes which contain keepalive data
data = handleKeepalive(bytes, data);
}
}
return { data: data };
} catch (e) {
// console.log(e);
throw new Error('Unhandled data');
}
}
// Milesight
function Encode(port, obj) {
var encoded = encodeDownlink({ fPort: port, data: obj }).bytes;
return encoded;
}
function Encoder(port, obj) {
var encoded = encodeDownlink({ fPort: port, data: obj }).bytes;
return encoded;
}
// The Things Industries / Main
function encodeDownlink(input) {
var bytes = [];
for (key in input.data) {
switch (key) {
case "setKeepAlive": {
bytes.push(0x02);
bytes.push(input.data.setKeepAlive);
break;
}
case "getKeepAliveTime": {
bytes.push(0x12);
break;
}
case "getDeviceVersions": {
bytes.push(0x04);
break;
}
case "setJoinRetryPeriod": {
var periodToPass = (input.data.setJoinRetryPeriod * 60) / 5;
periodToPass = int(periodToPass);
bytes.push(0x10);
bytes.push(periodToPass);
break;
}
case "getJoinRetryPeriod": {
bytes.push(0x19);
break;
}
case "setUplinkType": {
bytes.push(0x11);
bytes.push(input.data.setUplinkType);
break;
}
case "getUplinkType": {
bytes.push(0x1B);
break;
}
case "setWatchDogParams": {
bytes.push(0x1C);
bytes.push(input.data.SetWatchDogParams.confirmedUplinks);
bytes.push(input.data.SetWatchDogParams.unconfirmedUplinks);
break;
}
case "getWatchDogParams": {
bytes.push(0x1D);
break;
}
case "setCo2BoundaryLevels": {
var good_medium = input.data.setCo2BoundaryLevels.good_medium;
var medium_bad = input.data.setCo2BoundaryLevels.medium_bad;
var good_mediumFirstPart = good_medium & 0xff;
var good_mediumSecondPart = (good_medium >> 8) & 0xff;
var medium_badFirstPart = medium_bad & 0xff;
var medium_badSecondPart = (medium_bad >> 8) & 0xff;
bytes.push(0x1E);
bytes.push(good_mediumSecondPart);
bytes.push(good_mediumFirstPart);
bytes.push(medium_badSecondPart);
bytes.push(medium_badFirstPart);
break;
}
case "getCo2BoundaryLevels": {
bytes.push(0x1F);
break;
}
case "setAutoZeroValue": {
var autoZeroValue = input.data.setAutoZeroValue;
var autoZeroValueFirstPart = autoZeroValue & 0xff;
var autoZeroValueSecondPart = (autoZeroValue >> 8) & 0xff;
bytes.push(0x20);
bytes.push(autoZeroValueSecondPart);
bytes.push(autoZeroValueFirstPart);
break;
}
case "getAutoZeroValue": {
bytes.push(0x21);
break;
}
case "setCo2MeasurementPeriod": {
var good_zone = input.data.setCo2MeasurementPeriod.good_zone;
var medium_zone = input.data.setCo2MeasurementPeriod.medium_zone;
var bad_zone = input.data.setCo2MeasurementPeriod.bad_zone;
bytes.push(0x24);
bytes.push(good_zone);
bytes.push(medium_zone);
bytes.push(bad_zone);
break;
}
case "getCo2MeasurementPeriod": {
bytes.push(0x25);
break;
}
case "setCo2AutoZeroPeriod": {
bytes.push(0x2A);
bytes.push(input.data.setCo2AutoZeroPeriod);
break;
}
case "getCo2AutoZeroPeriod": {
bytes.push(0x2B);
break;
}
case "setUplinkSendingOnButtonPress": {
bytes.push(0x2E);
bytes.push(input.data.setUplinkSendingOnButtonPress);
break;
}
case "getUplinkSendingOnButtonPress": {
bytes.push(0x2F);
break;
}
case "setPIRSensorStatus": {
bytes.push(0x3C);
bytes.push(input.data.setPIRSensorStatus);
break;
}
case "getPIRSensorStatus": {
bytes.push(0x3D);
break;
}
case "setPIRSensorSensitivity": {
bytes.push(0x3E);
bytes.push(input.data.setPIRSensorSensitivity);
break;
}
case "getPIRSensorSensitivity": {
bytes.push(0x3F);
break;
}
case "setPIRMeasurementPeriod": {
bytes.push(0x48);
bytes.push(input.data.setPIRMeasurementPeriod);
break;
}
case "getPIRMeasurementPeriod": {
bytes.push(0x49);
break;
}
case "setPIRCheckPeriod": {
var time = input.data.setPIRCheckPeriod;
var timeFirstPart = time & 0xff;
var timeSecondPart = (time >> 8) & 0xff;
bytes.push(0x4A);
bytes.push(timeSecondPart);
bytes.push(timeFirstPart);
break;
}
case "getPIRCheckPeriod": {
bytes.push(0x4B);
break;
}
case "setPIRBlindPeriod": {
var time = input.data.setPIRBlindPeriod;
var timeFirstPart = time & 0xff;
var timeSecondPart = (time >> 8) & 0xff;
bytes.push(0x4C);
bytes.push(timeSecondPart);
bytes.push(timeFirstPart);
break;
}
case "getPIRBlindPeriod": {
bytes.push(0x4D);
break;
}
case "sendCustomHexCommand": {
var sendCustomHexCommand = input.data.sendCustomHexCommand;
for (var i = 0; i < sendCustomHexCommand.length; i += 2) {
var byte = parseInt(sendCustomHexCommand.substr(i, 2), 16);
bytes.push(byte);
}
break;
}
default: {
}
}
}
return {
bytes: bytes,
fPort: 1,
warnings: [],
errors: [],
};
}
function decodeDownlink(input) {
return {
data: {
bytes: input.bytes,
},
warnings: [],
errors: [],
};
}
// example downlink commands
// {"setPIRBlindPeriod":20} --> 0x4C0014
// {"setCo2MeasurementPeriod":{"good_zone":15,"medium_zone":21,"bad_zone":21}}
// {"sendCustomHexCommand":"080F15"} --> 0x080F15
Multitech BACnet Definition file
Last updated
Was this helpful?