@switchbot/homebridge-switchbot 5.0.0-beta.3 → 5.0.0-beta.30
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +13 -0
- package/config.schema.json +42 -4
- package/dist/devices-hap/device.d.ts +1 -0
- package/dist/devices-hap/device.d.ts.map +1 -1
- package/dist/devices-hap/device.js +70 -30
- package/dist/devices-hap/device.js.map +1 -1
- package/dist/devices-matter/BaseMatterAccessory.d.ts +23 -0
- package/dist/devices-matter/BaseMatterAccessory.d.ts.map +1 -1
- package/dist/devices-matter/BaseMatterAccessory.js +167 -5
- package/dist/devices-matter/BaseMatterAccessory.js.map +1 -1
- package/dist/devices-matter/ColorLightAccessory.d.ts.map +1 -1
- package/dist/devices-matter/ColorLightAccessory.js +12 -12
- package/dist/devices-matter/ColorLightAccessory.js.map +1 -1
- package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +1 -1
- package/dist/devices-matter/ColorTemperatureLightAccessory.js +5 -7
- package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +1 -1
- package/dist/devices-matter/DimmableLightAccessory.js +9 -9
- package/dist/devices-matter/DimmableLightAccessory.js.map +1 -1
- package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +1 -1
- package/dist/devices-matter/ExtendedColorLightAccessory.js +14 -15
- package/dist/devices-matter/ExtendedColorLightAccessory.js.map +1 -1
- package/dist/devices-matter/OnOffLightAccessory.d.ts.map +1 -1
- package/dist/devices-matter/OnOffLightAccessory.js +8 -16
- package/dist/devices-matter/OnOffLightAccessory.js.map +1 -1
- package/dist/devices-matter/OnOffOutletAccessory.d.ts +2 -0
- package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +1 -1
- package/dist/devices-matter/OnOffOutletAccessory.js +10 -7
- package/dist/devices-matter/OnOffOutletAccessory.js.map +1 -1
- package/dist/devices-matter/OnOffSwitchAccessory.js +2 -2
- package/dist/devices-matter/OnOffSwitchAccessory.js.map +1 -1
- package/dist/homebridge-ui/public/index.html +48 -1
- package/dist/homebridge-ui/server.js +35 -0
- package/dist/homebridge-ui/server.js.map +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +4 -5
- package/dist/index.js.map +1 -1
- package/dist/index.test.js +7 -2
- package/dist/index.test.js.map +1 -1
- package/dist/irdevice/irdevice.d.ts +11 -10
- package/dist/irdevice/irdevice.d.ts.map +1 -1
- package/dist/irdevice/irdevice.js +76 -35
- package/dist/irdevice/irdevice.js.map +1 -1
- package/dist/platform-hap.d.ts +11 -14
- package/dist/platform-hap.d.ts.map +1 -1
- package/dist/platform-hap.js +64 -64
- package/dist/platform-hap.js.map +1 -1
- package/dist/platform-matter.d.ts +85 -6
- package/dist/platform-matter.d.ts.map +1 -1
- package/dist/platform-matter.js +1736 -84
- package/dist/platform-matter.js.map +1 -1
- package/dist/settings.d.ts +9 -0
- package/dist/settings.d.ts.map +1 -1
- package/dist/settings.js.map +1 -1
- package/dist/test/devices-matter/baseMatterAccessory.test.d.ts +2 -0
- package/dist/test/devices-matter/baseMatterAccessory.test.d.ts.map +1 -0
- package/dist/test/devices-matter/baseMatterAccessory.test.js +71 -0
- package/dist/test/devices-matter/baseMatterAccessory.test.js.map +1 -0
- package/dist/test/helpers/platform-fixtures.d.ts +9 -0
- package/dist/test/helpers/platform-fixtures.d.ts.map +1 -0
- package/dist/test/helpers/platform-fixtures.js +30 -0
- package/dist/test/helpers/platform-fixtures.js.map +1 -0
- package/dist/test/platform-matter.additional.test.d.ts +2 -0
- package/dist/test/platform-matter.additional.test.d.ts.map +1 -0
- package/dist/test/platform-matter.additional.test.js +35 -0
- package/dist/test/platform-matter.additional.test.js.map +1 -0
- package/dist/test/platform-matter.bleparse.test.d.ts +2 -0
- package/dist/test/platform-matter.bleparse.test.d.ts.map +1 -0
- package/dist/test/platform-matter.bleparse.test.js +43 -0
- package/dist/test/platform-matter.bleparse.test.js.map +1 -0
- package/dist/test/platform-matter.cleanup.test.d.ts +2 -0
- package/dist/test/platform-matter.cleanup.test.d.ts.map +1 -0
- package/dist/test/platform-matter.cleanup.test.js +70 -0
- package/dist/test/platform-matter.cleanup.test.js.map +1 -0
- package/dist/test/platform-matter.keepstale.test.d.ts +2 -0
- package/dist/test/platform-matter.keepstale.test.d.ts.map +1 -0
- package/dist/test/platform-matter.keepstale.test.js +27 -0
- package/dist/test/platform-matter.keepstale.test.js.map +1 -0
- package/dist/test/platform-matter.mapping.test.d.ts +2 -0
- package/dist/test/platform-matter.mapping.test.d.ts.map +1 -0
- package/dist/test/platform-matter.mapping.test.js +43 -0
- package/dist/test/platform-matter.mapping.test.js.map +1 -0
- package/dist/test/platform-matter.openapi-mapping.test.d.ts +2 -0
- package/dist/test/platform-matter.openapi-mapping.test.d.ts.map +1 -0
- package/dist/test/platform-matter.openapi-mapping.test.js +84 -0
- package/dist/test/platform-matter.openapi-mapping.test.js.map +1 -0
- package/dist/test/platform-matter.test.d.ts +2 -0
- package/dist/test/platform-matter.test.d.ts.map +1 -0
- package/dist/test/platform-matter.test.js +117 -0
- package/dist/test/platform-matter.test.js.map +1 -0
- package/dist/test/platform-matter.unregister.test.d.ts +2 -0
- package/dist/test/platform-matter.unregister.test.d.ts.map +1 -0
- package/dist/test/platform-matter.unregister.test.js +30 -0
- package/dist/test/platform-matter.unregister.test.js.map +1 -0
- package/dist/utils.d.ts +127 -0
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +380 -0
- package/dist/utils.js.map +1 -1
- package/dist/utils.test.d.ts +2 -0
- package/dist/utils.test.d.ts.map +1 -0
- package/dist/utils.test.js +95 -0
- package/dist/utils.test.js.map +1 -0
- package/dist/verifyconfig.test.js +2 -2
- package/dist/verifyconfig.test.js.map +1 -1
- package/docs/assets/main.js +2 -2
- package/docs/index.html +2 -2
- package/docs/variables/default.html +1 -1
- package/package.json +14 -14
- package/src/devices-hap/device.ts +68 -30
- package/src/devices-matter/BaseMatterAccessory.ts +168 -5
- package/src/devices-matter/ColorLightAccessory.ts +12 -12
- package/src/devices-matter/ColorTemperatureLightAccessory.ts +5 -7
- package/src/devices-matter/DimmableLightAccessory.ts +9 -9
- package/src/devices-matter/ExtendedColorLightAccessory.ts +14 -15
- package/src/devices-matter/OnOffLightAccessory.ts +8 -16
- package/src/devices-matter/OnOffOutletAccessory.ts +12 -7
- package/src/devices-matter/OnOffSwitchAccessory.ts +2 -2
- package/src/homebridge-ui/public/index.html +48 -1
- package/src/homebridge-ui/server.ts +37 -0
- package/src/index.test.ts +7 -2
- package/src/index.ts +4 -5
- package/src/irdevice/irdevice.ts +74 -35
- package/src/platform-hap.ts +68 -73
- package/src/platform-matter.ts +1772 -87
- package/src/settings.ts +13 -0
- package/src/test/devices-matter/baseMatterAccessory.test.ts +88 -0
- package/src/test/helpers/platform-fixtures.ts +33 -0
- package/src/test/platform-matter.additional.test.ts +44 -0
- package/src/test/platform-matter.bleparse.test.ts +47 -0
- package/src/test/platform-matter.cleanup.test.ts +86 -0
- package/src/test/platform-matter.keepstale.test.ts +37 -0
- package/src/test/platform-matter.mapping.test.ts +57 -0
- package/src/test/platform-matter.openapi-mapping.test.ts +109 -0
- package/src/test/platform-matter.test.ts +144 -0
- package/src/test/platform-matter.unregister.test.ts +39 -0
- package/src/utils.test.ts +96 -0
- package/src/utils.ts +391 -3
- package/src/verifyconfig.test.ts +11 -10
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { SwitchBotMatterPlatform } from '../platform-matter.js';
|
|
3
|
+
import { formatDeviceIdAsMac } from '../utils.js';
|
|
4
|
+
import { makeApiStub, makeLogStub } from './helpers/platform-fixtures.js';
|
|
5
|
+
describe('platform-matter lifecycle cleanup', () => {
|
|
6
|
+
it('clearDeviceResources removes timers, instances and BLE handler entries', async () => {
|
|
7
|
+
// Setup stubbed API and logs
|
|
8
|
+
const handlers = {};
|
|
9
|
+
const api = makeApiStub({ registerPlatformAccessories: vi.fn(), unregisterPlatformAccessories: vi.fn(), clusterNames: { OnOff: 'OnOff' } });
|
|
10
|
+
api._handlers = handlers;
|
|
11
|
+
// keep api.on as a single-statement arrow to satisfy lint
|
|
12
|
+
api.on = (ev, fn) => (handlers[ev] = fn);
|
|
13
|
+
const log = makeLogStub();
|
|
14
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
15
|
+
// Insert a fake timer, accessory instance, and BLE handler
|
|
16
|
+
const deviceId = 'AA:BB:CC:11:22:33';
|
|
17
|
+
const nid = platform.normalizeDeviceId(deviceId);
|
|
18
|
+
const timer = setInterval(() => { }, 100000);
|
|
19
|
+
platform.refreshTimers.set(nid, timer);
|
|
20
|
+
platform.accessoryInstances.set(nid, { dummy: true });
|
|
21
|
+
platform.bleEventHandler[deviceId.toLowerCase()] = () => { };
|
|
22
|
+
// Ensure they exist prior
|
|
23
|
+
expect(platform.refreshTimers.get(nid)).toBeDefined();
|
|
24
|
+
expect(platform.accessoryInstances.get(nid)).toBeDefined();
|
|
25
|
+
expect(platform.bleEventHandler[deviceId.toLowerCase()]).toBeDefined();
|
|
26
|
+
platform.clearDeviceResources(deviceId);
|
|
27
|
+
// Now they should be removed
|
|
28
|
+
expect(platform.refreshTimers.get(nid)).toBeUndefined();
|
|
29
|
+
expect(platform.accessoryInstances.get(nid)).toBeUndefined();
|
|
30
|
+
expect(platform.bleEventHandler[deviceId.toLowerCase()]).toBeUndefined();
|
|
31
|
+
});
|
|
32
|
+
it('shutdown handler clears all timers and handlers when invoked', async () => {
|
|
33
|
+
// Setup stubbed API and logs
|
|
34
|
+
const handlers = {};
|
|
35
|
+
const api = makeApiStub({ registerPlatformAccessories: vi.fn(), unregisterPlatformAccessories: vi.fn(), clusterNames: { OnOff: 'OnOff' } });
|
|
36
|
+
api._handlers = handlers;
|
|
37
|
+
// keep api.on as a single-statement arrow to satisfy lint
|
|
38
|
+
api.on = (ev, fn) => (handlers[ev] = fn);
|
|
39
|
+
const log = makeLogStub();
|
|
40
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
41
|
+
// Add two timers to the platform (using normalized ids)
|
|
42
|
+
const ids = ['devA', 'devB'];
|
|
43
|
+
for (const id of ids) {
|
|
44
|
+
const nid = platform.normalizeDeviceId(id);
|
|
45
|
+
const t = setInterval(() => { }, 100000);
|
|
46
|
+
platform.refreshTimers.set(nid, t);
|
|
47
|
+
platform.accessoryInstances.set(nid, { dummy: true });
|
|
48
|
+
try {
|
|
49
|
+
const mac = formatDeviceIdAsMac(id).toLowerCase();
|
|
50
|
+
platform.bleEventHandler[mac] = () => { };
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
// ignore formatting errors in this test
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
// Invoke didFinishLaunching to ensure the platform registered its shutdown handler
|
|
57
|
+
await Promise.resolve(api._handlers.didFinishLaunching?.());
|
|
58
|
+
// Shutdown handler should now be registered on api._handlers.shutdown
|
|
59
|
+
expect(typeof api._handlers.shutdown).toBe('function');
|
|
60
|
+
// Call the shutdown handler
|
|
61
|
+
await Promise.resolve(api._handlers.shutdown());
|
|
62
|
+
// All refresh timers should be cleared
|
|
63
|
+
for (const id of ids) {
|
|
64
|
+
const nid = platform.normalizeDeviceId(id);
|
|
65
|
+
expect(platform.refreshTimers.get(nid)).toBeUndefined();
|
|
66
|
+
expect(platform.accessoryInstances.get(nid)).toBeUndefined();
|
|
67
|
+
}
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
//# sourceMappingURL=platform-matter.cleanup.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.cleanup.test.js","sourceRoot":"","sources":["../../src/test/platform-matter.cleanup.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.keepstale.test.d.ts","sourceRoot":"","sources":["../../src/test/platform-matter.keepstale.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { SwitchBotMatterPlatform } from '../platform-matter.js';
|
|
3
|
+
import { makeApiStub, makeLogStub } from './helpers/platform-fixtures.js';
|
|
4
|
+
describe('keepStaleAccessories config flag behavior', () => {
|
|
5
|
+
it('keeps previously-registered accessories when options.keepStaleAccessories=true', async () => {
|
|
6
|
+
const unregister = vi.fn();
|
|
7
|
+
const register = vi.fn();
|
|
8
|
+
const api = makeApiStub({ unregisterPlatformAccessories: unregister, registerPlatformAccessories: register });
|
|
9
|
+
const log = makeLogStub();
|
|
10
|
+
// Create platform with keepStaleAccessories = true
|
|
11
|
+
const platform = new SwitchBotMatterPlatform(log, { options: { keepStaleAccessories: true } }, api);
|
|
12
|
+
platform.discoveredDevices = [{ deviceId: 'DEV1', deviceType: 'Plug', deviceName: 'Device 1' }];
|
|
13
|
+
// Insert a stale accessory into matterAccessories
|
|
14
|
+
const staleUuid = api.matter.uuid.generate('stale-device');
|
|
15
|
+
const staleAcc = { uuid: staleUuid, displayName: 'Stale', context: { deviceId: 'STALE_DEVICE' } };
|
|
16
|
+
platform.matterAccessories.set(staleUuid, staleAcc);
|
|
17
|
+
// Mock createAccessoryFromDevice to return a simple accessory for DEV1
|
|
18
|
+
vi.spyOn(platform, 'createAccessoryFromDevice').mockResolvedValue({ displayName: 'Device 1', uuid: 'uuid-DEV1', context: { deviceId: 'DEV1' } });
|
|
19
|
+
// Run registration which includes stale-removal logic
|
|
20
|
+
await platform.registerMatterAccessories();
|
|
21
|
+
// Expect unregister NOT called (we kept stale accessory)
|
|
22
|
+
expect(unregister).not.toHaveBeenCalled();
|
|
23
|
+
// Stale accessory should still be present in matterAccessories
|
|
24
|
+
expect(platform.matterAccessories.get(staleUuid)).toBeDefined();
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
//# sourceMappingURL=platform-matter.keepstale.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.keepstale.test.js","sourceRoot":"","sources":["../../src/test/platform-matter.keepstale.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAA;AAEjD,OAAO,EAAE,uBAAuB,EAAE,MAAM,uBAAuB,CAAA;AAC/D,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,gCAAgC,CAAA;AAEzE,QAAQ,CAAC,2CAA2C,EAAE,GAAG,EAAE;IACzD,EAAE,CAAC,gFAAgF,EAAE,KAAK,IAAI,EAAE;QAC9F,MAAM,UAAU,GAAG,EAAE,CAAC,EAAE,EAAE,CAAA;QAC1B,MAAM,QAAQ,GAAG,EAAE,CAAC,EAAE,EAAE,CAAA;QACxB,MAAM,GAAG,GAAQ,WAAW,CAAC,EAAE,6BAA6B,EAAE,UAAU,EAAE,2BAA2B,EAAE,QAAQ,EAAE,CAAC,CAAA;QAElH,MAAM,GAAG,GAAQ,WAAW,EAAE,CAAA;QAE9B,mDAAmD;QACnD,MAAM,QAAQ,GAAG,IAAI,uBAAuB,CAAC,GAAU,EAAE,EAAE,OAAO,EAAE,EAAE,oBAAoB,EAAE,IAAI,EAAE,EAAS,EAAE,GAAG,CAAC,CAGhH;QAAC,QAAgB,CAAC,iBAAiB,GAAG,CAAC,EAAE,QAAQ,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC,CAAA;QAEzG,kDAAkD;QAClD,MAAM,SAAS,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAA;QAC1D,MAAM,QAAQ,GAAQ,EAAE,IAAI,EAAE,SAAS,EAAE,WAAW,EAAE,OAAO,EAAE,OAAO,EAAE,EAAE,QAAQ,EAAE,cAAc,EAAE,EAAE,CACrG;QAAC,QAAgB,CAAC,iBAAiB,CAAC,GAAG,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAA;QAE7D,uEAAuE;QACvE,EAAE,CAAC,KAAK,CAAC,QAAe,EAAE,2BAA2B,CAAC,CAAC,iBAAiB,CAAC,EAAE,WAAW,EAAE,UAAU,EAAE,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAS,CAAC,CAAA;QAE9J,sDAAsD;QACtD,MAAO,QAAgB,CAAC,yBAAyB,EAAE,CAAA;QAEnD,yDAAyD;QACzD,MAAM,CAAC,UAAU,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAA;QAEzC,+DAA+D;QAC/D,MAAM,CAAE,QAAgB,CAAC,iBAAiB,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,WAAW,EAAE,CAAA;IAC1E,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.mapping.test.d.ts","sourceRoot":"","sources":["../../src/test/platform-matter.mapping.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { SwitchBotMatterPlatform } from '../platform-matter.js';
|
|
3
|
+
import { makeApiStub, makeLogStub } from './helpers/platform-fixtures.js';
|
|
4
|
+
describe('platform-matter mapping helper', () => {
|
|
5
|
+
it('prefers accessory instance update methods for battery and falls back to api.matter.updateAccessoryState', async () => {
|
|
6
|
+
const updateAccessoryState = vi.fn();
|
|
7
|
+
const api = makeApiStub({ updateAccessoryState, clusterNames: { OnOff: 'OnOff', LevelControl: 'LevelControl', ColorControl: 'ColorControl', PowerSource: 'powerSource' } });
|
|
8
|
+
const log = makeLogStub();
|
|
9
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
10
|
+
// Create a fake accessory instance with updateBatteryPercentage
|
|
11
|
+
const deviceId = 'DEV123';
|
|
12
|
+
const nid = platform.normalizeDeviceId(deviceId);
|
|
13
|
+
const fakeInstance = { updateBatteryPercentage: vi.fn() };
|
|
14
|
+
platform.accessoryInstances.set(nid, fakeInstance);
|
|
15
|
+
// Call the private helper with different battery field names
|
|
16
|
+
await platform.applyStatusToAccessory('uuid-test', { deviceId }, { batteryPercentage: 55 });
|
|
17
|
+
expect(fakeInstance.updateBatteryPercentage).toHaveBeenCalled();
|
|
18
|
+
platform.accessoryInstances.delete(nid);
|
|
19
|
+
await platform.applyStatusToAccessory('uuid-test', { deviceId }, { battery: 30 });
|
|
20
|
+
expect(updateAccessoryState).toHaveBeenCalled();
|
|
21
|
+
});
|
|
22
|
+
it('handles temperature and humidity synonyms', async () => {
|
|
23
|
+
const updateAccessoryState = vi.fn();
|
|
24
|
+
const api = makeApiStub({ updateAccessoryState });
|
|
25
|
+
const log = makeLogStub();
|
|
26
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
27
|
+
const deviceId = 'DEV-TEMP';
|
|
28
|
+
await platform.applyStatusToAccessory('uuid-temp', { deviceId }, { temp: 21.5, humid: 48 });
|
|
29
|
+
// Expect updateAccessoryState to have been called for temperature and humidity
|
|
30
|
+
expect(updateAccessoryState).toHaveBeenCalled();
|
|
31
|
+
});
|
|
32
|
+
it('applies VOC and PM10 mappings when present', async () => {
|
|
33
|
+
const updateAccessoryState = vi.fn();
|
|
34
|
+
const api = makeApiStub({ updateAccessoryState });
|
|
35
|
+
const log = makeLogStub();
|
|
36
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
37
|
+
const deviceId = 'DEV-AQ';
|
|
38
|
+
await platform.applyStatusToAccessory('uuid-aq', { deviceId }, { voc: 123, pm10: 56 });
|
|
39
|
+
// Expect updateAccessoryState (or safeUpdate fallback) to have been called for voc and pm10
|
|
40
|
+
expect(updateAccessoryState).toHaveBeenCalled();
|
|
41
|
+
});
|
|
42
|
+
});
|
|
43
|
+
//# sourceMappingURL=platform-matter.mapping.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.mapping.test.js","sourceRoot":"","sources":["../../src/test/platform-matter.mapping.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.openapi-mapping.test.d.ts","sourceRoot":"","sources":["../../src/test/platform-matter.openapi-mapping.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { SwitchBotMatterPlatform } from '../platform-matter.js';
|
|
3
|
+
import { makeApiStub, makeLogStub } from './helpers/platform-fixtures.js';
|
|
4
|
+
describe('platform-matter OpenAPI -> Matter mapping', () => {
|
|
5
|
+
it('maps light OpenAPI status (on/off, brightness, color, battery) to accessory instance helpers', async () => {
|
|
6
|
+
const api = makeApiStub();
|
|
7
|
+
const log = makeLogStub();
|
|
8
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
9
|
+
const deviceId = 'LIGHT-1';
|
|
10
|
+
const nid = platform.normalizeDeviceId(deviceId);
|
|
11
|
+
const instance = {
|
|
12
|
+
updateOnOffState: vi.fn(),
|
|
13
|
+
updateBrightness: vi.fn(),
|
|
14
|
+
updateHueSaturation: vi.fn(),
|
|
15
|
+
updateBatteryPercentage: vi.fn(),
|
|
16
|
+
};
|
|
17
|
+
platform.accessoryInstances.set(nid, instance);
|
|
18
|
+
const status = { power: true, brightness: 80, color: '255:128:64', battery: 92 };
|
|
19
|
+
await platform.applyStatusToAccessory('uuid-light', { deviceId }, status);
|
|
20
|
+
expect(instance.updateOnOffState).toHaveBeenCalled();
|
|
21
|
+
expect(instance.updateBrightness).toHaveBeenCalled();
|
|
22
|
+
expect(instance.updateHueSaturation).toHaveBeenCalled();
|
|
23
|
+
expect(instance.updateBatteryPercentage).toHaveBeenCalled();
|
|
24
|
+
});
|
|
25
|
+
it('maps MeterPro OpenAPI status (temp, humid, co2, pm25, voc) to accessory instance helpers', async () => {
|
|
26
|
+
const api = makeApiStub();
|
|
27
|
+
const log = makeLogStub();
|
|
28
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
29
|
+
const deviceId = 'METERPRO-1';
|
|
30
|
+
const nid = platform.normalizeDeviceId(deviceId);
|
|
31
|
+
const instance = {
|
|
32
|
+
updateTemperature: vi.fn(),
|
|
33
|
+
updateHumidity: vi.fn(),
|
|
34
|
+
updateCO2: vi.fn(),
|
|
35
|
+
updatePM25: vi.fn(),
|
|
36
|
+
updateVOC: vi.fn(),
|
|
37
|
+
};
|
|
38
|
+
platform.accessoryInstances.set(nid, instance);
|
|
39
|
+
const status = { temp: 21.4, humid: 45, co2: 410, pm25: 12, voc: 85 };
|
|
40
|
+
await platform.applyStatusToAccessory('uuid-meter', { deviceId }, status);
|
|
41
|
+
expect(instance.updateTemperature).toHaveBeenCalled();
|
|
42
|
+
expect(instance.updateHumidity).toHaveBeenCalled();
|
|
43
|
+
expect(instance.updateCO2).toHaveBeenCalled();
|
|
44
|
+
expect(instance.updatePM25).toHaveBeenCalled();
|
|
45
|
+
expect(instance.updateVOC).toHaveBeenCalled();
|
|
46
|
+
});
|
|
47
|
+
it('maps lock OpenAPI status to lock helper', async () => {
|
|
48
|
+
const api = makeApiStub();
|
|
49
|
+
const log = makeLogStub();
|
|
50
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
51
|
+
const deviceId = 'LOCK-1';
|
|
52
|
+
const nid = platform.normalizeDeviceId(deviceId);
|
|
53
|
+
const instance = { updateLockState: vi.fn() };
|
|
54
|
+
platform.accessoryInstances.set(nid, instance);
|
|
55
|
+
const status = { lock: true };
|
|
56
|
+
await platform.applyStatusToAccessory('uuid-lock', { deviceId }, status);
|
|
57
|
+
expect(instance.updateLockState).toHaveBeenCalled();
|
|
58
|
+
});
|
|
59
|
+
it('maps curtain position synonyms to lift position helper', async () => {
|
|
60
|
+
const api = makeApiStub();
|
|
61
|
+
const log = makeLogStub();
|
|
62
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
63
|
+
const deviceId = 'CURTAIN-1';
|
|
64
|
+
const nid = platform.normalizeDeviceId(deviceId);
|
|
65
|
+
const instance = { updateLiftPosition: vi.fn() };
|
|
66
|
+
platform.accessoryInstances.set(nid, instance);
|
|
67
|
+
await platform.applyStatusToAccessory('uuid-cur', { deviceId, deviceType: 'Curtain' }, { position: 25 });
|
|
68
|
+
await platform.applyStatusToAccessory('uuid-cur', { deviceId, deviceType: 'Curtain' }, { slidePosition: 75 });
|
|
69
|
+
expect(instance.updateLiftPosition).toHaveBeenCalled();
|
|
70
|
+
});
|
|
71
|
+
it('maps robot vacuum run state / on to updateRunMode or updateOperationalState', async () => {
|
|
72
|
+
const api = makeApiStub();
|
|
73
|
+
const log = makeLogStub();
|
|
74
|
+
const platform = new SwitchBotMatterPlatform(log, {}, api);
|
|
75
|
+
const deviceId = 'RVC-1';
|
|
76
|
+
const nid = platform.normalizeDeviceId(deviceId);
|
|
77
|
+
const instance = { updateRunMode: vi.fn(), updateOperationalState: vi.fn() };
|
|
78
|
+
platform.accessoryInstances.set(nid, instance);
|
|
79
|
+
await platform.applyStatusToAccessory('uuid-rvc', { deviceId, deviceType: 'K10+' }, { runState: 'cleaning', power: true });
|
|
80
|
+
const calls = (instance.updateRunMode.mock?.calls?.length || 0) + (instance.updateOperationalState.mock?.calls?.length || 0);
|
|
81
|
+
expect(calls).toBeGreaterThan(0);
|
|
82
|
+
});
|
|
83
|
+
});
|
|
84
|
+
//# sourceMappingURL=platform-matter.openapi-mapping.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.openapi-mapping.test.js","sourceRoot":"","sources":["../../src/test/platform-matter.openapi-mapping.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.test.d.ts","sourceRoot":"","sources":["../../src/test/platform-matter.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { SwitchBotMatterPlatform } from '../platform-matter.js';
|
|
3
|
+
import { PLATFORM_NAME, PLUGIN_NAME } from '../settings.js';
|
|
4
|
+
import { makeApiStub, makeLogStub } from './helpers/platform-fixtures.js';
|
|
5
|
+
describe('platform-matter discovered devices', () => {
|
|
6
|
+
it('uses discovered devices and registers them (platform + robotic)', async () => {
|
|
7
|
+
const mockRegister = vi.fn();
|
|
8
|
+
const api = makeApiStub({
|
|
9
|
+
registerPlatformAccessories: mockRegister,
|
|
10
|
+
clusterNames: { OnOff: 'OnOff', LevelControl: 'LevelControl', ColorControl: 'ColorControl' },
|
|
11
|
+
deviceTypes: { RoboticVacuumCleaner: 'rvc' },
|
|
12
|
+
});
|
|
13
|
+
const log = makeLogStub();
|
|
14
|
+
const config = {};
|
|
15
|
+
const platform = new SwitchBotMatterPlatform(log, config, api);
|
|
16
|
+
platform.discoveredDevices = [
|
|
17
|
+
{ deviceId: 'dev1', deviceName: 'Lamp', deviceType: 'Plug' },
|
|
18
|
+
{ deviceId: 'vac1', deviceName: 'Vac', deviceType: 'K10+' },
|
|
19
|
+
];
|
|
20
|
+
platform.createAccessoryFromDevice = async (dev) => ({ displayName: dev.deviceName, uuid: api.matter.uuid.generate(dev.deviceId) });
|
|
21
|
+
await platform.registerMatterAccessories();
|
|
22
|
+
// Ensure registerPlatformAccessories was called at least once
|
|
23
|
+
expect(mockRegister).toHaveBeenCalled();
|
|
24
|
+
// Sum total accessories registered across all calls
|
|
25
|
+
const totalRegistered = mockRegister.mock.calls.reduce((sum, call) => sum + ((call[2] && call[2].length) || 0), 0);
|
|
26
|
+
expect(totalRegistered).toBe(2);
|
|
27
|
+
// Verify PLUGIN_NAME and PLATFORM_NAME were used in the calls
|
|
28
|
+
for (const call of mockRegister.mock.calls) {
|
|
29
|
+
expect(call[0]).toBe(PLUGIN_NAME);
|
|
30
|
+
expect(call[1]).toBe(PLATFORM_NAME);
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
it('applies per-device config overrides when deviceId matches discovered device', async () => {
|
|
34
|
+
const mockRegister = vi.fn();
|
|
35
|
+
const api = {
|
|
36
|
+
matter: {
|
|
37
|
+
uuid: { generate: (s) => `uuid-${s}` },
|
|
38
|
+
registerPlatformAccessories: mockRegister,
|
|
39
|
+
clusterNames: { OnOff: 'OnOff', LevelControl: 'LevelControl', ColorControl: 'ColorControl' },
|
|
40
|
+
deviceTypes: { RoboticVacuumCleaner: 'rvc' },
|
|
41
|
+
},
|
|
42
|
+
isMatterAvailable: () => true,
|
|
43
|
+
isMatterEnabled: () => true,
|
|
44
|
+
on: () => { },
|
|
45
|
+
};
|
|
46
|
+
const log = { info: vi.fn(), warn: vi.fn(), debug: vi.fn(), error: vi.fn() };
|
|
47
|
+
const config = {
|
|
48
|
+
options: {
|
|
49
|
+
devices: [{ deviceId: 'dev1', configDeviceName: 'Configured Lamp', logging: 'debug' }],
|
|
50
|
+
},
|
|
51
|
+
};
|
|
52
|
+
const platform = new SwitchBotMatterPlatform(log, config, api);
|
|
53
|
+
platform.discoveredDevices = [{ deviceId: 'dev1', deviceName: 'Lamp', deviceType: 'Plug' }];
|
|
54
|
+
const seen = [];
|
|
55
|
+
platform.createAccessoryFromDevice = async (dev) => {
|
|
56
|
+
seen.push(dev);
|
|
57
|
+
return {
|
|
58
|
+
displayName: dev.deviceName ?? dev.configDeviceName,
|
|
59
|
+
uuid: api.matter.uuid.generate(dev.deviceId),
|
|
60
|
+
};
|
|
61
|
+
};
|
|
62
|
+
await platform.registerMatterAccessories();
|
|
63
|
+
// createAccessoryFromDevice should have been called once with merged config
|
|
64
|
+
expect(seen.length).toBe(1);
|
|
65
|
+
expect(seen[0].deviceId).toBe('dev1');
|
|
66
|
+
// per-device config values should be present on the merged device
|
|
67
|
+
expect(seen[0].configDeviceName).toBe('Configured Lamp');
|
|
68
|
+
expect(seen[0].logging).toBe('debug');
|
|
69
|
+
});
|
|
70
|
+
it('ignores config-only devices by default but includes them when allowConfigOnlyDevices=true', async () => {
|
|
71
|
+
const mockRegister = vi.fn();
|
|
72
|
+
const api = {
|
|
73
|
+
matter: {
|
|
74
|
+
uuid: { generate: (s) => `uuid-${s}` },
|
|
75
|
+
registerPlatformAccessories: mockRegister,
|
|
76
|
+
clusterNames: { OnOff: 'OnOff', LevelControl: 'LevelControl', ColorControl: 'ColorControl' },
|
|
77
|
+
deviceTypes: { RoboticVacuumCleaner: 'rvc' },
|
|
78
|
+
},
|
|
79
|
+
isMatterAvailable: () => true,
|
|
80
|
+
isMatterEnabled: () => true,
|
|
81
|
+
on: () => { },
|
|
82
|
+
};
|
|
83
|
+
const log = { info: vi.fn(), warn: vi.fn(), debug: vi.fn(), error: vi.fn() };
|
|
84
|
+
// config-only device (cfg1) and one discovered device (dev1)
|
|
85
|
+
const cfgOnly = { deviceId: 'cfg1', configDeviceName: 'OnlyInConfig', deviceType: 'Plug' };
|
|
86
|
+
const configDefault = { options: { devices: [cfgOnly] } };
|
|
87
|
+
const platformDefault = new SwitchBotMatterPlatform(log, configDefault, api);
|
|
88
|
+
platformDefault.discoveredDevices = [{ deviceId: 'dev1', deviceName: 'Lamp', deviceType: 'Plug' }];
|
|
89
|
+
const seenDefault = [];
|
|
90
|
+
platformDefault.createAccessoryFromDevice = async (dev) => {
|
|
91
|
+
seenDefault.push(dev);
|
|
92
|
+
return {
|
|
93
|
+
displayName: dev.deviceName ?? dev.configDeviceName,
|
|
94
|
+
uuid: api.matter.uuid.generate(dev.deviceId),
|
|
95
|
+
};
|
|
96
|
+
};
|
|
97
|
+
await platformDefault.registerMatterAccessories();
|
|
98
|
+
// Should only see discovered device (cfg1 ignored)
|
|
99
|
+
expect(seenDefault.find((d) => d.deviceId === 'cfg1')).toBeUndefined();
|
|
100
|
+
// Now opt-in to include config-only devices
|
|
101
|
+
const configOptIn = { options: { devices: [cfgOnly], allowConfigOnlyDevices: true } };
|
|
102
|
+
const platformOptIn = new SwitchBotMatterPlatform(log, configOptIn, api);
|
|
103
|
+
platformOptIn.discoveredDevices = [{ deviceId: 'dev1', deviceName: 'Lamp', deviceType: 'Plug' }];
|
|
104
|
+
const seenOptIn = [];
|
|
105
|
+
platformOptIn.createAccessoryFromDevice = async (dev) => {
|
|
106
|
+
seenOptIn.push(dev);
|
|
107
|
+
return {
|
|
108
|
+
displayName: dev.deviceName ?? dev.configDeviceName,
|
|
109
|
+
uuid: api.matter.uuid.generate(dev.deviceId),
|
|
110
|
+
};
|
|
111
|
+
};
|
|
112
|
+
await platformOptIn.registerMatterAccessories();
|
|
113
|
+
// Should include the config-only device now
|
|
114
|
+
expect(seenOptIn.find((d) => d.deviceId === 'cfg1')).toBeDefined();
|
|
115
|
+
});
|
|
116
|
+
});
|
|
117
|
+
//# sourceMappingURL=platform-matter.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.test.js","sourceRoot":"","sources":["../../src/test/platform-matter.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.unregister.test.d.ts","sourceRoot":"","sources":["../../src/test/platform-matter.unregister.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { SwitchBotMatterPlatform } from '../platform-matter.js';
|
|
3
|
+
import { makeApiStub, makeLogStub } from './helpers/platform-fixtures.js';
|
|
4
|
+
describe('removeDisabledAccessories and unregister edge cases', () => {
|
|
5
|
+
it('clears resources and unregisters accessory even with invalid timer and non-MAC deviceId', async () => {
|
|
6
|
+
// Prepare API stub
|
|
7
|
+
const unregister = vi.fn();
|
|
8
|
+
const api = makeApiStub({ unregisterPlatformAccessories: unregister });
|
|
9
|
+
const log = makeLogStub();
|
|
10
|
+
const platform = new SwitchBotMatterPlatform(log, { enableOnOffLight: false }, api);
|
|
11
|
+
// Build a fake serialized accessory matching the generated UUID for OnOff Light
|
|
12
|
+
const uuid = api.matter.uuid.generate('matter-onoff-light');
|
|
13
|
+
const accessory = { uuid, displayName: 'OnOff Light', context: { deviceId: 'NOT-A-MAC' } };
|
|
14
|
+
// Put invalid timer object into refreshTimers to ensure code handles it
|
|
15
|
+
const nid = platform.normalizeDeviceId(accessory.context.deviceId);
|
|
16
|
+
platform.refreshTimers.set(nid, null);
|
|
17
|
+
platform.accessoryInstances.set(nid, { dummy: true });
|
|
18
|
+
platform.matterAccessories.set(uuid, accessory);
|
|
19
|
+
// Spy on clearDeviceResources
|
|
20
|
+
const spyClear = vi.spyOn(platform, 'clearDeviceResources');
|
|
21
|
+
// Call removeDisabledAccessories directly
|
|
22
|
+
await platform.removeDisabledAccessories();
|
|
23
|
+
// Expect unregister was called and matterAccessories cleared
|
|
24
|
+
expect(unregister).toHaveBeenCalled();
|
|
25
|
+
expect(platform.matterAccessories.get(uuid)).toBeUndefined();
|
|
26
|
+
// clearDeviceResources should have been called with the accessory.deviceId
|
|
27
|
+
expect(spyClear).toHaveBeenCalledWith(accessory.context.deviceId);
|
|
28
|
+
});
|
|
29
|
+
});
|
|
30
|
+
//# sourceMappingURL=platform-matter.unregister.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform-matter.unregister.test.js","sourceRoot":"","sources":["../../src/test/platform-matter.unregister.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAA;AAEjD,OAAO,EAAE,uBAAuB,EAAE,MAAM,uBAAuB,CAAA;AAC/D,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,gCAAgC,CAAA;AAEzE,QAAQ,CAAC,qDAAqD,EAAE,GAAG,EAAE;IACnE,EAAE,CAAC,yFAAyF,EAAE,KAAK,IAAI,EAAE;QACvG,mBAAmB;QACnB,MAAM,UAAU,GAAG,EAAE,CAAC,EAAE,EAAE,CAAA;QAC1B,MAAM,GAAG,GAAQ,WAAW,CAAC,EAAE,6BAA6B,EAAE,UAAU,EAAE,CAAC,CAAA;QAC3E,MAAM,GAAG,GAAQ,WAAW,EAAE,CAAA;QAC9B,MAAM,QAAQ,GAAG,IAAI,uBAAuB,CAAC,GAAU,EAAE,EAAE,gBAAgB,EAAE,KAAK,EAAS,EAAE,GAAG,CAAC,CAAA;QAEjG,gFAAgF;QAChF,MAAM,IAAI,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,oBAAoB,CAAC,CAAA;QAC3D,MAAM,SAAS,GAAQ,EAAE,IAAI,EAAE,WAAW,EAAE,aAAa,EAAE,OAAO,EAAE,EAAE,QAAQ,EAAE,WAAW,EAAE,EAAE,CAAA;QAE/F,wEAAwE;QACxE,MAAM,GAAG,GAAI,QAAgB,CAAC,iBAAiB,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,CAC1E;QAAC,QAAgB,CAAC,aAAa,CAAC,GAAG,CAAC,GAAG,EAAE,IAAW,CAAC,CACrD;QAAC,QAAgB,CAAC,kBAAkB,CAAC,GAAG,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAG9D;QAAC,QAAgB,CAAC,iBAAiB,CAAC,GAAG,CAAC,IAAI,EAAE,SAAS,CAAC,CAAA;QAEzD,8BAA8B;QAC9B,MAAM,QAAQ,GAAG,EAAE,CAAC,KAAK,CAAC,QAAe,EAAE,sBAAsB,CAAC,CAAA;QAElE,0CAA0C;QAC1C,MAAO,QAAgB,CAAC,yBAAyB,EAAE,CAAA;QAEnD,6DAA6D;QAC7D,MAAM,CAAC,UAAU,CAAC,CAAC,gBAAgB,EAAE,CAAA;QACrC,MAAM,CAAE,QAAgB,CAAC,iBAAiB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,aAAa,EAAE,CAAA;QAErE,2EAA2E;QAC3E,MAAM,CAAC,QAAQ,CAAC,CAAC,oBAAoB,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAA;IACnE,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA"}
|
package/dist/utils.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import type { API, Logging } from 'homebridge';
|
|
1
2
|
import type { blindTilt, curtain, curtain3, device } from 'node-switchbot';
|
|
2
3
|
import type { devicesConfig } from './settings.js';
|
|
3
4
|
export declare enum BlindTiltMappingMode {
|
|
@@ -48,4 +49,130 @@ export declare function m2hs(m: any): number[];
|
|
|
48
49
|
* Returns undefined if nothing meaningful remains.
|
|
49
50
|
*/
|
|
50
51
|
export declare function cleanDeviceConfig(deviceConfig?: Record<string, Record<string, any>>): Record<string, Record<string, any>> | undefined;
|
|
52
|
+
/**
|
|
53
|
+
* Factory that returns a function to send OpenAPI commands using a retry wrapper.
|
|
54
|
+
*
|
|
55
|
+
* @param retryCommandFunc - bound function that calls platform.retryCommand(device, bodyChange, maxRetries, delay)
|
|
56
|
+
* @param deviceObj - the device object to operate on
|
|
57
|
+
* @param opts - optional overrides for maxRetries and delayBetweenRetries
|
|
58
|
+
* @param opts.maxRetries - override for maxRetries
|
|
59
|
+
* @param opts.delayBetweenRetries - override for delayBetweenRetries
|
|
60
|
+
*/
|
|
61
|
+
export declare function makeOpenAPISender(retryCommandFunc: any, deviceObj: any, opts?: {
|
|
62
|
+
maxRetries?: number;
|
|
63
|
+
delayBetweenRetries?: number;
|
|
64
|
+
}): (command: string, parameter?: string) => Promise<any>;
|
|
65
|
+
/**
|
|
66
|
+
* Factory that returns a function to perform BLE actions using a SwitchBotBLE client.
|
|
67
|
+
* Handles discovery retries and method invocation on the discovered device instance.
|
|
68
|
+
*
|
|
69
|
+
* @param switchBotBLE - instance of SwitchBotBLE (may be undefined)
|
|
70
|
+
* @param deviceObj - the device object (used to obtain bleModel/deviceId)
|
|
71
|
+
* @param opts - optional retry settings
|
|
72
|
+
* @param opts.bleRetries - number of BLE discovery retries
|
|
73
|
+
* @param opts.bleRetryDelay - delay between BLE retries in ms
|
|
74
|
+
*/
|
|
75
|
+
export declare function makeBLESender(switchBotBLE: any, deviceObj: any, opts?: {
|
|
76
|
+
bleRetries?: number;
|
|
77
|
+
bleRetryDelay?: number;
|
|
78
|
+
}): (methodName: string, ...args: any[]) => Promise<any>;
|
|
79
|
+
/**
|
|
80
|
+
* Decide effective connection type for a device given platform options.
|
|
81
|
+
* Mirrors the logic previously in platform-matter.
|
|
82
|
+
*/
|
|
83
|
+
export declare function chooseConnectionType(platformOptions: any, deviceObj: any): 'BLE' | 'OpenAPI';
|
|
84
|
+
/**
|
|
85
|
+
* Detect whether Matter is enabled/available on the provided Homebridge API object.
|
|
86
|
+
* This encapsulates the multi-fallback detection used across the project.
|
|
87
|
+
*/
|
|
88
|
+
/**
|
|
89
|
+
* Detect whether Matter is enabled on the provided Homebridge API object.
|
|
90
|
+
* Returns an object with an `enabled` boolean and an optional `reason` string
|
|
91
|
+
* describing which check matched (useful for diagnostics).
|
|
92
|
+
*/
|
|
93
|
+
export declare function detectMatter(apiObj: API): {
|
|
94
|
+
enabled: boolean;
|
|
95
|
+
reason?: string;
|
|
96
|
+
};
|
|
97
|
+
/**
|
|
98
|
+
* Backwards-compatible boolean wrapper for detectMatter.
|
|
99
|
+
*/
|
|
100
|
+
export declare function detectMatterEnabled(apiObj: API): boolean;
|
|
101
|
+
/**
|
|
102
|
+
* Create platform logging helpers used by both HAP and Matter platforms.
|
|
103
|
+
*
|
|
104
|
+
* getPlatformLogging may be either a synchronous string-returning function or an
|
|
105
|
+
* async function that resolves to the current platform logging setting. The
|
|
106
|
+
* returned helpers mirror the instance methods previously implemented on the
|
|
107
|
+
* HAP platform (infoLog, warnLog, errorLog, debugLog, etc.).
|
|
108
|
+
*/
|
|
109
|
+
export declare function createPlatformLogger(getPlatformLogging: () => string | Promise<string | undefined>, log: Logging): {
|
|
110
|
+
infoLog: (...args: any[]) => Promise<void>;
|
|
111
|
+
successLog: (...args: any[]) => Promise<void>;
|
|
112
|
+
debugSuccessLog: (...args: any[]) => Promise<void>;
|
|
113
|
+
warnLog: (...args: any[]) => Promise<void>;
|
|
114
|
+
debugWarnLog: (...args: any[]) => Promise<void>;
|
|
115
|
+
errorLog: (...args: any[]) => Promise<void>;
|
|
116
|
+
debugErrorLog: (...args: any[]) => Promise<void>;
|
|
117
|
+
debugLog: (...args: any[]) => Promise<void>;
|
|
118
|
+
loggingIsDebug: () => Promise<boolean>;
|
|
119
|
+
enablingPlatformLogging: () => Promise<boolean>;
|
|
120
|
+
};
|
|
121
|
+
/**
|
|
122
|
+
* Create a Platform proxy class that selects between two platform constructors
|
|
123
|
+
* (HAP vs Matter) at runtime using `detectMatter`. Returns a class suitable
|
|
124
|
+
* for passing to `api.registerPlatform`.
|
|
125
|
+
*/
|
|
126
|
+
export declare function createPlatformProxy(HAPCtor: any, MatterCtor: any): {
|
|
127
|
+
new (log: any, config: any, api: API): {
|
|
128
|
+
delegate: any;
|
|
129
|
+
readonly log: any;
|
|
130
|
+
readonly config: any;
|
|
131
|
+
readonly api: API;
|
|
132
|
+
configureAccessory(accessory: any): void;
|
|
133
|
+
configureMatterAccessory?(accessory: any): void;
|
|
134
|
+
get accessories(): any;
|
|
135
|
+
get matterAccessories(): any;
|
|
136
|
+
};
|
|
137
|
+
};
|
|
138
|
+
/**
|
|
139
|
+
* API Request Tracker - Persistent tracking of SwitchBot API calls
|
|
140
|
+
* Tracks requests per day with automatic midnight rollover
|
|
141
|
+
*/
|
|
142
|
+
export declare class ApiRequestTracker {
|
|
143
|
+
private count;
|
|
144
|
+
private date;
|
|
145
|
+
private statsFile;
|
|
146
|
+
private hourlyTimer?;
|
|
147
|
+
private log;
|
|
148
|
+
constructor(api: API, log: Logging, pluginName?: string);
|
|
149
|
+
/**
|
|
150
|
+
* Load API request statistics from persistent storage
|
|
151
|
+
*/
|
|
152
|
+
private load;
|
|
153
|
+
/**
|
|
154
|
+
* Save API request statistics to persistent storage
|
|
155
|
+
*/
|
|
156
|
+
private save;
|
|
157
|
+
/**
|
|
158
|
+
* Increment API request counter and save
|
|
159
|
+
*/
|
|
160
|
+
track(): void;
|
|
161
|
+
/**
|
|
162
|
+
* Start hourly logging of API request count
|
|
163
|
+
*/
|
|
164
|
+
startHourlyLogging(): void;
|
|
165
|
+
/**
|
|
166
|
+
* Stop hourly logging
|
|
167
|
+
*/
|
|
168
|
+
stopHourlyLogging(): void;
|
|
169
|
+
/**
|
|
170
|
+
* Get current count
|
|
171
|
+
*/
|
|
172
|
+
getCount(): number;
|
|
173
|
+
/**
|
|
174
|
+
* Get current date
|
|
175
|
+
*/
|
|
176
|
+
getDate(): string;
|
|
177
|
+
}
|
|
51
178
|
//# sourceMappingURL=utils.d.ts.map
|
package/dist/utils.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../src/utils.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../src/utils.ts"],"names":[],"mappings":"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"}
|