MClimate Wireless Thermostat Uplink decoder
TTN V3 Decoder (JavaScript ES5):
function decodeUplink(input) {
try{
var bytes = input.bytes;
var data = {};
const toBool = value => value == '1';
var calculateTemperature = function (rawData){return (rawData - 400) / 10};
var calculateHumidity = function(rawData){return (rawData * 100) / 256};
var decbin = function (number) {
if (number < 0) {
number = 0xFFFFFFFF + number + 1
}
number = number.toString(2);
return "00000000".substr(number.length) + number;
}
function handleKeepalive(bytes, data){
var tempRaw = (bytes[1] << 8) | bytes[2];
var temperature = calculateTemperature(tempRaw);
var humidity = calculateHumidity(bytes[3]);
let batteryVoltage = ((bytes[4] << 8) | bytes[5])/1000;
var targetTemperature, powerSourceStatus, lux, pir;
if(bytes[0] == 1){
targetTemperature = bytes[6];
powerSourceStatus = bytes[7];
lux = (bytes[8] << 8) | bytes[9];
pir = toBool(bytes[10]);
}else{
targetTemperature = parseInt(`${decbin(bytes[6])}${decbin(bytes[7])}`, 2)/10;
powerSourceStatus = bytes[8];
lux = (bytes[9] << 8) | bytes[10];
pir = toBool(bytes[11]);
}
data.sensorTemperature = Number(temperature.toFixed(2));
data.relativeHumidity = Number(humidity.toFixed(2));
data.batteryVoltage = Number(batteryVoltage.toFixed(3));
data.targetTemperature = targetTemperature;
data.powerSourceStatus = powerSourceStatus;
data.lux = lux;
data.pir = pir;
return data;
}
function handleResponse(bytes, data, keepaliveLength){
var commands = bytes.map(function(byte){
return ("0" + byte.toString(16)).substr(-2);
});
commands = commands.slice(0,-keepaliveLength);
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 '14':
{
command_len = 1;
data.childLock = toBool(parseInt(commands[i + 1], 16)) ;
}
break;
case '15':
{
command_len = 2;
data.temperatureRangeSettings = { min: parseInt(commands[i + 1], 16), max: parseInt(commands[i + 2], 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 '2f':
{
command_len = 1;
data.targetTemperature = parseInt(commands[i + 1], 16) ;
}
break;
case '30':
{
command_len = 1;
data.manualTargetTemperatureUpdate = parseInt(commands[i + 1], 16) ;
}
break;
case '32':
{
command_len = 1;
data.heatingStatus = parseInt(commands[i + 1], 16) ;
}
break;
case '34':
{
command_len = 1;
data.displayRefreshPeriod = parseInt(commands[i + 1], 16) ;
}
break;
case '36':
{
command_len = 1;
data.sendTargetTempDelay = parseInt(commands[i + 1], 16) ;
}
break;
case '38':
{
command_len = 1;
data.automaticHeatingStatus = parseInt(commands[i + 1], 16) ;
}
break;
case '3a':
{
command_len = 1;
data.sensorMode = 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 '41':
{
command_len = 1;
data.currentTemperatureVisibility = parseInt(commands[i + 1], 16) ;
}
break;
case '43':
{
command_len = 1;
data.humidityVisibility = parseInt(commands[i + 1], 16) ;
}
break;
case '45':
{
command_len = 1;
data.lightIntensityVisibility = parseInt(commands[i + 1], 16) ;
}
break;
case '47':
{
command_len = 1;
data.pirInitPeriod = 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 '4d':
{
command_len = 1;
data.pirBlindPeriod = parseInt(commands[i + 1], 16) ;
}
break;
case '4f':
{
command_len = 1;
data.temperatureHysteresis = parseInt(commands[i + 1], 16)/10 ;
}
break;
case '51':
{
command_len = 2;
data.targetTemperature = parseInt(`0x${commands[i + 1]}${commands[i + 2]}`, 16)/10 ;
}
break;
case '53':
{
command_len = 1;
data.targetTemperatureStep = parseInt(commands[i + 1], 16) / 10;
}
break;
case '54':
{
command_len = 2;
data.manualTargetTemperatureUpdate = parseInt(`0x${commands[i + 1]}${commands[i + 2]}`, 16)/10;
}
break;
case 'a0':
{
command_len = 4;
var fuota_address = parseInt(`${commands[i + 1]}${commands[i + 2]}${commands[i + 3]}${commands[i + 4]}`, 16)
var fuota_address_raw = `${commands[i + 1]}${commands[i + 2]}${commands[i + 3]}${commands[i + 4]}`
data.fuota = { fuota_address, fuota_address_raw };
}
break;
default:
break;
}
commands.splice(i,command_len);
});
return data;
}
if (bytes[0] == 1|| bytes[0] == 129) {
data = handleKeepalive(bytes, data);
}else{
var keepaliveLength = 11;
var potentialKeepAlive = bytes.slice(-12);
if(potentialKeepAlive[0] == 129) keepaliveLength = 12;
data = handleResponse(bytes,data, keepaliveLength);
bytes = bytes.slice(-keepaliveLength);
data = handleKeepalive(bytes, data);
}
return {data: data};
} catch (e) {
console.log(e)
throw new Error('Unhandled data');
}
}
Milesight Decoder
function Decode(port, bytes) {
try {
var data = {};
function toBool(value) {
return value == '1';
}
function calculateTemperature(rawData) {
return (rawData - 400) / 10;
}
function calculateHumidity(rawData) {
return (rawData * 100) / 256;
}
function decbin(number) {
if (number < 0) {
number = 0xFFFFFFFF + number + 1;
}
number = number.toString(2);
return "00000000".substr(number.length) + number;
}
function handleKeepalive(bytes, data) {
var tempRaw = (bytes[1] << 8) | bytes[2];
var temperature = calculateTemperature(tempRaw);
var humidity = calculateHumidity(bytes[3]);
var batteryVoltage = ((bytes[4] << 8) | bytes[5]) / 1000;
var targetTemperature, powerSourceStatus, lux, pir;
if (bytes[0] == 1) {
targetTemperature = bytes[6];
powerSourceStatus = bytes[7];
lux = (bytes[8] << 8) | bytes[9];
pir = toBool(bytes[10]);
} else {
targetTemperature = parseInt(decbin(bytes[6]) + decbin(bytes[7]), 2) / 10;
powerSourceStatus = bytes[8];
lux = (bytes[9] << 8) | bytes[10];
pir = toBool(bytes[11]);
}
data.sensorTemperature = Number(temperature.toFixed(2));
data.relativeHumidity = Number(humidity.toFixed(2));
data.batteryVoltage = Number(batteryVoltage.toFixed(3));
data.targetTemperature = targetTemperature;
data.powerSourceStatus = powerSourceStatus;
data.lux = lux;
data.pir = pir;
return data;
}
function handleResponse(bytes, data, keepaliveLength) {
var commands = bytes.map(function (byte) {
return ("0" + byte.toString(16)).substr(-2);
});
commands = commands.slice(0, -keepaliveLength);
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 '14':
command_len = 1;
data.childLock = toBool(parseInt(commands[i + 1], 16));
break;
case '15':
command_len = 2;
data.temperatureRangeSettings = { min: parseInt(commands[i + 1], 16), max: parseInt(commands[i + 2], 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 '2f':
command_len = 1;
data.targetTemperature = parseInt(commands[i + 1], 16);
break;
case '30':
command_len = 1;
data.manualTargetTemperatureUpdate = parseInt(commands[i + 1], 16);
break;
case '32':
command_len = 1;
data.heatingStatus = parseInt(commands[i + 1], 16);
break;
case '34':
command_len = 1;
data.displayRefreshPeriod = parseInt(commands[i + 1], 16);
break;
case '36':
command_len = 1;
data.sendTargetTempDelay = parseInt(commands[i + 1], 16);
break;
case '38':
command_len = 1;
data.automaticHeatingStatus = parseInt(commands[i + 1], 16);
break;
case '3a':
command_len = 1;
data.sensorMode = 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 '41':
command_len = 1;
data.currentTemperatureVisibility = parseInt(commands[i + 1], 16);
break;
case '43':
command_len = 1;
data.humidityVisibility = parseInt(commands[i + 1], 16);
break;
case '45':
command_len = 1;
data.lightIntensityVisibility = parseInt(commands[i + 1], 16);
break;
case '47':
command_len = 1;
data.pirInitPeriod = 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 '4d':
command_len = 1;
data.pirBlindPeriod = parseInt(commands[i + 1], 16);
break;
case '4f':
command_len = 1;
data.temperatureHysteresis = parseInt(commands[i + 1], 16) / 10;
break;
case '51':
command_len = 2;
data.targetTemperature = parseInt("0x" + commands[i + 1] + commands[i + 2], 16) / 10;
break;
case '53':
command_len = 1;
data.targetTemperatureStep = parseInt(commands[i + 1], 16) / 10;
break;
case '54':
command_len = 2;
data.manualTargetTemperatureUpdate = parseInt("0x" + commands[i + 1] + commands[i + 2], 16) / 10;
break;
case 'a0':
command_len = 4;
var fuota_address = parseInt(commands[i + 1] + commands[i + 2] + commands[i + 3] + commands[i + 4], 16);
var fuota_address_raw = commands[i + 1] + commands[i + 2] + commands[i + 3] + commands[i + 4];
data.fuota = { fuota_address: fuota_address, fuota_address_raw: fuota_address_raw };
break;
default:
break;
}
commands.splice(i, command_len);
});
return data;
}
if (bytes[0] == 1 || bytes[0] == 129) {
data = handleKeepalive(bytes, data);
} else {
var keepaliveLength = 11;
var potentialKeepAlive = bytes.slice(-12);
if (potentialKeepAlive[0] == 129) keepaliveLength = 12;
data = handleResponse(bytes, data, keepaliveLength);
bytes = bytes.slice(-keepaliveLength);
data = handleKeepalive(bytes, data);
}
return { data: data };
} catch (e) {
console.log(e);
throw new Error('Unhandled data');
}
}
DataCake Decoder
function decodeUplink(input) {
var bytes = input.bytes;
var data = {};
var resultToPass = {};
var toBool = function(value) { return value == '1'; };
function merge_obj(obj1, obj2) {
var obj3 = {};
for (var attrname in obj1) { obj3[attrname] = obj1[attrname]; }
for (var attrname2 in obj2) { obj3[attrname2] = obj2[attrname2]; }
return obj3;
}
var calculateTemperature = function(rawData) { return (rawData - 400) / 10; };
var calculateHumidity = function(rawData) { return (rawData * 100) / 256; };
var decbin = function(number) {
if (number < 0) {
number = 0xFFFFFFFF + number + 1;
}
number = number.toString(2);
return "00000000".substr(number.length) + number;
};
function handleKeepalive(bytes, data) {
var tempRaw = (bytes[1] << 8) | bytes[2];
var temperature = calculateTemperature(tempRaw);
var humidity = calculateHumidity(bytes[3]);
var batteryVoltage = ((bytes[4] << 8) | bytes[5]) / 1000;
var targetTemperature, powerSourceStatus, lux, pir;
if (bytes[0] == 1) {
targetTemperature = bytes[6];
powerSourceStatus = bytes[7];
lux = (bytes[8] << 8) | bytes[9];
pir = toBool(bytes[10]);
} else {
targetTemperature = parseInt(decbin(bytes[6]) + decbin(bytes[7]), 2) / 10;
powerSourceStatus = bytes[8];
lux = (bytes[9] << 8) | bytes[10];
pir = toBool(bytes[11]);
}
data.sensorTemperature = Number(temperature.toFixed(2));
data.relativeHumidity = Number(humidity.toFixed(2));
data.batteryVoltage = Number(batteryVoltage.toFixed(3));
data.targetTemperature = targetTemperature;
data.powerSourceStatus = powerSourceStatus;
data.lux = lux;
data.pir = pir;
return data;
}
function handleResponse(bytes, data, keepaliveLength) {
var commands = bytes.map(function(byte) {
return ("0" + byte.toString(16)).substr(-2);
});
commands = commands.slice(0, -keepaliveLength);
var command_len = 0;
commands.forEach(function(command, i) {
switch (command) {
case '04':
command_len = 2;
var hardwareVersion = commands[i + 1];
var softwareVersion = commands[i + 2];
var dataK = { deviceVersions: { hardware: Number(hardwareVersion), software: Number(softwareVersion) } };
resultToPass = merge_obj(resultToPass, dataK);
break;
case '12':
command_len = 1;
var dataC = { keepAliveTime: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataC);
break;
case '14':
command_len = 1;
var dataB = { childLock: toBool(parseInt(commands[i + 1], 16)) };
resultToPass = merge_obj(resultToPass, dataB);
break;
case '15':
command_len = 2;
var dataA = { temperatureRangeSettings: { min: parseInt(commands[i + 1], 16), max: parseInt(commands[i + 2], 16) } };
resultToPass = merge_obj(resultToPass, dataA);
break;
case '19':
command_len = 1;
var commandResponse = parseInt(commands[i + 1], 16);
var periodInMinutes = commandResponse * 5 / 60;
var dataH = { joinRetryPeriod: periodInMinutes };
resultToPass = merge_obj(resultToPass, dataH);
break;
case '1b':
command_len = 1;
var dataG = { uplinkType: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataG);
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);
var dataJ = { watchDogParams: { wdpC: wdpC, wdpUc: wdpUc } };
resultToPass = merge_obj(resultToPass, dataJ);
break;
case '2f':
command_len = 1;
var dataM = { targetTemperature: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataM);
break;
case '30':
command_len = 1;
var dataN = { manualTargetTemperatureUpdate: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataN);
break;
case '32':
command_len = 1;
var dataO = { heatingStatus: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataO);
break;
case '34':
command_len = 1;
var dataP = { displayRefreshPeriod: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataP);
break;
case '36':
command_len = 1;
var dataQ = { sendTargetTempDelay: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataQ);
break;
case '38':
command_len = 1;
var dataR = { automaticHeatingStatus: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataR);
break;
case '3a':
command_len = 1;
var dataS = { sensorMode: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataS);
break;
case '3d':
command_len = 1;
var dataT = { pirSensorStatus: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataT);
break;
case '3f':
command_len = 1;
var dataU = { pirSensorSensitivity: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataU);
break;
case '41':
command_len = 1;
var dataV = { currentTemperatureVisibility: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataV);
break;
case '43':
command_len = 1;
var dataW = { humidityVisibility: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataW);
break;
case '45':
command_len = 1;
var dataX = { lightIntensityVisibility: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataX);
break;
case '47':
command_len = 1;
var dataY = { pirInitPeriod: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataY);
break;
case '49':
command_len = 1;
var dataZ = { pirMeasurementPeriod: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataZ);
break;
case '4b':
command_len = 1;
var dataAA = { pirCheckPeriod: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataAA);
break;
case '4d':
command_len = 1;
var dataAB = { pirBlindPeriod: parseInt(commands[i + 1], 16) };
resultToPass = merge_obj(resultToPass, dataAB);
break;
case '4f':
command_len = 1;
var dataAC = { temperatureHysteresis: parseInt(commands[i + 1], 16) / 10 };
resultToPass = merge_obj(resultToPass, dataAC);
break;
case '51':
command_len = 2;
var dataAD = { targetTemperature: parseInt('0x' + commands[i + 1] + commands[i + 2], 16) / 10 };
resultToPass = merge_obj(resultToPass, dataAD);
break;
case '53':
command_len = 1;
var dataAE = { targetTemperatureStep: parseInt(commands[i + 1], 16) / 10 };
resultToPass = merge_obj(resultToPass, dataAE);
break;
case '54':
command_len = 2;
var dataAF = { manualTargetTemperatureUpdate: parseInt('0x' + commands[i + 1] + commands[i + 2], 16) / 10 };
resultToPass = merge_obj(resultToPass, dataAF);
break;
case 'a0':
command_len = 4;
var fuota_address = parseInt(commands[i + 1] + commands[i + 2] + commands[i + 3] + commands[i + 4], 16);
var fuota_address_raw = commands[i + 1] + commands[i + 2] + commands[i + 3] + commands[i + 4];
var dataAG = { fuota: { fuota_address: fuota_address, fuota_address_raw: fuota_address_raw } };
resultToPass = merge_obj(resultToPass, dataAG);
break;
default:
break;
}
commands.splice(i, command_len);
});
return resultToPass;
}
if (bytes[0] == 1 || bytes[0] == 129) {
data = merge_obj(data, handleKeepalive(bytes, data));
} else {
var keepaliveLength = 11;
var potentialKeepAlive = bytes.slice(-12);
if (potentialKeepAlive[0] == 129) keepaliveLength = 12;
data = merge_obj(data, handleResponse(bytes, data, keepaliveLength));
bytes = bytes.slice(-keepaliveLength);
data = merge_obj(data, handleKeepalive(bytes, data));
}
return { data: data };
}
function Decoder(payload, port) {
var decoded = decodeUplink({ bytes: payload, fPort: port }).data;
// Extract Gateway Information
try {
decoded.LORA_RSSI = (!!normalizedPayload.gateways && !!normalizedPayload.gateways[0] && normalizedPayload.gateways[0].rssi) || 0;
decoded.LORA_SNR = (!!normalizedPayload.gateways && !!normalizedPayload.gateways[0] && normalizedPayload.gateways[0].snr) || 0;
decoded.LORA_DATARATE = normalizedPayload.data_rate;
} catch (e) {
console.log(JSON.stringify(e));
}
// Array where we store the fields that are being sent to Datacake
var datacakeFields = [];
// Take each field from decoded and convert them to Datacake format
for (var key in decoded) {
if (decoded.hasOwnProperty(key)) {
datacakeFields.push({ field: key.toUpperCase(), value: decoded[key] });
}
}
// Forward data to Datacake
return datacakeFields;
}
Last updated