chenyc
2026-04-21 8632fbd73fdb15f22fae9cd36b9ed3e0635360f1
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
#!/usr/bin/env node
 
const fs = require('fs');
const path = require('path');
const assert = require('assert');
const { JhmDecoder, additiveChecksum } = require('./decoder');
const { MqttService } = require('./mqtt-service');
 
const config = JSON.parse(fs.readFileSync(path.join(__dirname, 'config.json'), 'utf8'));
const decoder = new JhmDecoder({
  alModelPath: config.protocol.alModelPath,
  maxBufferBytes: config.tcp.maxBufferBytes,
});
const mqttService = new MqttService(config.mqtt, console);
const device = config.devices[0] || { deviceId: 'JHM-TEST', ip: '127.0.0.1', name: 'test-device' };
 
function buildFrame(command, payload) {
  const cy = (command + payload[0] + payload[1] + payload[2] + payload[3]) & 0xFF;
  return Buffer.from([0xEE, 0x55, command, ...payload, cy]);
}
 
function buildBloodPressureFrame({ systolic, diastolic, pulse, timeBytes }) {
  const frameWithoutChecksum = Buffer.from([
    0xAA,
    0x55,
    0x0E,
    0xBA,
    (systolic >> 8) & 0xFF,
    systolic & 0xFF,
    diastolic & 0xFF,
    pulse & 0xFF,
    ...timeBytes,
  ]);
 
  return Buffer.concat([
    frameWithoutChecksum,
    Buffer.from([additiveChecksum(frameWithoutChecksum)]),
  ]);
}
 
function payloadToHex(payload) {
  return payload.map((value) => `0x${value.toString(16).toUpperCase().padStart(2, '0')}`).join(' ');
}
 
const cases = [
  { command: 0x01, payload: [0x00, 0x00, 0x01, 0x77], expectedPublish: true, expectedMetric: { F: 37.5 }, note: 'temperature = 375 / 10' },
  { command: 0x02, payload: [0x00, 0x00, 0x01, 0x77], expectedPublish: false, reason: 'identifier-missing', note: 'set temperature not published' },
  { command: 0x03, payload: [0x00, 0x00, 0x04, 0xD2], expectedPublish: true, expectedMetric: { A: 1.234 }, note: 'ultrafiltration total = 1234mL => 1.234L' },
  { command: 0x04, payload: [0x00, 0x01, 0x00, 0x1E], expectedPublish: true, expectedMetric: { sysj: 90 }, note: 'remaining minutes = 1*60 + 30' },
  { command: 0x05, payload: [0x00, 0x00, 0x01, 0xF4], expectedPublish: true, expectedMetric: { C: 0.5 }, note: 'uf rate = 500mL => 0.5L' },
  { command: 0x06, payload: [0x00, 0x00, 0x00, 0xFA], expectedPublish: true, expectedMetric: { B: 0.25 }, note: 'uf volume = 250mL => 0.25L' },
  { command: 0x07, payload: [0x00, 0x00, 0x02, 0x58], expectedPublish: true, expectedMetric: { L: 600 }, note: 'dialysate flow = 600' },
  { command: 0x08, payload: [0x00, 0x00, 0x01, 0x40], expectedPublish: true, expectedMetric: { D: 320 }, note: 'effective blood flow = 320' },
  { command: 0x09, payload: [0x01, 0x00, 0x00, 0x64], expectedPublish: true, expectedMetric: { H: -100 }, note: 'venous pressure = -100' },
  { command: 0x0A, payload: [0x00, 0x00, 0x00, 0x78], expectedPublish: true, expectedMetric: { o: 120 }, note: 'arterial pressure = 120' },
  { command: 0x0B, payload: [0x01, 0x00, 0x00, 0x50], expectedPublish: true, expectedMetric: { J: -80 }, note: 'transmembrane pressure = -80' },
  { command: 0x0C, payload: [0x00, 0x00, 0x08, 0x00], expectedPublish: true, expectedMetric: { U: 2048 }, note: 'accumulated blood flow = 2048' },
  { command: 0x0D, payload: [0x00, 0x00, 0x00, 0x7B], expectedPublish: true, expectedMetric: { G: 12.3 }, note: 'conductivity = 123 / 10' },
  { command: 0x0E, payload: [0x00, 0x00, 0x00, 0x8C], expectedPublish: true, expectedMetric: { Na: 140 }, note: 'sodium concentration = 140' },
  { command: 0x0F, payload: [0x00, 0x00, 0x00, 0x19], expectedPublish: true, expectedMetric: { HCO3: 25 }, note: 'HCO3 concentration = 25' },
];
 
const bloodPressureCase = {
  frame: buildBloodPressureFrame({
    systolic: 120,
    diastolic: 80,
    pulse: 89,
    timeBytes: [0x1A, 0x04, 0x0F, 0x09, 0x1E],
  }),
  expectedMetric: {
    N: 120,
    O: 80,
    P: 89,
    M: '2026-04-15 09:30',
  },
};
 
function main() {
  const topic = mqttService.buildTopic(device);
 
  console.log(`verify device: ${device.deviceId} (${device.ip})`);
  console.log(`verify topic: ${topic}`);
  console.log('');
 
  for (const item of cases) {
    const result = decoder.push(buildFrame(item.command, item.payload))[0];
 
    assert.ok(result, `command 0x${item.command.toString(16).toUpperCase()} returned no result`);
    assert.strictEqual(result.publish, item.expectedPublish, `command 0x${item.command.toString(16).toUpperCase()} publish state mismatch`);
 
    if (item.expectedPublish) {
      assert.deepStrictEqual(result.metric, item.expectedMetric, `command 0x${item.command.toString(16).toUpperCase()} metric mismatch`);
      console.log(`[publish] command=0x${item.command.toString(16).toUpperCase().padStart(2, '0')} payload=${payloadToHex(item.payload)} topic=${topic} mqtt=${JSON.stringify(result.metric)} note=${item.note}`);
    } else {
      assert.strictEqual(result.reason, item.reason, `command 0x${item.command.toString(16).toUpperCase()} skip reason mismatch`);
      console.log(`[skip] command=0x${item.command.toString(16).toUpperCase().padStart(2, '0')} payload=${payloadToHex(item.payload)} reason=${result.reason} note=${item.note}`);
    }
  }
 
  const bloodPressureResult = decoder.push(bloodPressureCase.frame)[0];
  assert.ok(bloodPressureResult, 'blood pressure frame returned no result');
  assert.strictEqual(bloodPressureResult.publish, true, 'blood pressure frame should publish');
  assert.deepStrictEqual(bloodPressureResult.metric, bloodPressureCase.expectedMetric, 'blood pressure metric mismatch');
  console.log(`[publish] command=0xBA raw=${bloodPressureResult.rawHex} topic=${topic} mqtt=${JSON.stringify(bloodPressureResult.metric)} note=blood pressure frame`);
 
  console.log('');
  console.log('verify-all-commands passed');
}
 
main();