-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpayload_ltx.js
180 lines (170 loc) · 7.25 KB
/
payload_ltx.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
/* LTX Chirpstack Payload Decoder
* V1.12 (C) JoEmbedded.de
* LTX uses a flexible, compressed format, see LTX-Documentation
* using FLoat32 and Float16 on fPort 1-199 */
function decodeUplink(input) { // assume inpu is valid
if ((input.fPort > 0) && (input.fPort < 200)) {
// fPort 1-199: LTX Standard-Uplink, Port defines 'known types'
return { data: ltxDecode(input.bytes, input.fPort) }
} else {
return { data: { error: `LTX: fPort:${input.fPort} unknown` } };
}
}
// List of 'known types' with (opt. repeatet) units - LTX-Sensors
const deftypes = {
// 1-9 frei fuer 'Nix' oder Custom
10: ['°C'], // 10: All channels are Temperatures
11: ['%rH', '°C'], // 11: rH/T-Sensor
12: ['Bar', '°C'], // 12: Pressure/Level Sensor Bar
13: ['m', '°C'], // 13: Level Sensor m
14: ['m', 'dBm'], // 14: Distance(s) Sensor (Radar)
}
function ltxDecode(indata, port) {
const decoded = {};
// Byte 0: Hello-Flags for FLAGS and REASON
let ianz = indata.length;
if (ianz < 1) return { error: "LTX: Payload len < 1" };
// Transfer input bytes to BinaryArray
const view = new DataView(new ArrayBuffer(ianz));
let cursor = 0;
indata.forEach((b, i) => view.setUint8(i, b));
const flags = view.getUint8(cursor++);
ianz--;
decoded.flags = "";
if (flags & 128) decoded.flags += "(Reset)";
if (flags & 64) decoded.flags += "(Alarm)";
else if (flags & 32) decoded.flags += "(old Alarm)";
if (flags & 16) decoded.flags += "(Measure)";
const reason = flags & 15;
if (reason === 2) decoded.reason = "(Auto)";
else if (reason === 3) decoded.reason = "(Manual)";
else decoded.reason = `(Reason:${reason}?)`;
// Channels:0..89, HK:90..99(max 127) - Decoder V1.x does not care extended itoks >= 128!
let ichan = 0;
const typeunits = deftypes[port]
let dtidx = 0; // Index in deftypes
decoded.chans = [];
while (ianz-- > 0 && ichan < 127) {
const itok = view.getUint8(cursor++);
if (!(itok & 128)) {
let cmanz = itok & 63; // Channels: number of consecutive values
if (!cmanz) { // cmanz=0: New channel following
if (ianz < 1) return { error: "LTX: Format(1) invalid" };
ichan = view.getUint8(cursor++);
ianz--;
break;
}
const f16 = itok & 64;
while (cmanz-- > 0) {
const puob = { channel: ichan };
if (typeunits?.length > 0) { // Known Type with units
puob.unit = typeunits[dtidx % (typeunits.length)];
}
if (!f16) { // Following cmanz Float32
if (ianz < 4) return { error: "LTX: Format(2) invalid" };
puob.prec = "F32";
const u32 = view.getUint32(cursor, false); // Get Float32-Bits
if ((u32 >>> 23) == 0x1FF) puob.msg = getLTXError(u32 & 0x7FFFFF);
else puob.value = parseFloat(view.getFloat32(cursor, false).toPrecision(8));
cursor += 4;
ianz -= 4;
} else { // Following cmanz Float16
if (ianz < 2) return { error: "LTX: Format(3) invalid" };
puob.prec = "F16";
const u16 = view.getUint16(cursor, false); // Get Float16-Bits
if ((u16 >>> 10) == 0x3F) puob.msg = getLTXError(u16 & 1023);
else puob.value = parseFloat(view.getFloat16(cursor, false));
cursor += 2;
ianz -= 2;
}
decoded.chans.push(puob);
dtidx++;
ichan++;
}
} else { // HK, channels 90..
if (ichan < 90) ichan = 90;
let cbits = itok & 63; // HK: bitmask of consecutive values
if (!cbits) break; // > V1.0 - extended itoks (>=128) might follow
while (cbits) {
if (cbits & 1) {
if (ianz < 2) return { error: "LTX: Format(4) invalid Data" };
const puob = { channel: ichan, prec: "F16", unit: unitDescrHK(ichan) };
const u16 = view.getUint16(cursor, false); // Get Float16-Bits
if ((u16 >>> 10) == 0x3F) puob.msg = getLTXError(u16 & 1023);
else puob.value = parseFloat(view.getFloat16(cursor, false));
decoded.chans.push(puob);
cursor += 2;
ianz -= 2;
}
ichan++;
cbits >>>= 1;
}
}
}
return decoded;
}
// descHK - Unit and Description of HK-Channels
function unitDescrHK(ichan) {
switch (ichan) {
case 90: return "V (HK_Bat)";
case 91: return "°C (HK_intTemp)";
case 92: return "%rH (HK_intHum.)";
case 93: return "mAh (HK_usedEnergy)";
case 94: return "mBar (HK_Baro)";
default: return "HK_unknown";
}
}
// LTX Standard Error Codes, included in F16 or F32
function getLTXError(errno) {
switch (errno) {
case 1: return "NoValue";
case 2: return "NoReply";
case 3: return "OldValue";
// 4,5 unused
case 6: return "ErrorCRC";
case 7: return "DataError";
case 8: return "NoCachedValue";
default: return `Err${errno}`;
}
}
// Only a few JS engines support Float16
if (!DataView.prototype.getFloat16) {
DataView.prototype.getFloat16 = function (byteOffset, littleEndian) {
const uint16 = this.getUint16(byteOffset, littleEndian);
const sign = (uint16 & 0x8000) ? -1 : 1;
const exponent = (uint16 >>> 10) & 0x1F;
const fraction = uint16 & 0x03FF;
if (exponent === 0) return sign * Math.pow(2, -14) * (fraction / Math.pow(2, 10)); //Subnormal
else if (exponent === 0x1F) return fraction ? NaN : sign * Infinity; // NaN or Infinity
return (sign * Math.pow(2, exponent - 15) * (1 + fraction / Math.pow(2, 10))).toPrecision(6);
};
}
//----------------- TEST-CONSOLE, for Chirpstack remove the following parts: -----------------
// helper to pass test data to/from decoder
function testDecoder(hexString) {
const msg = hexString.replace(/\s/g, '');
testBytes = [];
for (let i = 0; i < msg.length; i += 2) {
testBytes.push(parseInt(msg.substring(i, i + 2), 16));
}
const theObject = decodeUplink({ fPort: 11, bytes: testBytes });
return JSON.stringify(theObject, null, 2);
}
function main() {
console.log("---TEST-DECODER---");
// Test-Payloads (in HEX), may contain spaces for better readability
// taken from LTX-INTENT Data Logger (Type 1500) with 1 or 3 SDI-12-Sensors
const testmsg = [
"13 01 41A4E148 9F 42A3 4DA8 5172 2B3F 63E8", // Manual, Value: 20.610001, HKs: 3.31836(V), 22.6250(°C), 43.5625(%), 0.0566101(mAh), 1012.00(mBar)
"12 42 4D24 4CE4 01 41948F5C 88 355B", // Auto 2 Values(F16): 20.5625, 19.5625, Value(F32): 18.570000 , HK: 0.334717(mAh)
"7242FC02FC0201FF800002884479", // Alarm, Auto, 3 Values: 'No Reply', HK: 4.47266(mAH)
]; 1
testmsg.forEach(e => {
console.log("----Test-Payload:-----");
console.log(e);
console.log("----JSON Result:-----");
console.log(testDecoder(e));
})
}
main();
//---