UNPKG

zigbee-herdsman

Version:

An open source ZigBee gateway solution with node.js.

198 lines (155 loc) 6.08 kB
/* v8 ignore start */ import {logger} from "../../../utils/logger"; const NS = "zh:zigate:frame"; enum ZiGateFrameChunkSize { UInt8 = 1, UInt16 = 2, UInt32 = 3, UInt64 = 4, } const hasStartByte = (startByte: number, frame: Buffer): boolean => { return frame.indexOf(startByte, 0) === 0; }; const hasStopByte = (stopByte: number, frame: Buffer): boolean => { return frame.indexOf(stopByte, frame.length - 1) === frame.length - 1; }; const combineBytes = (byte: number, idx: number, frame: number[]): [number, number] => { const nextByte = frame[idx + 1]; return [byte, nextByte]; }; // maybe any const removeDuplicate = (_: unknown, idx: number, frame: number[][]): boolean => { if (idx === 0) { return true; } const [first] = frame[idx - 1]; return first !== 0x2; }; const decodeBytes = (bytesPair: [number, number]): number => { return bytesPair[0] === 0x2 ? bytesPair[1] ^ 0x10 : bytesPair[0]; }; const readBytes = (bytes: Buffer): number => { return bytes.readUIntBE(0, bytes.length); }; const writeBytes = (bytes: Buffer, val: number): void => { bytes.writeUIntBE(val, 0, bytes.length); }; const xor = (checksum: number, byte: number): number => { return checksum ^ byte; }; const decodeFrame = (frame: Buffer): Buffer => { const arrFrame = Array.from(frame).map(combineBytes).filter(removeDuplicate).map(decodeBytes); return Buffer.from(arrFrame); }; const getFrameChunk = (frame: Buffer, pos: number, size: ZiGateFrameChunkSize): Buffer => { return frame.slice(pos, pos + size); }; export default class ZiGateFrame { static readonly START_BYTE = 0x1; static readonly STOP_BYTE = 0x3; msgCodeBytes: Buffer = Buffer.alloc(ZiGateFrameChunkSize.UInt16); msgLengthBytes: Buffer = Buffer.alloc(ZiGateFrameChunkSize.UInt16); checksumBytes: Buffer = Buffer.alloc(ZiGateFrameChunkSize.UInt8); msgPayloadBytes: Buffer = Buffer.alloc(0); rssiBytes: Buffer = Buffer.alloc(0); msgLengthOffset = 0; constructor(frame?: Buffer) { if (frame !== undefined) { const decodedFrame = decodeFrame(frame); // logger.debug(`decoded frame >>> %o`, decodedFrame, NS); // Due to ZiGate incoming frames with erroneous msg length this.msgLengthOffset = -1; if (!ZiGateFrame.isValid(frame)) { logger.error("Provided frame is not a valid ZiGate frame.", NS); return; } this.buildChunks(decodedFrame); try { if (this.readMsgCode() !== 0x8001) logger.debug(() => `${JSON.stringify(this)}`, NS); } catch (error) { // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress` logger.error((error as Error).stack!, NS); } if (this.readChecksum() !== this.calcChecksum()) { logger.error("Provided frame has an invalid checksum.", NS); return; } } } static isValid(frame: Buffer): boolean { return hasStartByte(ZiGateFrame.START_BYTE, frame) && hasStopByte(ZiGateFrame.STOP_BYTE, frame); } buildChunks(frame: Buffer): void { this.msgCodeBytes = getFrameChunk(frame, 1, this.msgCodeBytes.length); this.msgLengthBytes = getFrameChunk(frame, 3, this.msgLengthBytes.length); this.checksumBytes = getFrameChunk(frame, 5, this.checksumBytes.length); this.msgPayloadBytes = getFrameChunk(frame, 6, this.readMsgLength()); this.rssiBytes = getFrameChunk(frame, 6 + this.readMsgLength(), ZiGateFrameChunkSize.UInt8); } toBuffer(): Buffer { const length = 5 + this.readMsgLength(); const escapedData = this.escapeData( Buffer.concat([this.msgCodeBytes, this.msgLengthBytes, this.checksumBytes, this.msgPayloadBytes], length), ); return Buffer.concat([Uint8Array.from([ZiGateFrame.START_BYTE]), escapedData, Uint8Array.from([ZiGateFrame.STOP_BYTE])]); } escapeData(data: Buffer): Buffer { let encodedLength = 0; const encodedData = Buffer.alloc(data.length * 2); const FRAME_ESCAPE_XOR = 0x10; const FRAME_ESCAPE = 0x02; for (const b of data) { if (b <= FRAME_ESCAPE_XOR) { encodedData[encodedLength++] = FRAME_ESCAPE; encodedData[encodedLength++] = b ^ FRAME_ESCAPE_XOR; } else { encodedData[encodedLength++] = b; } } return encodedData.slice(0, encodedLength); } readMsgCode(): number { return readBytes(this.msgCodeBytes); } writeMsgCode(msgCode: number): ZiGateFrame { writeBytes(this.msgCodeBytes, msgCode); this.writeChecksum(); return this; } readMsgLength(): number { return readBytes(this.msgLengthBytes) + this.msgLengthOffset; } writeMsgLength(msgLength: number): ZiGateFrame { writeBytes(this.msgLengthBytes, msgLength); return this; } readChecksum(): number { return readBytes(this.checksumBytes); } writeMsgPayload(msgPayload: Buffer): ZiGateFrame { this.msgPayloadBytes = Buffer.from(msgPayload); this.writeMsgLength(msgPayload.length); this.writeChecksum(); return this; } readRSSI(): number { return readBytes(this.rssiBytes); } writeRSSI(rssi: number): ZiGateFrame { this.rssiBytes = Buffer.from([rssi]); this.writeChecksum(); return this; } calcChecksum(): number { let checksum = 0x00; checksum = this.msgCodeBytes.reduce(xor, checksum); checksum = this.msgLengthBytes.reduce(xor, checksum); checksum = this.rssiBytes.reduce(xor, checksum); checksum = this.msgPayloadBytes.reduce(xor, checksum); return checksum; } writeChecksum(): this { this.checksumBytes = Buffer.from([this.calcChecksum()]); return this; } }