mirror of
https://github.com/Koenkk/zigbee2mqtt.git
synced 2026-06-21 12:41:47 +00:00
399 lines
18 KiB
TypeScript
399 lines
18 KiB
TypeScript
// biome-ignore assist/source/organizeImports: import mocks first
|
|
import {afterAll, afterEach, beforeAll, beforeEach, describe, expect, it, vi} from "vitest";
|
|
import * as data from "../mocks/data";
|
|
import {mockLogger} from "../mocks/logger";
|
|
import {mockMQTTEndAsync, mockMQTTPublishAsync} from "../mocks/mqtt";
|
|
import {flushPromises} from "../mocks/utils";
|
|
import {devices, mockController as mockZHController, returnDevices} from "../mocks/zigbeeHerdsman";
|
|
|
|
import fs from "node:fs";
|
|
import path from "node:path";
|
|
import stringify from "json-stable-stringify-without-jsonify";
|
|
import {Controller} from "../../lib/controller";
|
|
import ExternalExtensions from "../../lib/extension/externalExtensions";
|
|
import * as settings from "../../lib/util/settings";
|
|
|
|
const BASE_DIR = "external_extensions";
|
|
|
|
describe("Extension: ExternalExtensions", () => {
|
|
let controller: Controller;
|
|
const mockBasePath = path.join(data.mockDir, BASE_DIR);
|
|
|
|
const rmSyncSpy = vi.spyOn(fs, "rmSync");
|
|
const writeFileSyncSpy = vi.spyOn(fs, "writeFileSync");
|
|
|
|
const mocksClear = [
|
|
mockMQTTEndAsync,
|
|
mockMQTTPublishAsync,
|
|
mockLogger.debug,
|
|
mockLogger.error,
|
|
mockZHController.stop,
|
|
devices.bulb.save,
|
|
rmSyncSpy,
|
|
writeFileSyncSpy,
|
|
];
|
|
|
|
const useAssets = (mtype: "cjs" | "mjs"): void => {
|
|
fs.cpSync(path.join(__dirname, "..", "assets", BASE_DIR, mtype), mockBasePath, {recursive: true});
|
|
};
|
|
|
|
const getFileCode = (mtype: "cjs" | "mjs", fileName: string): string => {
|
|
return fs.readFileSync(path.join(__dirname, "..", "assets", BASE_DIR, mtype, fileName), "utf8");
|
|
};
|
|
|
|
const resetExtension = async (): Promise<void> => {
|
|
await controller.removeExtension(controller.getExtension("ExternalExtensions")!);
|
|
await controller.addExtension(new ExternalExtensions(...controller.extensionArgs));
|
|
};
|
|
|
|
beforeAll(async () => {
|
|
vi.useFakeTimers();
|
|
|
|
controller = new Controller(vi.fn(), vi.fn());
|
|
|
|
await controller.start();
|
|
await flushPromises();
|
|
});
|
|
|
|
afterAll(() => {
|
|
vi.useRealTimers();
|
|
});
|
|
|
|
beforeEach(() => {
|
|
for (const mock of mocksClear) mock.mockClear();
|
|
data.writeDefaultConfiguration();
|
|
data.writeDefaultState();
|
|
settings.reRead();
|
|
returnDevices.splice(0);
|
|
});
|
|
|
|
afterEach(async () => {
|
|
await controller?.stop();
|
|
await flushPromises();
|
|
|
|
expect(fs.existsSync(path.join(mockBasePath, "node_modules"))).toStrictEqual(false);
|
|
|
|
fs.rmSync(mockBasePath, {recursive: true, force: true});
|
|
});
|
|
|
|
describe("from folder", () => {
|
|
beforeEach(() => {
|
|
controller = new Controller(vi.fn(), vi.fn());
|
|
});
|
|
|
|
it("loads nothing", async () => {
|
|
await controller.start();
|
|
await flushPromises();
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/bridge/extensions", stringify([]), {retain: true});
|
|
});
|
|
|
|
it("CJS: loads extensions", async () => {
|
|
useAssets("cjs");
|
|
|
|
await controller.start();
|
|
await flushPromises();
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example2/extension", "call2 from constructor", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example2/extension", "call2 from start", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from constructor", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from start", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/extensions",
|
|
stringify([
|
|
{name: "example2Extension.js", code: getFileCode("cjs", "example2Extension.js")},
|
|
{name: "exampleExtension.js", code: getFileCode("cjs", "exampleExtension.js")},
|
|
]),
|
|
{retain: true},
|
|
);
|
|
});
|
|
|
|
it("MJS: loads extensions", async () => {
|
|
useAssets("mjs");
|
|
|
|
await controller.start();
|
|
await flushPromises();
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example2/extension", "call2 from constructor", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example2/extension", "call2 from start", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from constructor", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from start", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/extensions",
|
|
stringify([
|
|
{name: "example2Extension.mjs", code: getFileCode("mjs", "example2Extension.mjs")},
|
|
{name: "exampleExtension.mjs", code: getFileCode("mjs", "exampleExtension.mjs")},
|
|
]),
|
|
{retain: true},
|
|
);
|
|
});
|
|
|
|
it("starts extensions only once", async () => {
|
|
useAssets("mjs");
|
|
|
|
await controller.start();
|
|
await flushPromises();
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension/counter", "start 0", {});
|
|
expect(mockMQTTPublishAsync).not.toHaveBeenCalledWith("zigbee2mqtt/example/extension/counter", "start 1", {});
|
|
});
|
|
|
|
it("loads all valid extensions, relocates & skips ones with errors", async () => {
|
|
useAssets("mjs");
|
|
|
|
const filepath = path.join(mockBasePath, "invalid.mjs");
|
|
|
|
fs.writeFileSync(filepath, "invalid js", "utf8");
|
|
|
|
await controller.start();
|
|
await flushPromises();
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/extensions",
|
|
stringify([
|
|
{name: "example2Extension.mjs", code: getFileCode("mjs", "example2Extension.mjs")},
|
|
{name: "exampleExtension.mjs", code: getFileCode("mjs", "exampleExtension.mjs")},
|
|
]),
|
|
{retain: true},
|
|
);
|
|
expect(fs.existsSync(filepath)).toStrictEqual(false);
|
|
expect(fs.existsSync(path.join(mockBasePath, "invalid.mjs.invalid"))).toStrictEqual(true);
|
|
});
|
|
|
|
it("updates after edit from MQTT", async () => {
|
|
const extensionName = "exampleExtension.js";
|
|
let extensionCode = getFileCode("cjs", extensionName);
|
|
|
|
useAssets("cjs");
|
|
await controller.start();
|
|
await flushPromises();
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example2/extension", "call2 from constructor", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example2/extension", "call2 from start", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from constructor", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from start", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/extensions",
|
|
stringify([
|
|
{name: "example2Extension.js", code: getFileCode("cjs", "example2Extension.js")},
|
|
{name: extensionName, code: extensionCode},
|
|
]),
|
|
{retain: true},
|
|
);
|
|
|
|
extensionCode = extensionCode
|
|
.replace('"call from start"', '"call from start - edited"')
|
|
.replace('"call from stop"', '"call from stop - edited"');
|
|
|
|
mockMQTTPublishAsync.mockClear();
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/save",
|
|
message: {name: extensionName, code: extensionCode},
|
|
});
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from stop", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from start - edited", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/extensions",
|
|
stringify([
|
|
{name: "example2Extension.js", code: getFileCode("cjs", "example2Extension.js")},
|
|
{name: "exampleExtension.js", code: extensionCode},
|
|
]),
|
|
{retain: true},
|
|
);
|
|
|
|
extensionCode = extensionCode
|
|
.replace('"call from start - edited"', '"call from start"')
|
|
.replace('"call from stop - edited"', '"call from stop"');
|
|
|
|
mockMQTTPublishAsync.mockClear();
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/save",
|
|
message: {name: "exampleExtension.js", code: extensionCode},
|
|
});
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from stop - edited", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from start", {});
|
|
expect(mockMQTTPublishAsync).not.toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from start - edited", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/extensions",
|
|
stringify([
|
|
{name: "example2Extension.js", code: getFileCode("cjs", "example2Extension.js")},
|
|
{name: "exampleExtension.js", code: extensionCode},
|
|
]),
|
|
{retain: true},
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("from MQTT", () => {
|
|
it("CJS: saves and removes", async () => {
|
|
const extensionName = "foo.js";
|
|
const extensionCode = getFileCode("cjs", "exampleExtension.js");
|
|
|
|
await resetExtension();
|
|
for (const mock of mocksClear) mock.mockClear();
|
|
|
|
//-- SAVE
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/save",
|
|
message: {name: extensionName, code: extensionCode},
|
|
});
|
|
|
|
expect(writeFileSyncSpy).toHaveBeenCalledWith(expect.stringContaining(extensionName), extensionCode, "utf8");
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from constructor", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from start", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/extensions",
|
|
stringify([{name: extensionName, code: extensionCode}]),
|
|
{
|
|
retain: true,
|
|
},
|
|
);
|
|
// Ensure that the .tmp import file is deleted.
|
|
expect(fs.readdirSync(mockBasePath)).toStrictEqual(["foo.js", "node_modules"]);
|
|
|
|
//-- REMOVE
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/remove",
|
|
message: {name: extensionName},
|
|
});
|
|
|
|
expect(rmSyncSpy).toHaveBeenCalledWith(expect.stringContaining(extensionName), {force: true});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from stop", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/bridge/extensions", stringify([]), {retain: true});
|
|
});
|
|
|
|
it("MJS: saves and removes", async () => {
|
|
const extensionName = "foo.mjs";
|
|
const extensionCode = getFileCode("mjs", "exampleExtension.mjs");
|
|
|
|
await resetExtension();
|
|
for (const mock of mocksClear) mock.mockClear();
|
|
|
|
//-- SAVE
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/save",
|
|
message: {name: extensionName, code: extensionCode},
|
|
});
|
|
|
|
expect(writeFileSyncSpy).toHaveBeenCalledWith(expect.stringContaining(extensionName), extensionCode, "utf8");
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from constructor", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from start", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/extensions",
|
|
stringify([{name: extensionName, code: extensionCode}]),
|
|
{
|
|
retain: true,
|
|
},
|
|
);
|
|
|
|
//-- REMOVE
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/remove",
|
|
message: {name: extensionName},
|
|
});
|
|
|
|
expect(rmSyncSpy).toHaveBeenCalledWith(expect.stringContaining(extensionName), {force: true});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/example/extension", "call from stop", {});
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith("zigbee2mqtt/bridge/extensions", stringify([]), {retain: true});
|
|
});
|
|
|
|
it("returns error on invalid name", async () => {
|
|
const extensionName = "foo1";
|
|
|
|
await resetExtension();
|
|
for (const mock of mocksClear) mock.mockClear();
|
|
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/save",
|
|
message: {name: extensionName, code: "a"},
|
|
});
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/response/extension/save",
|
|
expect.stringContaining(`JavaScript file must have '.mjs', '.js' or '.cjs' extension`),
|
|
{},
|
|
);
|
|
expect(writeFileSyncSpy).toHaveBeenCalledTimes(0);
|
|
});
|
|
|
|
it("returns error on invalid code", async () => {
|
|
const extensionName = "foo1.js";
|
|
const extensionCode = "definetly not a correct javascript code";
|
|
|
|
await resetExtension();
|
|
for (const mock of mocksClear) mock.mockClear();
|
|
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/save",
|
|
message: {name: extensionName, code: extensionCode},
|
|
});
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/response/extension/save",
|
|
expect.stringContaining(`"error":"${extensionName} contains invalid code`),
|
|
{},
|
|
);
|
|
expect(writeFileSyncSpy).toHaveBeenCalledWith(expect.stringContaining(extensionName), extensionCode, "utf8");
|
|
});
|
|
|
|
it("returns error on invalid removal", async () => {
|
|
const extensionName = "foo2.js";
|
|
|
|
await resetExtension();
|
|
for (const mock of mocksClear) mock.mockClear();
|
|
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/remove",
|
|
message: {name: extensionName},
|
|
});
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/response/extension/remove",
|
|
expect.stringContaining("doesn't exists"),
|
|
{},
|
|
);
|
|
expect(rmSyncSpy).not.toHaveBeenCalledWith(expect.stringContaining(extensionName), {force: true});
|
|
});
|
|
|
|
it("handles invalid payloads", async () => {
|
|
await resetExtension();
|
|
for (const mock of mocksClear) mock.mockClear();
|
|
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/save",
|
|
message: {name: "foo3.js", transaction: 1 /* code */},
|
|
});
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/response/extension/save",
|
|
stringify({data: {}, status: "error", error: "Invalid payload", transaction: 1}),
|
|
{},
|
|
);
|
|
|
|
await (controller.getExtension("ExternalExtensions")! as ExternalExtensions).onMQTTMessage({
|
|
topic: "zigbee2mqtt/bridge/request/extension/remove",
|
|
message: {namex: "foo3.js", transaction: 2},
|
|
});
|
|
|
|
expect(mockMQTTPublishAsync).toHaveBeenCalledWith(
|
|
"zigbee2mqtt/bridge/response/extension/remove",
|
|
stringify({data: {}, status: "error", error: "Invalid payload", transaction: 2}),
|
|
{},
|
|
);
|
|
});
|
|
});
|
|
|
|
it("doesn't add extension when external JS disabled", async () => {
|
|
settings.set(["advanced", "enable_external_js"], false);
|
|
|
|
controller = new Controller(vi.fn(), vi.fn());
|
|
|
|
await controller.start();
|
|
await flushPromises();
|
|
|
|
expect(controller.getExtension("ExternalExtensions")).toBeUndefined();
|
|
});
|
|
});
|