zigbee-herdsman
Version:
An open source ZigBee gateway solution with node.js.
153 lines (140 loc) • 6.95 kB
text/typescript
import * as fs from "node:fs";
import * as path from "node:path";
import * as ZStackStructs from "../adapter/z-stack/structs";
import * as ZStackUtils from "../adapter/z-stack/utils";
import type * as Models from "../models";
/**
* Converts internal backup format to unified backup storage format as described by
* [zigpy/open-coordinator-backup](https://github.com/zigpy/open-coordinator-backup).
*
* @param backup Backup to create unified backup format from.
*/
export const toUnifiedBackup = async (backup: Models.Backup): Promise<Models.UnifiedBackupStorage> => {
const packageInfo = JSON.parse(fs.readFileSync(path.join(__dirname, "../../", "package.json")).toString());
return {
metadata: {
format: "zigpy/open-coordinator-backup",
version: 1,
source: `${packageInfo.name}@${packageInfo.version}`,
internal: {
date: new Date().toISOString(),
...(backup.znp ? {znpVersion: backup.znp?.version ?? undefined} : /* v8 ignore next */ undefined),
...(backup.ezsp ? {ezspVersion: backup.ezsp?.version ?? undefined} : /* v8 ignore next */ undefined),
},
},
stack_specific: {
/* v8 ignore next */
...(backup.znp ? {zstack: {tclk_seed: backup.znp?.trustCenterLinkKeySeed?.toString("hex") || undefined}} : undefined),
...(backup.ezsp ? {ezsp: {hashed_tclk: backup.ezsp?.hashed_tclk?.toString("hex") || undefined}} : /* v8 ignore next */ undefined),
},
coordinator_ieee: backup.coordinatorIeeeAddress.toString("hex"),
pan_id: backup.networkOptions.panId.toString(16),
extended_pan_id: backup.networkOptions.extendedPanId.toString("hex"),
nwk_update_id: backup.networkUpdateId || 0,
security_level: backup.securityLevel,
channel: backup.logicalChannel,
channel_mask: backup.networkOptions.channelList,
network_key: {
key: backup.networkOptions.networkKey.toString("hex"),
sequence_number: backup.networkKeyInfo.sequenceNumber,
frame_counter: backup.networkKeyInfo.frameCounter,
},
devices: backup.devices.map((device) => {
return {
nwk_address: device.networkAddress !== null ? device.networkAddress.toString(16) : /* v8 ignore next */ null,
ieee_address: device.ieeeAddress.toString("hex"),
is_child: device.isDirectChild,
link_key: device.linkKey
? {key: device.linkKey.key.toString("hex"), rx_counter: device.linkKey.rxCounter, tx_counter: device.linkKey.txCounter}
: undefined,
};
}),
};
};
/**
* Converts unified backup storage format to internal backup format.
*
* @param backup Unified format to convert to internal backup format.
*/
export const fromUnifiedBackup = (backup: Models.UnifiedBackupStorage): Models.Backup => {
const tclkSeedString = backup.stack_specific?.zstack?.tclk_seed || undefined;
return {
networkOptions: {
panId: Number.parseInt(backup.pan_id, 16),
extendedPanId: Buffer.from(backup.extended_pan_id, "hex"),
channelList: backup.channel_mask,
networkKey: Buffer.from(backup.network_key.key, "hex"),
networkKeyDistribute: false,
},
logicalChannel: backup.channel,
networkKeyInfo: {
sequenceNumber: backup.network_key.sequence_number,
frameCounter: backup.network_key.frame_counter,
},
coordinatorIeeeAddress: Buffer.from(backup.coordinator_ieee, "hex"),
securityLevel: backup.security_level,
networkUpdateId: backup.nwk_update_id,
devices: backup.devices.map((device) => ({
networkAddress: device.nwk_address ? Number.parseInt(device.nwk_address, 16) : null,
ieeeAddress: Buffer.from(device.ieee_address, "hex"),
isDirectChild: typeof device.is_child === "boolean" ? device.is_child : true,
linkKey: device.link_key
? {key: Buffer.from(device.link_key.key, "hex"), rxCounter: device.link_key.rx_counter, txCounter: device.link_key.tx_counter}
: undefined,
})),
znp: {
version: backup.metadata.internal?.znpVersion || undefined,
trustCenterLinkKeySeed: tclkSeedString ? Buffer.from(tclkSeedString, "hex") : undefined,
},
ezsp: {
version: backup.metadata.internal?.ezspVersion || undefined,
hashed_tclk: backup.stack_specific?.ezsp?.hashed_tclk ? Buffer.from(backup.stack_specific.ezsp.hashed_tclk, "hex") : undefined,
},
};
};
/**
* Converts legacy Zigbee2MQTT format to internal backup format.
*
* @param backup Legacy format to convert.
*/
export const fromLegacyBackup = (backup: Models.LegacyBackupStorage): Models.Backup => {
if (!backup.data.ZCD_NV_NIB) {
throw new Error("Backup corrupted - missing NIB");
}
if (!backup.data.ZCD_NV_NWK_ACTIVE_KEY_INFO) {
throw new Error("Backup corrupted - missing active key info");
}
if (!backup.data.ZCD_NV_PRECFGKEY_ENABLE) {
throw new Error("Backup corrupted - missing pre-configured key enable attribute");
}
if (!backup.data.ZCD_NV_EX_NWK_SEC_MATERIAL_TABLE && !backup.data.ZCD_NV_LEGACY_NWK_SEC_MATERIAL_TABLE_START) {
throw new Error("Backup corrupted - missing network security material table");
}
if (!backup.data.ZCD_NV_EXTADDR) {
throw new Error("Backup corrupted - missing adapter IEEE address NV entry");
}
const ieeeAddress = Buffer.from(backup.data.ZCD_NV_EXTADDR.value).reverse();
const nib = ZStackStructs.nib(Buffer.from(backup.data.ZCD_NV_NIB.value));
const activeKeyInfo = ZStackStructs.nwkKeyDescriptor(Buffer.from(backup.data.ZCD_NV_NWK_ACTIVE_KEY_INFO.value));
const preconfiguredKeyEnabled = backup.data.ZCD_NV_PRECFGKEY_ENABLE.value[0] !== 0x00;
const nwkSecMaterialSource = backup.data.ZCD_NV_EX_NWK_SEC_MATERIAL_TABLE || backup.data.ZCD_NV_LEGACY_NWK_SEC_MATERIAL_TABLE_START;
const nwkSecMaterialEntry = ZStackStructs.nwkSecMaterialDescriptorEntry(Buffer.from(nwkSecMaterialSource.value));
return {
networkOptions: {
panId: nib.nwkPanId,
extendedPanId: nib.extendedPANID,
channelList: ZStackUtils.unpackChannelList(nib.channelList),
networkKey: activeKeyInfo.key,
networkKeyDistribute: preconfiguredKeyEnabled,
},
logicalChannel: nib.nwkLogicalChannel,
networkKeyInfo: {
sequenceNumber: activeKeyInfo.keySeqNum,
frameCounter: nwkSecMaterialEntry.FrameCounter,
},
coordinatorIeeeAddress: ieeeAddress,
securityLevel: nib.SecurityLevel,
networkUpdateId: nib.nwkUpdateId,
devices: [],
};
};