UNPKG

zigbee-herdsman

Version:

An open source ZigBee gateway solution with node.js.

1,134 lines (991 loc) 37.3 kB
import {Buffalo} from "../../buffalo"; import {logger} from "../../utils/logger"; import {isNumberArray} from "../../utils/utils"; import {BuffaloZclDataType, DataType, StructuredIndicatorType} from "./definition/enums"; import type {BuffaloZclOptions, StructuredSelector, ZclArray} from "./definition/tstype"; import * as Utils from "./utils"; const NS = "zh:zcl:buffalo"; interface KeyValue { [s: string | number]: number | string; } const SEC_KEY_LENGTH = 16; const EXTENSION_FIELD_SETS_DATA_TYPE: {[key: number]: DataType[]} = { 6: [DataType.UINT8], 8: [DataType.UINT8], 258: [DataType.UINT8, DataType.UINT8], 768: [DataType.UINT16, DataType.UINT16, DataType.UINT16, DataType.UINT8, DataType.UINT8, DataType.UINT8, DataType.UINT16, DataType.UINT16], }; interface Struct { elmType: DataType; elmVal: unknown; } interface ZclTimeOfDay { /** [0-23] */ hours?: number; /** [0-59] */ minutes?: number; /** [0-59] */ seconds?: number; /** [0-99] */ hundredths?: number; } interface ZclDate { /** [1900-2155], converted to/from [0-255] => value+1900=year */ year?: number; /** [1-12] */ month?: number; /** [1-31] */ dayOfMonth?: number; /** [1-7] */ dayOfWeek?: number; } interface ZoneInfo { zoneID: number; zoneStatus: number; } interface ExtensionFieldSet { clstId: number; len: number; extField: unknown[]; } interface ThermoTransition { transitionTime: number; heatSetpoint?: number; coolSetpoint?: number; } interface Gpd { deviceID: number; options: number; extendedOptions: number; securityKey: Buffer; keyMic: number; outgoingCounter: number; applicationInfo: number; manufacturerID: number; modelID: number; numGpdCommands: number; gpdCommandIdList: Buffer; numServerClusters: number; numClientClusters: number; gpdServerClusters: Buffer; gpdClientClusters: Buffer; genericSwitchConfig: number; currentContactStatus: number; } interface GPDChannelRequest { nextChannel: number; nextNextChannel: number; } export interface GPDChannelConfiguration { commandID: number; operationalChannel: number; basic: boolean; } export interface GPDCommissioningReply { commandID: number; options: number; /** expected valid if corresponding `options` bits set */ panID?: number; /** expected valid if corresponding `options` bits set */ securityKey?: Buffer; /** expected valid if corresponding `options` bits set */ keyMic?: number; /** expected valid if corresponding `options` bits set */ frameCounter?: number; } interface GPDCustomReply { commandID: number; buffer: Buffer; } interface GPDAttributeReport { manufacturerCode: number; clusterID: number; attributes: KeyValue; } interface TuyaDataPointValue { dp: number; datatype: number; data: Buffer; } interface MiboxerZone { zoneNum: number; groupId: number; } export class BuffaloZcl extends Buffalo { private writeOctetStr(value: number[]): void { // TODO: this does not allow "non-value" 0xFF this.writeUInt8(value.length); this.writeBuffer(value, value.length); } private readOctetStr(): Buffer { const length = this.readUInt8(); return length < 0xff ? this.readBuffer(length) : Buffer.from([]); // non-value } private writeCharStr(value: string | number[]): void { // TODO: this does not allow "non-value" 0xFF if (typeof value === "string") { this.writeUInt8(Buffer.byteLength(value, "utf8")); this.writeUtf8String(value); } else { // XXX: value.length not written? this.writeBuffer(value, value.length); } } private readCharStr(): string { const length = this.readUInt8(); return length < 0xff ? this.readUtf8String(length) : ""; } private writeLongOctetStr(value: number[]): void { // TODO: this does not allow "non-value" 0xFF this.writeUInt16(value.length); this.writeBuffer(value, value.length); } private readLongOctetStr(): Buffer { const length = this.readUInt16(); return length < 0xffff ? this.readBuffer(length) : Buffer.from([]); // non-value } private writeLongCharStr(value: string): void { // TODO: this does not allow "non-value" 0xFF this.writeUInt16(Buffer.byteLength(value, "utf8")); this.writeUtf8String(value); } private readLongCharStr(): string { const length = this.readUInt16(); return length < 0xffff ? this.readUtf8String(length) : ""; // non-value } private writeArray(value: ZclArray): void { const elTypeNumeric = typeof value.elementType === "number" ? value.elementType : DataType[value.elementType]; this.writeUInt8(elTypeNumeric); // TODO: this does not allow writing "non-value" 0xFFFF this.writeUInt16(value.elements.length); for (const element of value.elements) { this.write(elTypeNumeric, element, {}); } } private readArray(): unknown[] { const values: unknown[] = []; const elementType = this.readUInt8(); const numberOfElements = this.readUInt16(); if (numberOfElements < 0xffff) { for (let i = 0; i < numberOfElements; i++) { const value = this.read(elementType, {}); values.push(value); } } return values; } private writeStruct(value: Struct[]): void { // XXX: from ZCL spec: "The zeroth element may not be written to." // how does this translates to writing here? // TODO: this does not allow writing "non-value" 0xFFFF this.writeUInt16(value.length); for (const v of value) { this.writeUInt8(v.elmType); this.write(v.elmType, v.elmVal, {}); } } private readStruct(): Struct[] { const values: Struct[] = []; const numberOfElements = this.readUInt16(); if (numberOfElements < 0xffff) { for (let i = 0; i < numberOfElements; i++) { const elementType = this.readUInt8(); const value = this.read(elementType, {}); values.push({elmType: elementType, elmVal: value}); } } return values; } private writeToD(value: ZclTimeOfDay): void { this.writeUInt8(value.hours ?? 0xff); this.writeUInt8(value.minutes ?? 0xff); this.writeUInt8(value.seconds ?? 0xff); this.writeUInt8(value.hundredths ?? 0xff); } private readToD(): ZclTimeOfDay { const hours = this.readUInt8(); const minutes = this.readUInt8(); const seconds = this.readUInt8(); const hundredths = this.readUInt8(); return { hours: hours < 0xff ? hours : undefined, minutes: minutes < 0xff ? minutes : undefined, seconds: seconds < 0xff ? seconds : undefined, hundredths: hundredths < 0xff ? hundredths : undefined, }; } private writeDate(value: ZclDate): void { this.writeUInt8(value.year !== undefined ? value.year - 1900 : 0xff); this.writeUInt8(value.month ?? 0xff); this.writeUInt8(value.dayOfMonth ?? 0xff); this.writeUInt8(value.dayOfWeek ?? 0xff); } private readDate(): ZclDate { const year = this.readUInt8(); const month = this.readUInt8(); const dayOfMonth = this.readUInt8(); const dayOfWeek = this.readUInt8(); return { year: year < 0xff ? year + 1900 : undefined, month: month < 0xff ? month : undefined, dayOfMonth: dayOfMonth < 0xff ? dayOfMonth : undefined, dayOfWeek: dayOfWeek < 0xff ? dayOfWeek : undefined, }; } //--- BuffaloZclDataType private writeListZoneInfo(values: ZoneInfo[]): void { for (const value of values) { this.writeUInt8(value.zoneID); this.writeUInt16(value.zoneStatus); } } private readListZoneInfo(length: number): ZoneInfo[] { const value: ZoneInfo[] = []; for (let i = 0; i < length; i++) { value.push({ zoneID: this.readUInt8(), zoneStatus: this.readUInt16(), }); } return value; } private writeExtensionFieldSets(values: {clstId: number; len: number; extField: number[]}[]): void { for (const value of values) { this.writeUInt16(value.clstId); this.writeUInt8(value.len); let index = 0; for (const entry of value.extField) { this.write(EXTENSION_FIELD_SETS_DATA_TYPE[value.clstId][index], entry, {}); index++; } } } private readExtensionFieldSets(): ExtensionFieldSet[] { const value: ExtensionFieldSet[] = []; // XXX: doesn't work if buffer has more unrelated fields after this one while (this.isMore()) { const clstId = this.readUInt16(); const len = this.readUInt8(); const end = this.getPosition() + len; let index = 0; const extField: unknown[] = []; while (this.getPosition() < end) { extField.push(this.read(EXTENSION_FIELD_SETS_DATA_TYPE[clstId][index], {})); index++; } value.push({extField, clstId, len}); } return value; } private writeListThermoTransitions(value: ThermoTransition[]): void { for (const entry of value) { this.writeUInt16(entry.transitionTime); if (entry.heatSetpoint != null) { this.writeUInt16(entry.heatSetpoint); } if (entry.coolSetpoint != null) { this.writeUInt16(entry.coolSetpoint); } } } private readListThermoTransitions(options: BuffaloZclOptions): ThermoTransition[] { if (options.payload == null || options.payload.mode == null || options.payload.numoftrans == null) { throw new Error("Cannot read LIST_THERMO_TRANSITIONS without required payload options specified"); } const heat = options.payload.mode & 1; const cool = options.payload.mode & 2; const result: ThermoTransition[] = []; for (let i = 0; i < options.payload.numoftrans; i++) { const entry: ThermoTransition = { transitionTime: this.readUInt16(), }; if (heat) { entry.heatSetpoint = this.readUInt16(); } if (cool) { entry.coolSetpoint = this.readUInt16(); } result.push(entry); } return result; } private writeGpdFrame(value: GPDCommissioningReply | GPDChannelConfiguration | GPDCustomReply): void { if (value.commandID === 0xf0) { // Commissioning Reply const v = value as GPDCommissioningReply; const panIDPresent = Boolean(v.options & 0x1); const gpdSecurityKeyPresent = Boolean(v.options & 0x2); const gpdKeyEncryption = Boolean((v.options >> 2) & 0x1); const securityLevel = (v.options >> 3) & 0x3; const hasGPDKeyMIC = gpdKeyEncryption && gpdSecurityKeyPresent; const hasFrameCounter = gpdSecurityKeyPresent && gpdKeyEncryption && (securityLevel === 0b10 || securityLevel === 0b11); this.writeUInt8(1 + (panIDPresent ? 2 : 0) + (gpdSecurityKeyPresent ? 16 : 0) + (hasGPDKeyMIC ? 4 : 0) + (hasFrameCounter ? 4 : 0)); // Length this.writeUInt8(v.options); if (panIDPresent) { // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress` this.writeUInt16(v.panID!); } if (gpdSecurityKeyPresent) { // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress` this.writeBuffer(v.securityKey!, 16); } if (hasGPDKeyMIC) { // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress` this.writeUInt32(v.keyMic!); } if (hasFrameCounter) { // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress` this.writeUInt32(v.frameCounter!); } } else if (value.commandID === 0xf3) { // Channel configuration const v = value as GPDChannelConfiguration; this.writeUInt8(1); this.writeUInt8((v.operationalChannel & 0xf) | ((v.basic ? 1 : 0) << 4)); } else if (value.commandID === 0xf4 || value.commandID === 0xf5 || (value.commandID >= 0xf7 && value.commandID <= 0xff)) { // Other commands sent to GPD const v = value as GPDCustomReply; this.writeUInt8(v.buffer.length); this.writeBuffer(v.buffer, v.buffer.length); } // 0xf1: Write Attributes // 0xf2: Read Attributes // 0xf6: ZCL Tunneling } private readGpdFrame(options: BuffaloZclOptions): Gpd | GPDChannelRequest | GPDAttributeReport | {raw: Buffer} | Record<string, never> { // ensure offset by options.payload.payloadSize (if any) at end of parsing to not cause issues with spec changes (until supported) const startPosition = this.position; if (options.payload?.commandID === 0xe0) { // Commisioning const frame = { deviceID: this.readUInt8(), options: this.readUInt8(), extendedOptions: 0, securityKey: Buffer.alloc(16), keyMic: 0, outgoingCounter: 0, applicationInfo: 0, manufacturerID: 0, modelID: 0, numGpdCommands: 0, gpdCommandIdList: Buffer.alloc(0), numServerClusters: 0, numClientClusters: 0, gpdServerClusters: Buffer.alloc(0), gpdClientClusters: Buffer.alloc(0), genericSwitchConfig: 0, currentContactStatus: 0, }; if (frame.options & 0x80) { frame.extendedOptions = this.readUInt8(); } if (frame.extendedOptions & 0x20) { frame.securityKey = this.readBuffer(16); } if (frame.extendedOptions & 0x40) { frame.keyMic = this.readUInt32(); } if (frame.extendedOptions & 0x80) { frame.outgoingCounter = this.readUInt32(); } if (frame.options & 0x04) { frame.applicationInfo = this.readUInt8(); } if (frame.applicationInfo & 0x01) { frame.manufacturerID = this.readUInt16(); } if (frame.applicationInfo & 0x02) { frame.modelID = this.readUInt16(); } if (frame.applicationInfo & 0x04) { frame.numGpdCommands = this.readUInt8(); frame.gpdCommandIdList = this.readBuffer(frame.numGpdCommands); } if (frame.applicationInfo & 0x08) { const len = this.readUInt8(); frame.numServerClusters = len & 0xf; frame.numClientClusters = (len >> 4) & 0xf; frame.gpdServerClusters = this.readBuffer(2 * frame.numServerClusters); frame.gpdClientClusters = this.readBuffer(2 * frame.numClientClusters); } if (frame.applicationInfo & 0x10) { const len = this.readUInt8(); if (len >= 1) { frame.genericSwitchConfig = this.readUInt8(); } if (len >= 2) { frame.currentContactStatus = this.readUInt8(); } } if (options.payload.payloadSize) { this.position = startPosition + options.payload.payloadSize; } return frame; } if (options.payload?.commandID === 0xe3) { // Channel Request const channelOpts = this.readUInt8(); /* v8 ignore start */ if (options.payload?.payloadSize) { this.position = startPosition + options.payload.payloadSize; } /* v8 ignore stop */ return { nextChannel: channelOpts & 0xf, nextNextChannel: channelOpts >> 4, }; } if (options.payload?.commandID === 0xa1) { // Manufacturer-specific Attribute Reporting if (options.payload.payloadSize === undefined) { throw new Error("Cannot read GPD_FRAME with commandID=0xA1 without payloadSize options specified"); } const start = this.position; const frame = { manufacturerCode: this.readUInt16(), clusterID: this.readUInt16(), attributes: {} as KeyValue, }; const cluster = Utils.getCluster(frame.clusterID, frame.manufacturerCode, {}); while (this.position - start < options.payload.payloadSize) { const attributeID = this.readUInt16(); const type = this.readUInt8(); let attribute: number | string = attributeID; try { attribute = cluster.getAttribute(attributeID).name; } catch { // this is spammy because of the many manufacturer-specific attributes not currently used logger.debug(`Unknown attribute ${attributeID} in cluster ${cluster.name}`, NS); } frame.attributes[attribute] = this.read(type, options); } this.position = startPosition + options.payload.payloadSize; return frame; } if (options.payload?.payloadSize && this.isMore()) { // might contain `gppNwkAddr`, `gppGpdLink` & `mic` from ZCL cluster, so limit by `payloadSize` return {raw: this.readBuffer(options.payload.payloadSize)}; } if (options.payload?.payloadSize) { this.position = startPosition + options.payload.payloadSize; } return {}; } private writeStructuredSelector(value: StructuredSelector): void { if (value != null) { const indexes = value.indexes || []; const indicatorType = value.indicatorType || StructuredIndicatorType.Whole; const indicator = indexes.length + indicatorType; this.writeUInt8(indicator); for (const index of indexes) { this.writeUInt16(index); } } } private readStructuredSelector(): StructuredSelector { /** [0-15] range */ const indicator = this.readUInt8(); if (indicator === 0) { // no indexes, whole attribute value is to be read return {indicatorType: StructuredIndicatorType.Whole}; } if (indicator < StructuredIndicatorType.WriteAdd) { const indexes: StructuredSelector["indexes"] = []; for (let i = 0; i < indicator; i++) { const index = this.readUInt16(); indexes.push(index); } return {indexes}; } throw new Error("Read structured selector was outside [0-15] range."); } private writeListTuyaDataPointValues(dpValues: TuyaDataPointValue[]): void { for (const dpValue of dpValues) { this.writeUInt8(dpValue.dp); this.writeUInt8(dpValue.datatype); const dataLen = dpValue.data.length; // UInt16BE this.writeUInt8((dataLen >> 8) & 0xff); this.writeUInt8(dataLen & 0xff); this.writeBuffer(dpValue.data, dataLen); } } private readListTuyaDataPointValues(): TuyaDataPointValue[] { const value: TuyaDataPointValue[] = []; // XXX: doesn't work if buffer has more unrelated fields after this one while (this.isMore()) { try { const dp = this.readUInt8(); const datatype = this.readUInt8(); const len_hi = this.readUInt8(); const len_lo = this.readUInt8(); const data = this.readBuffer(len_lo + (len_hi << 8)); value.push({dp, datatype, data}); } catch { break; } } return value; } private writeListMiboxerZones(values: MiboxerZone[]): void { this.writeUInt8(values.length); for (const value of values) { this.writeUInt16(value.groupId); this.writeUInt8(value.zoneNum); } } private readListMiboxerZones(): MiboxerZone[] { const value: MiboxerZone[] = []; const len = this.readUInt8(); for (let i = 0; i < len; i++) { const groupId = this.readUInt16(); const zoneNum = this.readUInt8(); value.push({groupId, zoneNum}); } return value; } private writeBigEndianUInt24(value: number): void { this.buffer.writeUIntBE(value, this.position, 3); this.position += 3; } private readBigEndianUInt24(): number { const value = this.buffer.readUIntBE(this.position, 3); this.position += 3; return value; } // NOTE: writeMiStruct is not supported. private readMiStruct(): Record<number, number | number[]> { const length = this.readUInt8(); const value: Record<number, number | number[]> = {}; if (length === 0xff) { return value; } for (let i = 0; i < length; i++) { const index = this.readUInt8(); const dataType = this.readUInt8(); value[index] = this.read(dataType, {}); const remaining = this.buffer.length - this.position; if (remaining <= 1) { if (remaining === 1) { // Some Xiaomi structs have a trailing byte, skip it. this.position += 1; } break; } } return value; } // biome-ignore lint/suspicious/noExplicitAny: API public write(type: DataType | BuffaloZclDataType, value: any, options: BuffaloZclOptions): void { switch (type) { case DataType.NO_DATA: case DataType.UNKNOWN: { return; // nothing to write } case DataType.DATA8: case DataType.BOOLEAN: case DataType.BITMAP8: case DataType.UINT8: case DataType.ENUM8: { this.writeUInt8(value); break; } case DataType.DATA16: case DataType.BITMAP16: case DataType.UINT16: case DataType.ENUM16: case DataType.CLUSTER_ID: case DataType.ATTR_ID: { this.writeUInt16(value); break; } case DataType.DATA24: case DataType.BITMAP24: case DataType.UINT24: { this.writeUInt24(value); break; } case DataType.DATA32: case DataType.BITMAP32: case DataType.UINT32: case DataType.UTC: case DataType.BAC_OID: { this.writeUInt32(value); break; } case DataType.DATA40: case DataType.BITMAP40: case DataType.UINT40: { this.writeUInt40(value); break; } case DataType.DATA48: case DataType.BITMAP48: case DataType.UINT48: { this.writeUInt48(value); break; } case DataType.DATA56: case DataType.BITMAP56: case DataType.UINT56: { this.writeUInt56(value); break; } case DataType.DATA64: case DataType.BITMAP64: case DataType.UINT64: { this.writeUInt64(value); break; } case DataType.INT8: { this.writeInt8(value); break; } case DataType.INT16: { this.writeInt16(value); break; } case DataType.INT24: { this.writeInt24(value); break; } case DataType.INT32: { this.writeInt32(value); break; } case DataType.INT40: { this.writeInt40(value); break; } case DataType.INT48: { this.writeInt48(value); break; } case DataType.INT56: { this.writeInt56(value); break; } case DataType.INT64: { this.writeInt64(value); break; } // case DataType.SEMI_PREC: { // // https://tc39.es/proposal-float16array/ // // not currently used // this.writeSemiFloatLE(value); // break; // } case DataType.SINGLE_PREC: { this.writeFloatLE(value); break; } case DataType.DOUBLE_PREC: { this.writeDoubleLE(value); break; } case DataType.OCTET_STR: { this.writeOctetStr(value); break; } case DataType.CHAR_STR: { this.writeCharStr(value); break; } case DataType.LONG_OCTET_STR: { this.writeLongOctetStr(value); break; } case DataType.LONG_CHAR_STR: { this.writeLongCharStr(value); break; } case DataType.ARRAY: case DataType.SET: case DataType.BAG: { this.writeArray(value); break; } case DataType.STRUCT: { this.writeStruct(value); break; } case DataType.TOD: { this.writeToD(value); break; } case DataType.DATE: { this.writeDate(value); break; } case DataType.IEEE_ADDR: { this.writeIeeeAddr(value); break; } case DataType.SEC_KEY: { this.writeBuffer(value, SEC_KEY_LENGTH); break; } case BuffaloZclDataType.USE_DATA_TYPE: { if (options.dataType == null) { if (Buffer.isBuffer(value) || isNumberArray(value)) { this.writeBuffer(value, value.length); break; } throw new Error("Cannot write USE_DATA_TYPE without dataType option specified"); } this.write(options.dataType, value, options); break; } case BuffaloZclDataType.LIST_UINT8: { this.writeListUInt8(value); break; } case BuffaloZclDataType.LIST_UINT16: { this.writeListUInt16(value); break; } case BuffaloZclDataType.LIST_UINT24: { this.writeListUInt24(value); break; } case BuffaloZclDataType.LIST_UINT32: { this.writeListUInt32(value); break; } case BuffaloZclDataType.LIST_ZONEINFO: { this.writeListZoneInfo(value); break; } case BuffaloZclDataType.EXTENSION_FIELD_SETS: { this.writeExtensionFieldSets(value); break; } case BuffaloZclDataType.LIST_THERMO_TRANSITIONS: { this.writeListThermoTransitions(value); break; } case BuffaloZclDataType.BUFFER: { // XXX: inconsistent with read that allows partial with options.length, here always "whole" this.writeBuffer(value, value.length); break; } case BuffaloZclDataType.GPD_FRAME: { this.writeGpdFrame(value); break; } case BuffaloZclDataType.STRUCTURED_SELECTOR: { this.writeStructuredSelector(value); break; } case BuffaloZclDataType.LIST_TUYA_DATAPOINT_VALUES: { this.writeListTuyaDataPointValues(value); break; } case BuffaloZclDataType.LIST_MIBOXER_ZONES: { this.writeListMiboxerZones(value); break; } case BuffaloZclDataType.BIG_ENDIAN_UINT24: { this.writeBigEndianUInt24(value); break; } default: { // In case the type is undefined, write it as a buffer to easily allow for custom types // e.g. for https://github.com/Koenkk/zigbee-herdsman/issues/127 if (Buffer.isBuffer(value) || isNumberArray(value)) { this.writeBuffer(value, value.length); break; } throw new Error(`Write for '${type}' not available`); } } } // biome-ignore lint/suspicious/noExplicitAny: API public read(type: DataType | BuffaloZclDataType, options: BuffaloZclOptions): any { switch (type) { case DataType.NO_DATA: case DataType.UNKNOWN: { return; // nothing to write } case DataType.DATA8: case DataType.BOOLEAN: case DataType.BITMAP8: case DataType.UINT8: case DataType.ENUM8: { return this.readUInt8(); } case DataType.DATA16: case DataType.BITMAP16: case DataType.UINT16: case DataType.ENUM16: case DataType.CLUSTER_ID: case DataType.ATTR_ID: { return this.readUInt16(); } case DataType.DATA24: case DataType.BITMAP24: case DataType.UINT24: { return this.readUInt24(); } case DataType.DATA32: case DataType.BITMAP32: case DataType.UINT32: case DataType.UTC: case DataType.BAC_OID: { return this.readUInt32(); } case DataType.DATA40: case DataType.BITMAP40: case DataType.UINT40: { return this.readUInt40(); } case DataType.DATA48: case DataType.BITMAP48: case DataType.UINT48: { return this.readUInt48(); } case DataType.DATA56: case DataType.BITMAP56: case DataType.UINT56: { return this.readUInt56(); } case DataType.DATA64: case DataType.BITMAP64: case DataType.UINT64: { return this.readUInt64(); } case DataType.INT8: { return this.readInt8(); } case DataType.INT16: { return this.readInt16(); } case DataType.INT24: { return this.readInt24(); } case DataType.INT32: { return this.readInt32(); } case DataType.INT40: { return this.readInt40(); } case DataType.INT48: { return this.readInt48(); } case DataType.INT56: { return this.readInt56(); } case DataType.INT64: { return this.readInt64(); } // case DataType.SEMI_PREC: { // // https://tc39.es/proposal-float16array/ // // not currently used // return this.readSemiFloatLE(); // } case DataType.SINGLE_PREC: { return this.readFloatLE(); } case DataType.DOUBLE_PREC: { return this.readDoubleLE(); } case DataType.OCTET_STR: { return this.readOctetStr(); } case DataType.CHAR_STR: { return this.readCharStr(); } case DataType.LONG_OCTET_STR: { return this.readLongOctetStr(); } case DataType.LONG_CHAR_STR: { return this.readLongCharStr(); } case DataType.ARRAY: case DataType.SET: case DataType.BAG: { return this.readArray(); } case DataType.STRUCT: { return this.readStruct(); } case DataType.TOD: { return this.readToD(); } case DataType.DATE: { return this.readDate(); } case DataType.IEEE_ADDR: { return this.readIeeeAddr(); } case DataType.SEC_KEY: { return this.readBuffer(SEC_KEY_LENGTH); } case BuffaloZclDataType.USE_DATA_TYPE: { if (options.dataType == null) { return this.readBuffer(options.length ?? this.buffer.length); } return this.read(options.dataType, options); } case BuffaloZclDataType.LIST_UINT8: { if (options.length == null) { throw new Error("Cannot read LIST_UINT8 without length option specified"); } return this.readListUInt8(options.length); } case BuffaloZclDataType.LIST_UINT16: { if (options.length == null) { throw new Error("Cannot read LIST_UINT16 without length option specified"); } return this.readListUInt16(options.length); } case BuffaloZclDataType.LIST_UINT24: { if (options.length == null) { throw new Error("Cannot read LIST_UINT24 without length option specified"); } return this.readListUInt24(options.length); } case BuffaloZclDataType.LIST_UINT32: { if (options.length == null) { throw new Error("Cannot read LIST_UINT32 without length option specified"); } return this.readListUInt32(options.length); } case BuffaloZclDataType.LIST_ZONEINFO: { if (options.length == null) { throw new Error("Cannot read LIST_ZONEINFO without length option specified"); } return this.readListZoneInfo(options.length); } case BuffaloZclDataType.EXTENSION_FIELD_SETS: { return this.readExtensionFieldSets(); } case BuffaloZclDataType.LIST_THERMO_TRANSITIONS: { return this.readListThermoTransitions(options); } case BuffaloZclDataType.BUFFER: { // if length option not specified, read the whole buffer return this.readBuffer(options.length ?? this.buffer.length); } case BuffaloZclDataType.GPD_FRAME: { return this.readGpdFrame(options); } case BuffaloZclDataType.STRUCTURED_SELECTOR: { return this.readStructuredSelector(); } case BuffaloZclDataType.LIST_TUYA_DATAPOINT_VALUES: { return this.readListTuyaDataPointValues(); } case BuffaloZclDataType.LIST_MIBOXER_ZONES: { return this.readListMiboxerZones(); } case BuffaloZclDataType.BIG_ENDIAN_UINT24: { return this.readBigEndianUInt24(); } case BuffaloZclDataType.MI_STRUCT: { return this.readMiStruct(); } } throw new Error(`Read for '${type}' not available`); } }