@switchbot/homebridge-switchbot 5.0.0-beta.6 → 5.0.0-beta.61
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 +15 -0
- package/README.md +45 -3
- package/config.schema.json +866 -13754
- package/dist/devices-hap/airpurifier.d.ts.map +1 -1
- package/dist/devices-hap/airpurifier.js +12 -6
- package/dist/devices-hap/airpurifier.js.map +1 -1
- package/dist/devices-hap/blindtilt.js +3 -3
- package/dist/devices-hap/bot.d.ts.map +1 -1
- package/dist/devices-hap/bot.js +16 -5
- package/dist/devices-hap/bot.js.map +1 -1
- package/dist/devices-hap/ceilinglight.d.ts.map +1 -1
- package/dist/devices-hap/ceilinglight.js +13 -7
- package/dist/devices-hap/ceilinglight.js.map +1 -1
- package/dist/devices-hap/colorbulb.d.ts.map +1 -1
- package/dist/devices-hap/colorbulb.js +49 -9
- package/dist/devices-hap/colorbulb.js.map +1 -1
- package/dist/devices-hap/contact.js +3 -3
- package/dist/devices-hap/curtain.js +2 -2
- package/dist/devices-hap/curtain.js.map +1 -1
- package/dist/devices-hap/device.d.ts +18 -8
- package/dist/devices-hap/device.d.ts.map +1 -1
- package/dist/devices-hap/device.js +141 -69
- package/dist/devices-hap/device.js.map +1 -1
- package/dist/devices-hap/fan.d.ts.map +1 -1
- package/dist/devices-hap/fan.js +12 -6
- package/dist/devices-hap/fan.js.map +1 -1
- package/dist/devices-hap/hub.d.ts.map +1 -1
- package/dist/devices-hap/hub.js +6 -5
- package/dist/devices-hap/hub.js.map +1 -1
- package/dist/devices-hap/humidifier.d.ts +5 -0
- package/dist/devices-hap/humidifier.d.ts.map +1 -1
- package/dist/devices-hap/humidifier.js +92 -4
- package/dist/devices-hap/humidifier.js.map +1 -1
- package/dist/devices-hap/iosensor.d.ts.map +1 -1
- package/dist/devices-hap/iosensor.js +36 -21
- package/dist/devices-hap/iosensor.js.map +1 -1
- package/dist/devices-hap/lightstrip.d.ts.map +1 -1
- package/dist/devices-hap/lightstrip.js +38 -8
- package/dist/devices-hap/lightstrip.js.map +1 -1
- package/dist/devices-hap/lock.d.ts.map +1 -1
- package/dist/devices-hap/lock.js +14 -6
- package/dist/devices-hap/lock.js.map +1 -1
- package/dist/devices-hap/meter.d.ts.map +1 -1
- package/dist/devices-hap/meter.js +6 -5
- package/dist/devices-hap/meter.js.map +1 -1
- package/dist/devices-hap/meterplus.d.ts.map +1 -1
- package/dist/devices-hap/meterplus.js +6 -5
- package/dist/devices-hap/meterplus.js.map +1 -1
- package/dist/devices-hap/meterpro.d.ts.map +1 -1
- package/dist/devices-hap/meterpro.js +7 -6
- package/dist/devices-hap/meterpro.js.map +1 -1
- package/dist/devices-hap/motion.js +3 -3
- package/dist/devices-hap/plug.d.ts.map +1 -1
- package/dist/devices-hap/plug.js +11 -6
- package/dist/devices-hap/plug.js.map +1 -1
- package/dist/devices-hap/relayswitch.js +3 -3
- package/dist/devices-hap/robotvacuumcleaner.d.ts.map +1 -1
- package/dist/devices-hap/robotvacuumcleaner.js +13 -6
- package/dist/devices-hap/robotvacuumcleaner.js.map +1 -1
- package/dist/devices-hap/waterdetector.js +3 -3
- package/dist/devices-matter/BaseMatterAccessory.d.ts +27 -0
- package/dist/devices-matter/BaseMatterAccessory.d.ts.map +1 -1
- package/dist/devices-matter/BaseMatterAccessory.js +169 -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/devices-matter/RoboticVacuumAccessory.d.ts +29 -43
- package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +1 -1
- package/dist/devices-matter/RoboticVacuumAccessory.js +287 -262
- package/dist/devices-matter/RoboticVacuumAccessory.js.map +1 -1
- package/dist/homebridge-ui/public/index.html +200 -18
- package/dist/homebridge-ui/server.js +0 -31
- package/dist/homebridge-ui/server.js.map +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +4 -7
- package/dist/index.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 +26 -15
- package/dist/platform-hap.d.ts.map +1 -1
- package/dist/platform-hap.js +333 -153
- package/dist/platform-hap.js.map +1 -1
- package/dist/platform-matter.d.ts +93 -6
- package/dist/platform-matter.d.ts.map +1 -1
- package/dist/platform-matter.js +1822 -224
- package/dist/platform-matter.js.map +1 -1
- package/dist/settings.d.ts +58 -7
- package/dist/settings.d.ts.map +1 -1
- package/dist/settings.js.map +1 -1
- package/dist/test/apiRequestTracker.test.d.ts +2 -0
- package/dist/test/apiRequestTracker.test.d.ts.map +1 -0
- package/dist/test/apiRequestTracker.test.js +392 -0
- package/dist/test/apiRequestTracker.test.js.map +1 -0
- package/dist/test/hap/device-webhook-context.test.d.ts +2 -0
- package/dist/test/hap/device-webhook-context.test.d.ts.map +1 -0
- package/dist/test/hap/device-webhook-context.test.js +128 -0
- package/dist/test/hap/device-webhook-context.test.js.map +1 -0
- package/dist/test/hap/platform-hap.logging.test.d.ts +2 -0
- package/dist/test/hap/platform-hap.logging.test.d.ts.map +1 -0
- package/dist/test/hap/platform-hap.logging.test.js +33 -0
- package/dist/test/hap/platform-hap.logging.test.js.map +1 -0
- package/dist/test/hap/platform-hap.test.d.ts +2 -0
- package/dist/test/hap/platform-hap.test.d.ts.map +1 -0
- package/dist/test/hap/platform-hap.test.js +62 -0
- package/dist/test/hap/platform-hap.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/homebridge-ui/server.test.d.ts +2 -0
- package/dist/test/homebridge-ui/server.test.d.ts.map +1 -0
- package/dist/test/homebridge-ui/server.test.js +445 -0
- package/dist/test/homebridge-ui/server.test.js.map +1 -0
- package/dist/{index.test.d.ts.map → test/index.test.d.ts.map} +1 -1
- package/dist/test/index.test.js +19 -0
- package/dist/test/index.test.js.map +1 -0
- package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts +2 -0
- package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts.map +1 -0
- package/dist/test/matter/devices-matter/baseMatterAccessory.test.js +71 -0
- package/dist/test/matter/devices-matter/baseMatterAccessory.test.js.map +1 -0
- package/dist/test/matter/devices-matter/roboticVacuumAccessory.test.d.ts +2 -0
- package/dist/test/matter/devices-matter/roboticVacuumAccessory.test.d.ts.map +1 -0
- package/dist/test/matter/devices-matter/roboticVacuumAccessory.test.js +366 -0
- package/dist/test/matter/devices-matter/roboticVacuumAccessory.test.js.map +1 -0
- package/dist/test/matter/platform-matter.additional.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.additional.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.additional.test.js +35 -0
- package/dist/test/matter/platform-matter.additional.test.js.map +1 -0
- package/dist/test/matter/platform-matter.bleparse.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.bleparse.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.bleparse.test.js +43 -0
- package/dist/test/matter/platform-matter.bleparse.test.js.map +1 -0
- package/dist/test/matter/platform-matter.cleanup.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.cleanup.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.cleanup.test.js +70 -0
- package/dist/test/matter/platform-matter.cleanup.test.js.map +1 -0
- package/dist/test/matter/platform-matter.keepstale.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.keepstale.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.keepstale.test.js +27 -0
- package/dist/test/matter/platform-matter.keepstale.test.js.map +1 -0
- package/dist/test/matter/platform-matter.logging.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.logging.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.logging.test.js +29 -0
- package/dist/test/matter/platform-matter.logging.test.js.map +1 -0
- package/dist/test/matter/platform-matter.mapping.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.mapping.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.mapping.test.js +43 -0
- package/dist/test/matter/platform-matter.mapping.test.js.map +1 -0
- package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.openapi-mapping.test.js +84 -0
- package/dist/test/matter/platform-matter.openapi-mapping.test.js.map +1 -0
- package/dist/test/matter/platform-matter.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.test.js +117 -0
- package/dist/test/matter/platform-matter.test.js.map +1 -0
- package/dist/test/matter/platform-matter.unregister.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.unregister.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.unregister.test.js +30 -0
- package/dist/test/matter/platform-matter.unregister.test.js.map +1 -0
- package/dist/test/matter/platform-matter.webhook.test.d.ts +2 -0
- package/dist/test/matter/platform-matter.webhook.test.d.ts.map +1 -0
- package/dist/test/matter/platform-matter.webhook.test.js +46 -0
- package/dist/test/matter/platform-matter.webhook.test.js.map +1 -0
- package/dist/test/utils.test.d.ts +2 -0
- package/dist/test/utils.test.d.ts.map +1 -0
- package/dist/test/utils.test.js +95 -0
- package/dist/test/utils.test.js.map +1 -0
- package/dist/test/verifyconfig.test.d.ts.map +1 -0
- package/dist/{verifyconfig.test.js → test/verifyconfig.test.js} +2 -2
- package/dist/test/verifyconfig.test.js.map +1 -0
- package/dist/utils.d.ts +204 -3
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +713 -33
- package/dist/utils.js.map +1 -1
- package/docs/assets/highlight.css +14 -0
- package/docs/assets/main.js +2 -2
- package/docs/index.html +31 -2
- package/docs/variables/default.html +1 -1
- package/package.json +15 -15
- package/src/devices-hap/airpurifier.ts +11 -6
- package/src/devices-hap/blindtilt.ts +3 -3
- package/src/devices-hap/bot.ts +15 -5
- package/src/devices-hap/ceilinglight.ts +12 -7
- package/src/devices-hap/colorbulb.ts +46 -10
- package/src/devices-hap/contact.ts +3 -3
- package/src/devices-hap/curtain.ts +2 -2
- package/src/devices-hap/device.ts +149 -70
- package/src/devices-hap/fan.ts +11 -6
- package/src/devices-hap/hub.ts +6 -5
- package/src/devices-hap/humidifier.ts +97 -4
- package/src/devices-hap/iosensor.ts +36 -21
- package/src/devices-hap/lightstrip.ts +35 -8
- package/src/devices-hap/lock.ts +13 -6
- package/src/devices-hap/meter.ts +6 -5
- package/src/devices-hap/meterplus.ts +6 -5
- package/src/devices-hap/meterpro.ts +7 -6
- package/src/devices-hap/motion.ts +3 -3
- package/src/devices-hap/plug.ts +10 -6
- package/src/devices-hap/relayswitch.ts +3 -3
- package/src/devices-hap/robotvacuumcleaner.ts +12 -6
- package/src/devices-hap/waterdetector.ts +3 -3
- package/src/devices-matter/BaseMatterAccessory.ts +176 -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/devices-matter/RoboticVacuumAccessory.ts +340 -313
- package/src/homebridge-ui/public/index.html +200 -18
- package/src/homebridge-ui/server.ts +0 -34
- package/src/index.ts +4 -7
- package/src/irdevice/irdevice.ts +74 -35
- package/src/platform-hap.ts +365 -169
- package/src/platform-matter.ts +1872 -229
- package/src/settings.ts +62 -3
- package/src/test/apiRequestTracker.test.ts +417 -0
- package/src/test/hap/device-webhook-context.test.ts +136 -0
- package/src/test/hap/platform-hap.logging.test.ts +36 -0
- package/src/test/hap/platform-hap.test.ts +70 -0
- package/src/test/helpers/platform-fixtures.ts +33 -0
- package/src/test/homebridge-ui/server.test.ts +486 -0
- package/src/test/index.test.ts +24 -0
- package/src/test/matter/devices-matter/baseMatterAccessory.test.ts +88 -0
- package/src/test/matter/devices-matter/roboticVacuumAccessory.test.ts +453 -0
- package/src/test/matter/platform-matter.additional.test.ts +44 -0
- package/src/test/matter/platform-matter.bleparse.test.ts +47 -0
- package/src/test/matter/platform-matter.cleanup.test.ts +86 -0
- package/src/test/matter/platform-matter.keepstale.test.ts +37 -0
- package/src/test/matter/platform-matter.logging.test.ts +33 -0
- package/src/test/matter/platform-matter.mapping.test.ts +57 -0
- package/src/test/matter/platform-matter.openapi-mapping.test.ts +109 -0
- package/src/test/matter/platform-matter.test.ts +144 -0
- package/src/test/matter/platform-matter.unregister.test.ts +39 -0
- package/src/test/matter/platform-matter.webhook.test.ts +54 -0
- package/src/test/utils.test.ts +96 -0
- package/src/{verifyconfig.test.ts → test/verifyconfig.test.ts} +12 -11
- package/src/utils.ts +777 -36
- package/dist/index.test.js +0 -14
- package/dist/index.test.js.map +0 -1
- package/dist/verifyconfig.test.d.ts.map +0 -1
- package/dist/verifyconfig.test.js.map +0 -1
- package/src/index.test.ts +0 -19
- /package/dist/{index.test.d.ts → test/index.test.d.ts} +0 -0
- /package/dist/{verifyconfig.test.d.ts → test/verifyconfig.test.d.ts} +0 -0
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/* eslint-disable import/first */
|
|
2
|
+
import { describe, expect, it, vi } from 'vitest'
|
|
3
|
+
// Mock modules used by the HAP platform constructor to avoid requiring a real Homebridge API
|
|
4
|
+
vi.mock('fakegato-history', () => ({ default: () => ({}) }))
|
|
5
|
+
vi.mock('homebridge-lib/EveHomeKitTypes', () => ({ EveHomeKitTypes: class { constructor() {} } }))
|
|
6
|
+
|
|
7
|
+
import { SwitchBotHAPPlatform } from '../../platform-hap.js'
|
|
8
|
+
import { makeLogStub } from '../helpers/platform-fixtures.js'
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* High-level smoke tests for the HAP platform to ensure it initializes,
|
|
12
|
+
* subscribes to lifecycle events, and handles cached accessories.
|
|
13
|
+
*/
|
|
14
|
+
describe('platform-hap (smoke)', () => {
|
|
15
|
+
it('initializes and subscribes to didFinishLaunching', async () => {
|
|
16
|
+
const api: any = { on: vi.fn() }
|
|
17
|
+
const log: any = makeLogStub()
|
|
18
|
+
|
|
19
|
+
// Enable debug so debug* logs print via our shared logger
|
|
20
|
+
// Construct the platform (smoke)
|
|
21
|
+
new SwitchBotHAPPlatform(log as any, {
|
|
22
|
+
name: 'SwitchBot',
|
|
23
|
+
credentials: {},
|
|
24
|
+
options: { logging: 'debug' },
|
|
25
|
+
devices: [],
|
|
26
|
+
} as any, api)
|
|
27
|
+
|
|
28
|
+
// Should register didFinishLaunching handler
|
|
29
|
+
expect(api.on).toHaveBeenCalled()
|
|
30
|
+
const calledWithDL = (api.on as any).mock.calls.some((args: any[]) => args[0] === 'didFinishLaunching' && typeof args[1] === 'function')
|
|
31
|
+
expect(calledWithDL).toBe(true)
|
|
32
|
+
|
|
33
|
+
// Should log effective platform logging at startup
|
|
34
|
+
const debugCalls = (log.debug as any).mock.calls as Array<string[]>
|
|
35
|
+
const hasStartupLine = debugCalls.some(args => String(args[0]).includes('[SwitchBot HAP] effective platformLogging='))
|
|
36
|
+
expect(hasStartupLine).toBe(true)
|
|
37
|
+
|
|
38
|
+
// Missing credentials results in a debug error log
|
|
39
|
+
// Allow async platform logger to flush
|
|
40
|
+
await new Promise(resolve => setTimeout(resolve, 0))
|
|
41
|
+
const errorCalls = (log.error as any).mock.calls as Array<string[]>
|
|
42
|
+
const hasMissingCreds = errorCalls.some(args => String(args[0]).includes('Missing SwitchBot API credentials'))
|
|
43
|
+
expect(hasMissingCreds).toBe(true)
|
|
44
|
+
})
|
|
45
|
+
|
|
46
|
+
it('adds cached accessories via configureAccessory', async () => {
|
|
47
|
+
const api: any = { on: vi.fn() }
|
|
48
|
+
const log: any = makeLogStub()
|
|
49
|
+
|
|
50
|
+
const platform = new SwitchBotHAPPlatform(log as any, {
|
|
51
|
+
name: 'SwitchBot',
|
|
52
|
+
credentials: {},
|
|
53
|
+
options: { logging: 'debug' },
|
|
54
|
+
devices: [],
|
|
55
|
+
} as any, api)
|
|
56
|
+
|
|
57
|
+
const accessory: any = { displayName: 'Cached Device' }
|
|
58
|
+
await (platform as any).configureAccessory(accessory)
|
|
59
|
+
|
|
60
|
+
// Should be stored in platform.accessories
|
|
61
|
+
expect(Array.isArray((platform as any).accessories)).toBe(true)
|
|
62
|
+
expect((platform as any).accessories.length).toBe(1)
|
|
63
|
+
|
|
64
|
+
// And should log the cache load message including device name
|
|
65
|
+
await new Promise(resolve => setTimeout(resolve, 0))
|
|
66
|
+
const infoCalls = (log.info as any).mock.calls as Array<string[]>
|
|
67
|
+
const hasLine = infoCalls.some(args => String(args[0]).includes('Loading accessory from cache: Cached Device'))
|
|
68
|
+
expect(hasLine).toBe(true)
|
|
69
|
+
})
|
|
70
|
+
})
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { vi } from 'vitest'
|
|
2
|
+
|
|
3
|
+
// Shared test helpers for platform-matter tests
|
|
4
|
+
export function makeApiStub(matterProps: Record<string, any> = {}) {
|
|
5
|
+
const handlers: Record<string, (...args: any[]) => any> = {}
|
|
6
|
+
|
|
7
|
+
const api: any = {
|
|
8
|
+
matter: {
|
|
9
|
+
uuid: { generate: (s: string) => `uuid-${s}` },
|
|
10
|
+
registerPlatformAccessories: matterProps.registerPlatformAccessories ?? vi.fn(),
|
|
11
|
+
unregisterPlatformAccessories: matterProps.unregisterPlatformAccessories ?? vi.fn(),
|
|
12
|
+
updateAccessoryState: matterProps.updateAccessoryState ?? vi.fn(),
|
|
13
|
+
clusterNames: matterProps.clusterNames ?? {},
|
|
14
|
+
deviceTypes: matterProps.deviceTypes ?? {},
|
|
15
|
+
},
|
|
16
|
+
isMatterAvailable: matterProps.isMatterAvailable ?? (() => true),
|
|
17
|
+
isMatterEnabled: matterProps.isMatterEnabled ?? (() => true),
|
|
18
|
+
on: (ev: string, fn: (...args: any[]) => any) => { handlers[ev] = fn },
|
|
19
|
+
_handlers: handlers,
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
return api
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export function makeLogStub() {
|
|
26
|
+
return {
|
|
27
|
+
info: vi.fn(),
|
|
28
|
+
warn: vi.fn(),
|
|
29
|
+
debug: vi.fn(),
|
|
30
|
+
error: vi.fn(),
|
|
31
|
+
success: vi.fn(),
|
|
32
|
+
}
|
|
33
|
+
}
|
|
@@ -0,0 +1,486 @@
|
|
|
1
|
+
import { existsSync, mkdirSync, readdirSync, readFileSync, rmdirSync, statSync, unlinkSync, writeFileSync } from 'node:fs'
|
|
2
|
+
import { tmpdir } from 'node:os'
|
|
3
|
+
import { join } from 'node:path'
|
|
4
|
+
|
|
5
|
+
import { describe, expect, it } from 'vitest'
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Test suite for the homebridge-ui server handler logic
|
|
9
|
+
*
|
|
10
|
+
* These tests validate the cached accessory file reading logic that powers
|
|
11
|
+
* the getCachedAccessories and getCachedMatterAccessories handlers.
|
|
12
|
+
*
|
|
13
|
+
* Note: These are integration tests that test the file system logic rather than
|
|
14
|
+
* the UI server infrastructure itself, since HomebridgePluginUiServer is designed
|
|
15
|
+
* to run as a standalone process with IPC communication.
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
// Helper to create isolated test environment
|
|
19
|
+
function createTestEnvironment() {
|
|
20
|
+
const testId = Math.random().toString(36).substring(7)
|
|
21
|
+
const testDir = join(tmpdir(), `switchbot-ui-test-${testId}`)
|
|
22
|
+
const accessoriesDir = join(testDir, 'accessories')
|
|
23
|
+
|
|
24
|
+
// Create test directories
|
|
25
|
+
if (!existsSync(testDir)) {
|
|
26
|
+
mkdirSync(testDir, { recursive: true })
|
|
27
|
+
}
|
|
28
|
+
if (!existsSync(accessoriesDir)) {
|
|
29
|
+
mkdirSync(accessoriesDir, { recursive: true })
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
return { testDir, accessoriesDir }
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// Cleanup helper
|
|
36
|
+
function cleanup(testDir: string) {
|
|
37
|
+
try {
|
|
38
|
+
if (existsSync(testDir)) {
|
|
39
|
+
const removeRecursive = (dir: string) => {
|
|
40
|
+
if (!existsSync(dir)) {
|
|
41
|
+
return
|
|
42
|
+
}
|
|
43
|
+
const files = readdirSync(dir)
|
|
44
|
+
for (const file of files) {
|
|
45
|
+
const filePath = join(dir, file)
|
|
46
|
+
try {
|
|
47
|
+
const stat = statSync(filePath)
|
|
48
|
+
if (stat.isDirectory()) {
|
|
49
|
+
removeRecursive(filePath)
|
|
50
|
+
} else {
|
|
51
|
+
unlinkSync(filePath)
|
|
52
|
+
}
|
|
53
|
+
} catch {
|
|
54
|
+
// ignore
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
try {
|
|
58
|
+
rmdirSync(dir)
|
|
59
|
+
} catch {
|
|
60
|
+
// ignore
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
removeRecursive(testDir)
|
|
64
|
+
}
|
|
65
|
+
} catch {
|
|
66
|
+
// ignore
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* Re-implementation of the getCachedAccessories handler logic for testing
|
|
72
|
+
* This mirrors the actual implementation in server.ts
|
|
73
|
+
*/
|
|
74
|
+
function getCachedAccessories(homebridgeStoragePath: string): any[] {
|
|
75
|
+
try {
|
|
76
|
+
const pluginNames = ['@switchbot/homebridge-switchbot', 'homebridge-switchbot']
|
|
77
|
+
const devicesToReturn = []
|
|
78
|
+
|
|
79
|
+
const accFile = `${homebridgeStoragePath}/accessories/cachedAccessories`
|
|
80
|
+
|
|
81
|
+
if (existsSync(accFile)) {
|
|
82
|
+
const cachedAccessories: any[] = JSON.parse(readFileSync(accFile, 'utf8'))
|
|
83
|
+
|
|
84
|
+
cachedAccessories.forEach((entry: any) => {
|
|
85
|
+
const pluginName = entry.plugin || entry?.accessory?.plugin || entry?.accessory?.pluginName
|
|
86
|
+
const acc = entry.accessory ?? entry
|
|
87
|
+
if (pluginNames.includes(pluginName)) {
|
|
88
|
+
devicesToReturn.push(acc as never)
|
|
89
|
+
}
|
|
90
|
+
})
|
|
91
|
+
}
|
|
92
|
+
return devicesToReturn
|
|
93
|
+
} catch {
|
|
94
|
+
return []
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* Re-implementation of the getCachedMatterAccessories handler logic for testing
|
|
100
|
+
* This mirrors the actual implementation in server.ts
|
|
101
|
+
*/
|
|
102
|
+
function getCachedMatterAccessories(homebridgeStoragePath: string): any[] {
|
|
103
|
+
try {
|
|
104
|
+
const pluginNames = ['@switchbot/homebridge-switchbot', 'homebridge-switchbot']
|
|
105
|
+
const devicesToReturn: any[] = []
|
|
106
|
+
|
|
107
|
+
const accFile = `${homebridgeStoragePath}/accessories/cachedAccessories`
|
|
108
|
+
const matterFile = `${homebridgeStoragePath}/accessories/cachedMatterAccessories`
|
|
109
|
+
|
|
110
|
+
const readAndCollect = (filePath: string) => {
|
|
111
|
+
if (!existsSync(filePath)) {
|
|
112
|
+
return
|
|
113
|
+
}
|
|
114
|
+
try {
|
|
115
|
+
const parsed: any[] = JSON.parse(readFileSync(filePath, 'utf8'))
|
|
116
|
+
parsed.forEach((entry: any) => {
|
|
117
|
+
const pluginName = entry.plugin || entry?.accessory?.plugin || entry?.accessory?.pluginName
|
|
118
|
+
const acc = entry.accessory ?? entry
|
|
119
|
+
if (pluginNames.includes(pluginName)) {
|
|
120
|
+
devicesToReturn.push(acc as never)
|
|
121
|
+
}
|
|
122
|
+
})
|
|
123
|
+
} catch {
|
|
124
|
+
// ignore parse errors for a single file
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
readAndCollect(accFile)
|
|
129
|
+
readAndCollect(matterFile)
|
|
130
|
+
|
|
131
|
+
return devicesToReturn
|
|
132
|
+
} catch {
|
|
133
|
+
return []
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
describe('homebridge-ui server handler logic', () => {
|
|
138
|
+
describe('getCachedAccessories', () => {
|
|
139
|
+
it('should return empty array when no cached accessories file exists', () => {
|
|
140
|
+
const { testDir } = createTestEnvironment()
|
|
141
|
+
try {
|
|
142
|
+
const result = getCachedAccessories(testDir)
|
|
143
|
+
expect(result).toEqual([])
|
|
144
|
+
} finally {
|
|
145
|
+
cleanup(testDir)
|
|
146
|
+
}
|
|
147
|
+
})
|
|
148
|
+
|
|
149
|
+
it('should return accessories with scoped plugin name', () => {
|
|
150
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
151
|
+
try {
|
|
152
|
+
const cachedAccessories = [
|
|
153
|
+
{
|
|
154
|
+
plugin: '@switchbot/homebridge-switchbot',
|
|
155
|
+
accessory: {
|
|
156
|
+
displayName: 'Test Bot',
|
|
157
|
+
UUID: 'test-uuid-1',
|
|
158
|
+
services: [],
|
|
159
|
+
},
|
|
160
|
+
},
|
|
161
|
+
{
|
|
162
|
+
plugin: 'homebridge-other-plugin',
|
|
163
|
+
accessory: {
|
|
164
|
+
displayName: 'Other Device',
|
|
165
|
+
UUID: 'test-uuid-2',
|
|
166
|
+
services: [],
|
|
167
|
+
},
|
|
168
|
+
},
|
|
169
|
+
]
|
|
170
|
+
|
|
171
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
172
|
+
writeFileSync(accFile, JSON.stringify(cachedAccessories), 'utf8')
|
|
173
|
+
|
|
174
|
+
const result = getCachedAccessories(testDir)
|
|
175
|
+
|
|
176
|
+
expect(result).toHaveLength(1)
|
|
177
|
+
expect(result[0].displayName).toBe('Test Bot')
|
|
178
|
+
expect(result[0].UUID).toBe('test-uuid-1')
|
|
179
|
+
} finally {
|
|
180
|
+
cleanup(testDir)
|
|
181
|
+
}
|
|
182
|
+
})
|
|
183
|
+
|
|
184
|
+
it('should return accessories with unscoped plugin name', () => {
|
|
185
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
186
|
+
try {
|
|
187
|
+
const cachedAccessories = [
|
|
188
|
+
{
|
|
189
|
+
plugin: 'homebridge-switchbot',
|
|
190
|
+
accessory: {
|
|
191
|
+
displayName: 'Test Curtain',
|
|
192
|
+
UUID: 'test-uuid-3',
|
|
193
|
+
services: [],
|
|
194
|
+
},
|
|
195
|
+
},
|
|
196
|
+
]
|
|
197
|
+
|
|
198
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
199
|
+
writeFileSync(accFile, JSON.stringify(cachedAccessories), 'utf8')
|
|
200
|
+
|
|
201
|
+
const result = getCachedAccessories(testDir)
|
|
202
|
+
|
|
203
|
+
expect(result).toHaveLength(1)
|
|
204
|
+
expect(result[0].displayName).toBe('Test Curtain')
|
|
205
|
+
} finally {
|
|
206
|
+
cleanup(testDir)
|
|
207
|
+
}
|
|
208
|
+
})
|
|
209
|
+
|
|
210
|
+
it('should handle entries with plugin name in accessory.plugin', () => {
|
|
211
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
212
|
+
try {
|
|
213
|
+
const cachedAccessories = [
|
|
214
|
+
{
|
|
215
|
+
accessory: {
|
|
216
|
+
plugin: '@switchbot/homebridge-switchbot',
|
|
217
|
+
displayName: 'Test Contact Sensor',
|
|
218
|
+
UUID: 'test-uuid-4',
|
|
219
|
+
services: [],
|
|
220
|
+
},
|
|
221
|
+
},
|
|
222
|
+
]
|
|
223
|
+
|
|
224
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
225
|
+
writeFileSync(accFile, JSON.stringify(cachedAccessories), 'utf8')
|
|
226
|
+
|
|
227
|
+
const result = getCachedAccessories(testDir)
|
|
228
|
+
|
|
229
|
+
expect(result).toHaveLength(1)
|
|
230
|
+
expect(result[0].displayName).toBe('Test Contact Sensor')
|
|
231
|
+
} finally {
|
|
232
|
+
cleanup(testDir)
|
|
233
|
+
}
|
|
234
|
+
})
|
|
235
|
+
|
|
236
|
+
it('should handle entries with plugin name in accessory.pluginName', () => {
|
|
237
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
238
|
+
try {
|
|
239
|
+
const cachedAccessories = [
|
|
240
|
+
{
|
|
241
|
+
accessory: {
|
|
242
|
+
pluginName: '@switchbot/homebridge-switchbot',
|
|
243
|
+
displayName: 'Test Motion Sensor',
|
|
244
|
+
UUID: 'test-uuid-5',
|
|
245
|
+
services: [],
|
|
246
|
+
},
|
|
247
|
+
},
|
|
248
|
+
]
|
|
249
|
+
|
|
250
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
251
|
+
writeFileSync(accFile, JSON.stringify(cachedAccessories), 'utf8')
|
|
252
|
+
|
|
253
|
+
const result = getCachedAccessories(testDir)
|
|
254
|
+
|
|
255
|
+
expect(result).toHaveLength(1)
|
|
256
|
+
expect(result[0].displayName).toBe('Test Motion Sensor')
|
|
257
|
+
} finally {
|
|
258
|
+
cleanup(testDir)
|
|
259
|
+
}
|
|
260
|
+
})
|
|
261
|
+
|
|
262
|
+
it('should filter out accessories from other plugins', () => {
|
|
263
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
264
|
+
try {
|
|
265
|
+
const cachedAccessories = [
|
|
266
|
+
{
|
|
267
|
+
plugin: '@switchbot/homebridge-switchbot',
|
|
268
|
+
accessory: { displayName: 'SwitchBot Device 1', UUID: 'uuid-1', services: [] },
|
|
269
|
+
},
|
|
270
|
+
{
|
|
271
|
+
plugin: 'homebridge-other',
|
|
272
|
+
accessory: { displayName: 'Other Device', UUID: 'uuid-2', services: [] },
|
|
273
|
+
},
|
|
274
|
+
{
|
|
275
|
+
plugin: 'homebridge-switchbot',
|
|
276
|
+
accessory: { displayName: 'SwitchBot Device 2', UUID: 'uuid-3', services: [] },
|
|
277
|
+
},
|
|
278
|
+
{
|
|
279
|
+
plugin: 'homebridge-another',
|
|
280
|
+
accessory: { displayName: 'Another Device', UUID: 'uuid-4', services: [] },
|
|
281
|
+
},
|
|
282
|
+
]
|
|
283
|
+
|
|
284
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
285
|
+
writeFileSync(accFile, JSON.stringify(cachedAccessories), 'utf8')
|
|
286
|
+
|
|
287
|
+
const result = getCachedAccessories(testDir)
|
|
288
|
+
|
|
289
|
+
expect(result).toHaveLength(2)
|
|
290
|
+
expect(result[0].displayName).toBe('SwitchBot Device 1')
|
|
291
|
+
expect(result[1].displayName).toBe('SwitchBot Device 2')
|
|
292
|
+
} finally {
|
|
293
|
+
cleanup(testDir)
|
|
294
|
+
}
|
|
295
|
+
})
|
|
296
|
+
|
|
297
|
+
it('should return empty array on malformed JSON', () => {
|
|
298
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
299
|
+
try {
|
|
300
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
301
|
+
writeFileSync(accFile, '{ invalid json }', 'utf8')
|
|
302
|
+
|
|
303
|
+
const result = getCachedAccessories(testDir)
|
|
304
|
+
|
|
305
|
+
expect(result).toEqual([])
|
|
306
|
+
} finally {
|
|
307
|
+
cleanup(testDir)
|
|
308
|
+
}
|
|
309
|
+
})
|
|
310
|
+
})
|
|
311
|
+
|
|
312
|
+
describe('getCachedMatterAccessories', () => {
|
|
313
|
+
it('should return empty array when no cached files exist', () => {
|
|
314
|
+
const { testDir } = createTestEnvironment()
|
|
315
|
+
try {
|
|
316
|
+
const result = getCachedMatterAccessories(testDir)
|
|
317
|
+
expect(result).toEqual([])
|
|
318
|
+
} finally {
|
|
319
|
+
cleanup(testDir)
|
|
320
|
+
}
|
|
321
|
+
})
|
|
322
|
+
|
|
323
|
+
it('should read from cachedAccessories file', () => {
|
|
324
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
325
|
+
try {
|
|
326
|
+
const cachedAccessories = [
|
|
327
|
+
{
|
|
328
|
+
plugin: '@switchbot/homebridge-switchbot',
|
|
329
|
+
accessory: {
|
|
330
|
+
displayName: 'HAP Device',
|
|
331
|
+
UUID: 'hap-uuid-1',
|
|
332
|
+
services: [],
|
|
333
|
+
},
|
|
334
|
+
},
|
|
335
|
+
]
|
|
336
|
+
|
|
337
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
338
|
+
writeFileSync(accFile, JSON.stringify(cachedAccessories), 'utf8')
|
|
339
|
+
|
|
340
|
+
const result = getCachedMatterAccessories(testDir)
|
|
341
|
+
|
|
342
|
+
expect(result).toHaveLength(1)
|
|
343
|
+
expect(result[0].displayName).toBe('HAP Device')
|
|
344
|
+
} finally {
|
|
345
|
+
cleanup(testDir)
|
|
346
|
+
}
|
|
347
|
+
})
|
|
348
|
+
|
|
349
|
+
it('should read from cachedMatterAccessories file', () => {
|
|
350
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
351
|
+
try {
|
|
352
|
+
const matterAccessories = [
|
|
353
|
+
{
|
|
354
|
+
plugin: '@switchbot/homebridge-switchbot',
|
|
355
|
+
accessory: {
|
|
356
|
+
displayName: 'Matter Device',
|
|
357
|
+
UUID: 'matter-uuid-1',
|
|
358
|
+
services: [],
|
|
359
|
+
},
|
|
360
|
+
},
|
|
361
|
+
]
|
|
362
|
+
|
|
363
|
+
const matterFile = join(accessoriesDir, 'cachedMatterAccessories')
|
|
364
|
+
writeFileSync(matterFile, JSON.stringify(matterAccessories), 'utf8')
|
|
365
|
+
|
|
366
|
+
const result = getCachedMatterAccessories(testDir)
|
|
367
|
+
|
|
368
|
+
expect(result).toHaveLength(1)
|
|
369
|
+
expect(result[0].displayName).toBe('Matter Device')
|
|
370
|
+
} finally {
|
|
371
|
+
cleanup(testDir)
|
|
372
|
+
}
|
|
373
|
+
})
|
|
374
|
+
|
|
375
|
+
it('should combine accessories from both files', () => {
|
|
376
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
377
|
+
try {
|
|
378
|
+
const cachedAccessories = [
|
|
379
|
+
{
|
|
380
|
+
plugin: '@switchbot/homebridge-switchbot',
|
|
381
|
+
accessory: { displayName: 'HAP Device', UUID: 'hap-uuid', services: [] },
|
|
382
|
+
},
|
|
383
|
+
]
|
|
384
|
+
const matterAccessories = [
|
|
385
|
+
{
|
|
386
|
+
plugin: 'homebridge-switchbot',
|
|
387
|
+
accessory: { displayName: 'Matter Device', UUID: 'matter-uuid', services: [] },
|
|
388
|
+
},
|
|
389
|
+
]
|
|
390
|
+
|
|
391
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
392
|
+
const matterFile = join(accessoriesDir, 'cachedMatterAccessories')
|
|
393
|
+
writeFileSync(accFile, JSON.stringify(cachedAccessories), 'utf8')
|
|
394
|
+
writeFileSync(matterFile, JSON.stringify(matterAccessories), 'utf8')
|
|
395
|
+
|
|
396
|
+
const result = getCachedMatterAccessories(testDir)
|
|
397
|
+
|
|
398
|
+
expect(result).toHaveLength(2)
|
|
399
|
+
expect(result[0].displayName).toBe('HAP Device')
|
|
400
|
+
expect(result[1].displayName).toBe('Matter Device')
|
|
401
|
+
} finally {
|
|
402
|
+
cleanup(testDir)
|
|
403
|
+
}
|
|
404
|
+
})
|
|
405
|
+
|
|
406
|
+
it('should filter out accessories from other plugins across both files', () => {
|
|
407
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
408
|
+
try {
|
|
409
|
+
const cachedAccessories = [
|
|
410
|
+
{
|
|
411
|
+
plugin: '@switchbot/homebridge-switchbot',
|
|
412
|
+
accessory: { displayName: 'SwitchBot HAP 1', UUID: 'hap-1', services: [] },
|
|
413
|
+
},
|
|
414
|
+
{
|
|
415
|
+
plugin: 'homebridge-other',
|
|
416
|
+
accessory: { displayName: 'Other HAP', UUID: 'hap-2', services: [] },
|
|
417
|
+
},
|
|
418
|
+
]
|
|
419
|
+
const matterAccessories = [
|
|
420
|
+
{
|
|
421
|
+
plugin: 'homebridge-switchbot',
|
|
422
|
+
accessory: { displayName: 'SwitchBot Matter 1', UUID: 'matter-1', services: [] },
|
|
423
|
+
},
|
|
424
|
+
{
|
|
425
|
+
plugin: 'homebridge-another',
|
|
426
|
+
accessory: { displayName: 'Another Matter', UUID: 'matter-2', services: [] },
|
|
427
|
+
},
|
|
428
|
+
]
|
|
429
|
+
|
|
430
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
431
|
+
const matterFile = join(accessoriesDir, 'cachedMatterAccessories')
|
|
432
|
+
writeFileSync(accFile, JSON.stringify(cachedAccessories), 'utf8')
|
|
433
|
+
writeFileSync(matterFile, JSON.stringify(matterAccessories), 'utf8')
|
|
434
|
+
|
|
435
|
+
const result = getCachedMatterAccessories(testDir)
|
|
436
|
+
|
|
437
|
+
expect(result).toHaveLength(2)
|
|
438
|
+
expect(result[0].displayName).toBe('SwitchBot HAP 1')
|
|
439
|
+
expect(result[1].displayName).toBe('SwitchBot Matter 1')
|
|
440
|
+
} finally {
|
|
441
|
+
cleanup(testDir)
|
|
442
|
+
}
|
|
443
|
+
})
|
|
444
|
+
|
|
445
|
+
it('should handle malformed JSON in one file gracefully', () => {
|
|
446
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
447
|
+
try {
|
|
448
|
+
const matterAccessories = [
|
|
449
|
+
{
|
|
450
|
+
plugin: '@switchbot/homebridge-switchbot',
|
|
451
|
+
accessory: { displayName: 'Valid Matter', UUID: 'valid', services: [] },
|
|
452
|
+
},
|
|
453
|
+
]
|
|
454
|
+
|
|
455
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
456
|
+
const matterFile = join(accessoriesDir, 'cachedMatterAccessories')
|
|
457
|
+
writeFileSync(accFile, '{ invalid json }', 'utf8')
|
|
458
|
+
writeFileSync(matterFile, JSON.stringify(matterAccessories), 'utf8')
|
|
459
|
+
|
|
460
|
+
const result = getCachedMatterAccessories(testDir)
|
|
461
|
+
|
|
462
|
+
// Should still return the valid Matter accessory
|
|
463
|
+
expect(result).toHaveLength(1)
|
|
464
|
+
expect(result[0].displayName).toBe('Valid Matter')
|
|
465
|
+
} finally {
|
|
466
|
+
cleanup(testDir)
|
|
467
|
+
}
|
|
468
|
+
})
|
|
469
|
+
|
|
470
|
+
it('should return empty array if both files have malformed JSON', () => {
|
|
471
|
+
const { testDir, accessoriesDir } = createTestEnvironment()
|
|
472
|
+
try {
|
|
473
|
+
const accFile = join(accessoriesDir, 'cachedAccessories')
|
|
474
|
+
const matterFile = join(accessoriesDir, 'cachedMatterAccessories')
|
|
475
|
+
writeFileSync(accFile, '{ bad json', 'utf8')
|
|
476
|
+
writeFileSync(matterFile, 'also bad }', 'utf8')
|
|
477
|
+
|
|
478
|
+
const result = getCachedMatterAccessories(testDir)
|
|
479
|
+
|
|
480
|
+
expect(result).toEqual([])
|
|
481
|
+
} finally {
|
|
482
|
+
cleanup(testDir)
|
|
483
|
+
}
|
|
484
|
+
})
|
|
485
|
+
})
|
|
486
|
+
})
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import type { API } from 'homebridge'
|
|
2
|
+
|
|
3
|
+
import { describe, expect, it, vi } from 'vitest'
|
|
4
|
+
|
|
5
|
+
import registerPlatform from '../index.js'
|
|
6
|
+
import { PLATFORM_NAME, PLUGIN_NAME } from '../settings.js'
|
|
7
|
+
|
|
8
|
+
describe('index.ts', () => {
|
|
9
|
+
it('should register the platform with homebridge', () => {
|
|
10
|
+
const api = {
|
|
11
|
+
registerPlatform: vi.fn(),
|
|
12
|
+
} as unknown as API
|
|
13
|
+
|
|
14
|
+
registerPlatform(api)
|
|
15
|
+
|
|
16
|
+
// The platform registration now uses a runtime proxy/delegate constructor so
|
|
17
|
+
// assert the call happened and the third argument is a constructor function.
|
|
18
|
+
expect(api.registerPlatform).toHaveBeenCalled()
|
|
19
|
+
const callArgs = (api.registerPlatform as any).mock.calls[0]
|
|
20
|
+
expect(callArgs[0]).toBe(PLUGIN_NAME)
|
|
21
|
+
expect(callArgs[1]).toBe(PLATFORM_NAME)
|
|
22
|
+
expect(typeof callArgs[2]).toBe('function')
|
|
23
|
+
})
|
|
24
|
+
})
|