Skip to content

Commit

Permalink
Merge branch 'zwave-js:master' into import-config-ShellyQubino-00004
Browse files Browse the repository at this point in the history
  • Loading branch information
QubinoHelp authored Feb 23, 2024
2 parents 256f65c + 4229045 commit 6f8bb38
Show file tree
Hide file tree
Showing 30 changed files with 424 additions and 117 deletions.
8 changes: 8 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,14 @@
<!--
Add placeholder for next release with `wip` snippet
-->
## 12.4.4 (2024-02-10)
### Bugfixes
* NVM backups can now be restored onto 800 series controllers (#6670)

### Config file changes
* Use Color Switch V2 for Inovelli LZW42 (#6654)
* Correct Zooz ZEN1x timer config params (#6648)

## 12.4.3 (2024-01-25)
### Bugfixes
* Reduce idle CPU load (#6640)
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@zwave-js/repo",
"version": "12.4.3",
"version": "12.4.4",
"private": true,
"description": "Z-Wave driver written entirely in JavaScript/TypeScript",
"keywords": [],
Expand Down
2 changes: 1 addition & 1 deletion packages/cc/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@zwave-js/cc",
"version": "12.4.3",
"version": "12.4.4",
"description": "zwave-js: Command Classes",
"keywords": [],
"publishConfig": {
Expand Down
2 changes: 1 addition & 1 deletion packages/config/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@zwave-js/config",
"version": "12.4.3",
"version": "12.4.4",
"description": "zwave-js: configuration files",
"publishConfig": {
"access": "public"
Expand Down
2 changes: 1 addition & 1 deletion packages/flash/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@zwave-js/flash",
"version": "12.4.3",
"version": "12.4.4",
"description": "zwave-js: firmware flash utility",
"keywords": [],
"publishConfig": {
Expand Down
2 changes: 1 addition & 1 deletion packages/host/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@zwave-js/host",
"version": "12.4.3",
"version": "12.4.4",
"description": "zwave-js: Host abstractions",
"keywords": [],
"publishConfig": {
Expand Down
2 changes: 1 addition & 1 deletion packages/nvmedit/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@zwave-js/nvmedit",
"version": "12.4.3",
"version": "12.4.4",
"description": "zwave-js: library to edit NVM backups",
"keywords": [],
"publishConfig": {
Expand Down
38 changes: 37 additions & 1 deletion packages/nvmedit/src/convert.test.ts.md

Large diffs are not rendered by default.

Binary file modified packages/nvmedit/src/convert.test.ts.snap
Binary file not shown.
104 changes: 82 additions & 22 deletions packages/nvmedit/src/convert.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import {
type CommandClasses,
ControllerCapabilityFlags,
NodeIDType,
type NodeProtocolInfo,
NodeType,
RFRegion,
Expand All @@ -9,7 +10,7 @@ import {
isZWaveError,
stripUndefined,
} from "@zwave-js/core/safe";
import { cloneDeep, pick } from "@zwave-js/shared/safe";
import { cloneDeep, num2hex, pick } from "@zwave-js/shared/safe";
import { isObject } from "alcalzone-shared/typeguards";
import semver from "semver";
import { MAX_PROTOCOL_FILE_FORMAT, SUC_MAX_UPDATES } from "./consts";
Expand All @@ -23,6 +24,8 @@ import {
ApplicationTypeFile,
ApplicationTypeFileID,
ApplicationVersionFile,
ApplicationVersionFile800,
ApplicationVersionFile800ID,
ApplicationVersionFileID,
ControllerInfoFile,
ControllerInfoFileID,
Expand Down Expand Up @@ -65,6 +68,10 @@ import {
nodeIdToRouteCacheFileIDV1,
sucUpdateIndexToSUCUpdateEntriesFileIDV5,
} from "./files";
import {
ApplicationNameFile,
ApplicationNameFileID,
} from "./files/ApplicationNameFile";
import {
type NVM3Objects,
type NVMMeta,
Expand Down Expand Up @@ -122,6 +129,7 @@ export interface NVMJSONController {
};

applicationData?: string | null;
applicationName?: string | null;
}

export interface NVMJSONControllerRFConfig {
Expand All @@ -130,6 +138,7 @@ export interface NVMJSONControllerRFConfig {
measured0dBm: number;
enablePTI: number | null;
maxTXPower: number | null;
nodeIdType: NodeIDType | null;
}

export interface NVMJSONNodeWithInfo
Expand Down Expand Up @@ -202,8 +211,7 @@ function createEmptyPhysicalNode(): NVMJSONNodeWithInfo {

/** Converts a compressed set of NVM objects to a JSON representation */
export function nvmObjectsToJSON(
applicationObjects: ReadonlyMap<number, NVM3Object>,
protocolObjects: ReadonlyMap<number, NVM3Object>,
objects: ReadonlyMap<number, NVM3Object>,
): NVMJSON {
const nodes = new Map<number, NVMJSONNode>();
const getNode = (id: number): NVMJSONNode => {
Expand All @@ -215,12 +223,9 @@ export function nvmObjectsToJSON(
id: number | ((id: number) => boolean),
): NVM3Object | undefined => {
if (typeof id === "number") {
return protocolObjects.get(id) ?? applicationObjects.get(id);
return objects.get(id);
} else {
for (const [key, obj] of protocolObjects) {
if (id(key)) return obj;
}
for (const [key, obj] of applicationObjects) {
for (const [key, obj] of objects) {
if (id(key)) return obj;
}
}
Expand All @@ -232,7 +237,9 @@ export function nvmObjectsToJSON(
const ret = getObject(id);
if (ret) return ret;
throw new ZWaveError(
`Object${typeof id === "number" ? ` ${id}` : ""} not found!`,
`Object${
typeof id === "number" ? ` ${num2hex(id)} (${id})` : ""
} not found!`,
ZWaveErrorCodes.NVM_ObjectNotFound,
);
};
Expand Down Expand Up @@ -414,10 +421,22 @@ export function nvmObjectsToJSON(

// === Application NVM files ===

const applicationVersionFile = getFileOrThrow<ApplicationVersionFile>(
const applicationVersionFile700 = getFile<ApplicationVersionFile>(
ApplicationVersionFileID,
"7.0.0", // We don't know the version here yet
);
const applicationVersionFile800 = getFile<ApplicationVersionFile800>(
ApplicationVersionFile800ID,
"7.0.0", // We don't know the version here yet
);
const applicationVersionFile = applicationVersionFile700
?? applicationVersionFile800;
if (!applicationVersionFile) {
throw new ZWaveError(
"ApplicationVersionFile not found!",
ZWaveErrorCodes.NVM_ObjectNotFound,
);
}
const applicationVersion =
`${applicationVersionFile.major}.${applicationVersionFile.minor}.${applicationVersionFile.patch}`;

Expand All @@ -437,6 +456,11 @@ export function nvmObjectsToJSON(
ApplicationTypeFileID,
applicationVersion,
);
const applicationNameFile = getFile<ApplicationNameFile>(
ApplicationNameFileID,
applicationVersion,
);

const preferredRepeaters = getFile<ProtocolPreferredRepeatersFile>(
ProtocolPreferredRepeatersFileID,
applicationVersion,
Expand All @@ -459,10 +483,8 @@ export function nvmObjectsToJSON(
"dcdcConfig",
] as const;
const controller: NVMJSONController = {
protocolVersion:
`${protocolVersionFile.major}.${protocolVersionFile.minor}.${protocolVersionFile.patch}`,
applicationVersion:
`${applicationVersionFile.major}.${applicationVersionFile.minor}.${applicationVersionFile.patch}`,
protocolVersion,
applicationVersion,
homeId: `0x${controllerInfoFile.homeId.toString("hex")}`,
...pick(controllerInfoFile, controllerProps),
...pick(applicationTypeFile, [
Expand All @@ -485,11 +507,13 @@ export function nvmObjectsToJSON(
measured0dBm: rfConfigFile.measured0dBm,
enablePTI: rfConfigFile.enablePTI ?? null,
maxTXPower: rfConfigFile.maxTXPower ?? null,
nodeIdType: rfConfigFile.nodeIdType ?? null,
},
}
: {}),
sucUpdateEntries,
applicationData: applicationDataFile?.data.toString("hex") ?? null,
applicationName: applicationNameFile?.name ?? null,
};

// Make sure all props are defined
Expand Down Expand Up @@ -609,6 +633,7 @@ function serializeCommonApplicationObjects(nvm: NVMJSON): NVM3Object[] {
]),
enablePTI: nvm.controller.rfConfig.enablePTI ?? undefined,
maxTXPower: nvm.controller.rfConfig.maxTXPower ?? undefined,
nodeIdType: nvm.controller.rfConfig.nodeIdType ?? undefined,
fileVersion: nvm.controller.applicationVersion,
});
ret.push(applRFConfigFile.serialize());
Expand All @@ -623,6 +648,15 @@ function serializeCommonApplicationObjects(nvm: NVMJSON): NVM3Object[] {
ret.push(applDataFile.serialize());
}

if (nvm.controller.applicationName && nvm.meta?.sharedFileSystem) {
// The application name only seems to be used with the shared file system
const applNameFile = new ApplicationNameFile({
name: nvm.controller.applicationName,
fileVersion: nvm.controller.applicationVersion,
});
ret.push(applNameFile.serialize());
}

return ret;
}

Expand Down Expand Up @@ -858,12 +892,13 @@ export function jsonToNVMObjects_v7_11_0(
let targetProtocolVersion: semver.SemVer;
let targetProtocolFormat: number;

// We currently support application version migrations up to 7.19.1
// We currently support application version migrations up to:
const HIGHEST_SUPPORTED_SDK_VERSION = "7.21.0";
// For all higher ones, set the highest version we support and let the controller handle the migration itself
if (semver.lte(targetSDKVersion, "7.19.1")) {
if (semver.lte(targetSDKVersion, HIGHEST_SUPPORTED_SDK_VERSION)) {
targetApplicationVersion = semver.parse(targetSDKVersion)!;
} else {
targetApplicationVersion = semver.parse("7.19.1")!;
targetApplicationVersion = semver.parse(HIGHEST_SUPPORTED_SDK_VERSION)!;
}

// The protocol version file only seems to be updated when the format of the protocol file system changes
Expand Down Expand Up @@ -905,7 +940,10 @@ export function jsonToNVMObjects_v7_11_0(
};

// Application files
const applVersionFile = new ApplicationVersionFile({
const ApplicationVersionConstructor = json.meta?.sharedFileSystem
? ApplicationVersionFile800
: ApplicationVersionFile;
const applVersionFile = new ApplicationVersionConstructor({
// The SDK compares 4-byte values where the format is set to 0 to determine whether a migration is needed
format: 0,
major: targetApplicationVersion.major,
Expand All @@ -923,6 +961,7 @@ export function jsonToNVMObjects_v7_11_0(
measured0dBm: +3.3,
enablePTI: null,
maxTXPower: null,
nodeIdType: null,
};

// Make sure the RF config format matches the application version.
Expand All @@ -931,6 +970,9 @@ export function jsonToNVMObjects_v7_11_0(
target.controller.rfConfig.enablePTI ??= 0;
target.controller.rfConfig.maxTXPower ??= 14.0;
}
if (semver.gte(targetSDKVersion, "7.21.0")) {
target.controller.rfConfig.nodeIdType ??= NodeIDType.Short;
}

addApplicationObjects(...serializeCommonApplicationObjects(target));

Expand Down Expand Up @@ -1041,8 +1083,18 @@ export function nvmToJSON(
debugLogs: boolean = false,
): Required<NVMJSON> {
const nvm = parseNVM(buffer, debugLogs);
const ret = nvmObjectsToJSON(nvm.applicationObjects, nvm.protocolObjects);
ret.meta = getNVMMeta(nvm.protocolPages[0]);
const objects = new Map([
...nvm.applicationObjects,
...nvm.protocolObjects,
]);
// 800 series doesn't distinguish between the storage for application and protocol objects
const sharedFileSystem = nvm.applicationObjects.size > 0
&& nvm.protocolObjects.size === 0;
const ret = nvmObjectsToJSON(objects);
const firstPage = sharedFileSystem
? nvm.applicationPages[0]
: nvm.protocolPages[0];
ret.meta = getNVMMeta(firstPage, sharedFileSystem);
return ret as Required<NVMJSON>;
}

Expand Down Expand Up @@ -1374,12 +1426,20 @@ export function migrateNVM(sourceNVM: Buffer, targetNVM: Buffer): Buffer {
&& targetProtocolFileFormat > MAX_PROTOCOL_FILE_FORMAT
&& sourceProtocolFileFormat
&& sourceProtocolFileFormat <= targetProtocolFileFormat
&& sourceNVM.length === targetNVM.length
) {
// ...both the source and the target are 700 series, but at least the target uses an unsupported protocol version.
// We can be sure hwoever that the target can upgrade any 700 series NVM to its protocol version, as long as the
// We can be sure however that the target can upgrade any 700 series NVM to its protocol version, as long as the
// source protocol version is not higher than the target's
return sourceNVM;
} else if (source.type === 700 && target.type === 700) {
} else if (
source.type === 700
&& target.type === 700
// ...the source and target NVMs have the same size and structure
&& sourceNVM.length === targetNVM.length
&& source.json.meta.sharedFileSystem
=== target.json.meta.sharedFileSystem
) {
// ... the source and target protocol versions are compatible without conversion
const sourceProtocolVersion = source.json.controller.protocolVersion;
const targetProtocolVersion = target.json.controller.protocolVersion;
Expand Down
39 changes: 39 additions & 0 deletions packages/nvmedit/src/files/ApplicationNameFile.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
import { cpp2js } from "@zwave-js/shared";
import { type NVMObject } from "..";
import {
NVMFile,
type NVMFileCreationOptions,
type NVMFileDeserializationOptions,
getNVMFileIDStatic,
gotDeserializationOptions,
nvmFileID,
} from "./NVMFile";

export interface ApplicationNameFileOptions extends NVMFileCreationOptions {
name: string;
}

@nvmFileID(0x4100c)
export class ApplicationNameFile extends NVMFile {
public constructor(
options: NVMFileDeserializationOptions | ApplicationNameFileOptions,
) {
super(options);
if (gotDeserializationOptions(options)) {
this.name = cpp2js(this.payload.toString("utf8"));
} else {
this.name = options.name;
}
}

public name: string;

public serialize(): NVMObject {
// Return a zero-terminated string with a fixed length of 30 bytes
const nameAsString = Buffer.from(this.name, "utf8");
this.payload = Buffer.alloc(30, 0);
nameAsString.subarray(0, this.payload.length - 1).copy(this.payload);
return super.serialize();
}
}
export const ApplicationNameFileID = getNVMFileIDStatic(ApplicationNameFile);
Loading

0 comments on commit 6f8bb38

Please sign in to comment.